f2ffa2a1022df5a44ff213b80b08a8deda5f5b31
[people/sha0/gpxe.git] / src / arch / i386 / include / virtaddr.h
1 #ifndef VIRTADDR_H
2 #define VIRTADDR_H
3
4 /* Segment selectors as used in our protected-mode GDTs.
5  *
6  * Don't change these unless you really know what you're doing.
7  */
8
9 #define VIRTUAL_CS 0x08
10 #define VIRTUAL_DS 0x10
11 #define PHYSICAL_CS 0x18
12 #define PHYSICAL_DS 0x20
13 #define REAL_CS 0x28
14 #define REAL_DS 0x30
15 #if 0
16 #define LONG_CS 0x38
17 #define LONG_DS 0x40
18 #endif
19
20 #ifndef ASSEMBLY
21
22 #include "stdint.h"
23 #include "string.h"
24
25 #ifndef KEEP_IT_REAL
26
27 /*
28  * Without -DKEEP_IT_REAL, we are in 32-bit protected mode with a
29  * fixed link address but an unknown physical start address.  Our GDT
30  * sets up code and data segments with an offset of virt_offset, so
31  * that link-time addresses can still work.
32  *
33  */
34
35 /* C-callable function prototypes */
36
37 extern void relocate_to ( uint32_t new_phys_addr );
38
39 /* Variables in virtaddr.S */
40 extern unsigned long virt_offset;
41
42 /*
43  * Convert between virtual and physical addresses
44  *
45  */
46 static inline unsigned long virt_to_phys ( volatile const void *virt_addr ) {
47         return ( ( unsigned long ) virt_addr ) + virt_offset;
48 }
49
50 static inline void * phys_to_virt ( unsigned long phys_addr ) {
51         return ( void * ) ( phys_addr - virt_offset );
52 }
53
54 static inline void copy_to_phys ( physaddr_t dest, const void *src,
55                                   size_t len ) {
56         memcpy ( phys_to_virt ( dest ), src, len );
57 }
58
59 static inline void copy_from_phys ( void *dest, physaddr_t src, size_t len ) {
60         memcpy ( dest, phys_to_virt ( src ), len );
61 }
62
63 static inline void copy_phys_to_phys ( physaddr_t dest, physaddr_t src,
64                                        size_t len ) {
65         memcpy ( phys_to_virt ( dest ), phys_to_virt ( src ), len );
66 }
67
68 #else /* KEEP_IT_REAL */
69
70 /*
71  * With -DKEEP_IT_REAL, we are in 16-bit real mode with fixed link
72  * addresses and a segmented memory model.  We have separate code and
73  * data segments.
74  *
75  * Because we may be called in 16-bit protected mode (damn PXE spec),
76  * we cannot simply assume that physical = segment * 16 + offset.
77  * Instead, we have to look up the physical start address of the
78  * segment in the !PXE structure.  We have to assume that
79  * virt_to_phys() is called only on pointers within the data segment,
80  * because nothing passes segment information to us.
81  *
82  * We don't implement phys_to_virt at all, because there will be many
83  * addresses that simply cannot be reached via a virtual address when
84  * the virtual address space is limited to 64kB!
85  */
86
87 static inline unsigned long virt_to_phys ( volatile const void *virt_addr ) {
88         /* Cheat: just for now, do the segment*16+offset calculation */
89         uint16_t ds;
90
91         __asm__ ( "movw %%ds, %%ax" : "=a" ( ds ) : );
92         return ( 16 * ds + ( ( unsigned long ) virt_addr ) );
93 }
94
95 /* Define it as a deprecated function so that we get compile-time
96  * warnings, rather than just the link-time errors.
97  */
98 extern void * phys_to_virt ( unsigned long phys_addr )
99      __attribute__ ((deprecated));
100
101 #endif /* KEEP_IT_REAL */
102
103 #endif /* ASSEMBLY */
104
105 #endif /* VIRTADDR_H */