137 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2015 Imagination Technologies
 | |
|  * Author: Alex Smith <alex.smith@imgtec.com>
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of the GNU General Public License as published by the
 | |
|  * Free Software Foundation;  either version 2 of the  License, or (at your
 | |
|  * option) any later version.
 | |
|  */
 | |
| 
 | |
| #ifndef __ASM_VDSO_H
 | |
| #define __ASM_VDSO_H
 | |
| 
 | |
| #include <linux/mm_types.h>
 | |
| 
 | |
| #include <asm/barrier.h>
 | |
| 
 | |
| /**
 | |
|  * struct mips_vdso_image - Details of a VDSO image.
 | |
|  * @data: Pointer to VDSO image data (page-aligned).
 | |
|  * @size: Size of the VDSO image data (page-aligned).
 | |
|  * @off_sigreturn: Offset of the sigreturn() trampoline.
 | |
|  * @off_rt_sigreturn: Offset of the rt_sigreturn() trampoline.
 | |
|  * @mapping: Special mapping structure.
 | |
|  *
 | |
|  * This structure contains details of a VDSO image, including the image data
 | |
|  * and offsets of certain symbols required by the kernel. It is generated as
 | |
|  * part of the VDSO build process, aside from the mapping page array, which is
 | |
|  * populated at runtime.
 | |
|  */
 | |
| struct mips_vdso_image {
 | |
| 	void *data;
 | |
| 	unsigned long size;
 | |
| 
 | |
| 	unsigned long off_sigreturn;
 | |
| 	unsigned long off_rt_sigreturn;
 | |
| 
 | |
| 	struct vm_special_mapping mapping;
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * The following structures are auto-generated as part of the build for each
 | |
|  * ABI by genvdso, see arch/mips/vdso/Makefile.
 | |
|  */
 | |
| 
 | |
| extern struct mips_vdso_image vdso_image;
 | |
| 
 | |
| #ifdef CONFIG_MIPS32_O32
 | |
| extern struct mips_vdso_image vdso_image_o32;
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_MIPS32_N32
 | |
| extern struct mips_vdso_image vdso_image_n32;
 | |
| #endif
 | |
| 
 | |
| /**
 | |
|  * union mips_vdso_data - Data provided by the kernel for the VDSO.
 | |
|  * @xtime_sec:		Current real time (seconds part).
 | |
|  * @xtime_nsec:		Current real time (nanoseconds part, shifted).
 | |
|  * @wall_to_mono_sec:	Wall-to-monotonic offset (seconds part).
 | |
|  * @wall_to_mono_nsec:	Wall-to-monotonic offset (nanoseconds part).
 | |
|  * @seq_count:		Counter to synchronise updates (odd = updating).
 | |
|  * @cs_shift:		Clocksource shift value.
 | |
|  * @clock_mode:		Clocksource to use for time functions.
 | |
|  * @cs_mult:		Clocksource multiplier value.
 | |
|  * @cs_cycle_last:	Clock cycle value at last update.
 | |
|  * @cs_mask:		Clocksource mask value.
 | |
|  * @tz_minuteswest:	Minutes west of Greenwich (from timezone).
 | |
|  * @tz_dsttime:		Type of DST correction (from timezone).
 | |
|  *
 | |
|  * This structure contains data needed by functions within the VDSO. It is
 | |
|  * populated by the kernel and mapped read-only into user memory. The time
 | |
|  * fields are mirrors of internal data from the timekeeping infrastructure.
 | |
|  *
 | |
|  * Note: Care should be taken when modifying as the layout must remain the same
 | |
|  * for both 64- and 32-bit (for 32-bit userland on 64-bit kernel).
 | |
|  */
 | |
| union mips_vdso_data {
 | |
| 	struct {
 | |
| 		u64 xtime_sec;
 | |
| 		u64 xtime_nsec;
 | |
| 		u64 wall_to_mono_sec;
 | |
| 		u64 wall_to_mono_nsec;
 | |
| 		u32 seq_count;
 | |
| 		u32 cs_shift;
 | |
| 		u8 clock_mode;
 | |
| 		u32 cs_mult;
 | |
| 		u64 cs_cycle_last;
 | |
| 		u64 cs_mask;
 | |
| 		s32 tz_minuteswest;
 | |
| 		s32 tz_dsttime;
 | |
| 	};
 | |
| 
 | |
| 	u8 page[PAGE_SIZE];
 | |
| };
 | |
| 
 | |
| static inline u32 vdso_data_read_begin(const union mips_vdso_data *data)
 | |
| {
 | |
| 	u32 seq;
 | |
| 
 | |
| 	while (true) {
 | |
| 		seq = READ_ONCE(data->seq_count);
 | |
| 		if (likely(!(seq & 1))) {
 | |
| 			/* Paired with smp_wmb() in vdso_data_write_*(). */
 | |
| 			smp_rmb();
 | |
| 			return seq;
 | |
| 		}
 | |
| 
 | |
| 		cpu_relax();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static inline bool vdso_data_read_retry(const union mips_vdso_data *data,
 | |
| 					u32 start_seq)
 | |
| {
 | |
| 	/* Paired with smp_wmb() in vdso_data_write_*(). */
 | |
| 	smp_rmb();
 | |
| 	return unlikely(data->seq_count != start_seq);
 | |
| }
 | |
| 
 | |
| static inline void vdso_data_write_begin(union mips_vdso_data *data)
 | |
| {
 | |
| 	++data->seq_count;
 | |
| 
 | |
| 	/* Ensure sequence update is written before other data page values. */
 | |
| 	smp_wmb();
 | |
| }
 | |
| 
 | |
| static inline void vdso_data_write_end(union mips_vdso_data *data)
 | |
| {
 | |
| 	/* Ensure data values are written before updating sequence again. */
 | |
| 	smp_wmb();
 | |
| 	++data->seq_count;
 | |
| }
 | |
| 
 | |
| #endif /* __ASM_VDSO_H */
 | 
