| 
									
										
										
										
											2014-01-07 22:17:13 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (C) 2013 Huawei Ltd. | 
					
						
							|  |  |  |  * Author: Jiang Liu <liuj97@gmail.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Based on arch/arm/kernel/jump_label.c | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License version 2 as | 
					
						
							|  |  |  |  * published by the Free Software Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/jump_label.h>
 | 
					
						
							|  |  |  | #include <asm/insn.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAVE_JUMP_LABEL
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 15:44:18 +00:00
										 |  |  | void arch_jump_label_transform(struct jump_entry *entry, | 
					
						
							|  |  |  | 			       enum jump_label_type type) | 
					
						
							| 
									
										
										
										
											2014-01-07 22:17:13 +08:00
										 |  |  | { | 
					
						
							|  |  |  | 	void *addr = (void *)entry->code; | 
					
						
							|  |  |  | 	u32 insn; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (type == JUMP_LABEL_ENABLE) { | 
					
						
							|  |  |  | 		insn = aarch64_insn_gen_branch_imm(entry->code, | 
					
						
							|  |  |  | 						   entry->target, | 
					
						
							|  |  |  | 						   AARCH64_INSN_BRANCH_NOLINK); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		insn = aarch64_insn_gen_nop(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 15:44:18 +00:00
										 |  |  | 	aarch64_insn_patch_text(&addr, &insn, 1); | 
					
						
							| 
									
										
										
										
											2014-01-07 22:17:13 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void arch_jump_label_transform_static(struct jump_entry *entry, | 
					
						
							|  |  |  | 				      enum jump_label_type type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-11-25 15:44:18 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * We use the architected A64 NOP in arch_static_branch, so there's no | 
					
						
							|  |  |  | 	 * need to patch an identical A64 NOP over the top of it here. The core | 
					
						
							|  |  |  | 	 * will call arch_jump_label_transform from a module notifier if the | 
					
						
							|  |  |  | 	 * NOP needs to be replaced by a branch. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2014-01-07 22:17:13 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif	/* HAVE_JUMP_LABEL */
 |