mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-25 17:57:15 +00:00 
			
		
		
		
	 400d34944c
			
		
	
	
		400d34944c
		
	
	
	
	
		
			
			Some Xen hypercalls accept an array of operations to work on. In general this is because its more efficient for the hypercall to the work all at once rather than as separate hypercalls (even batched as a multicall). This patch adds a mechanism (xen_mc_extend_args()) to allocate more argument space to the last-issued multicall, in order to extend its argument list. The user of this mechanism is xen/mmu.c, which uses it to extend the args array of mmu_update. This is particularly valuable when doing the update for a large mprotect, which goes via ptep_modify_prot_commit(), but it also manages to batch updates to pgd/pmds as well. Signed-off-by: Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> Acked-by: Linus Torvalds <torvalds@linux-foundation.org> Acked-by: Hugh Dickins <hugh@veritas.com> Signed-off-by: Ingo Molnar <mingo@elte.hu>
		
			
				
	
	
		
			61 lines
		
	
	
		
			1.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			61 lines
		
	
	
		
			1.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef _XEN_MULTICALLS_H
 | |
| #define _XEN_MULTICALLS_H
 | |
| 
 | |
| #include "xen-ops.h"
 | |
| 
 | |
| /* Multicalls */
 | |
| struct multicall_space
 | |
| {
 | |
| 	struct multicall_entry *mc;
 | |
| 	void *args;
 | |
| };
 | |
| 
 | |
| /* Allocate room for a multicall and its args */
 | |
| struct multicall_space __xen_mc_entry(size_t args);
 | |
| 
 | |
| DECLARE_PER_CPU(unsigned long, xen_mc_irq_flags);
 | |
| 
 | |
| /* Call to start a batch of multiple __xen_mc_entry()s.  Must be
 | |
|    paired with xen_mc_issue() */
 | |
| static inline void xen_mc_batch(void)
 | |
| {
 | |
| 	/* need to disable interrupts until this entry is complete */
 | |
| 	local_irq_save(__get_cpu_var(xen_mc_irq_flags));
 | |
| }
 | |
| 
 | |
| static inline struct multicall_space xen_mc_entry(size_t args)
 | |
| {
 | |
| 	xen_mc_batch();
 | |
| 	return __xen_mc_entry(args);
 | |
| }
 | |
| 
 | |
| /* Flush all pending multicalls */
 | |
| void xen_mc_flush(void);
 | |
| 
 | |
| /* Issue a multicall if we're not in a lazy mode */
 | |
| static inline void xen_mc_issue(unsigned mode)
 | |
| {
 | |
| 	if ((paravirt_get_lazy_mode() & mode) == 0)
 | |
| 		xen_mc_flush();
 | |
| 
 | |
| 	/* restore flags saved in xen_mc_batch */
 | |
| 	local_irq_restore(x86_read_percpu(xen_mc_irq_flags));
 | |
| }
 | |
| 
 | |
| /* Set up a callback to be called when the current batch is flushed */
 | |
| void xen_mc_callback(void (*fn)(void *), void *data);
 | |
| 
 | |
| /*
 | |
|  * Try to extend the arguments of the previous multicall command.  The
 | |
|  * previous command's op must match.  If it does, then it attempts to
 | |
|  * extend the argument space allocated to the multicall entry by
 | |
|  * arg_size bytes.
 | |
|  *
 | |
|  * The returned multicall_space will return with mc pointing to the
 | |
|  * command on success, or NULL on failure, and args pointing to the
 | |
|  * newly allocated space.
 | |
|  */
 | |
| struct multicall_space xen_mc_extend_args(unsigned long op, size_t arg_size);
 | |
| 
 | |
| #endif /* _XEN_MULTICALLS_H */
 |