[iSCSI] Support Windows Server 2008 direct iSCSI installation
[people/mdeck/gpxe.git] / src / arch / i386 / firmware / pcbios / hidemem.c
1 /* Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
2  *
3  * This program is free software; you can redistribute it and/or
4  * modify it under the terms of the GNU General Public License as
5  * published by the Free Software Foundation; either version 2 of the
6  * License, or any later version.
7  *
8  * This program is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11  * General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program; if not, write to the Free Software
15  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
16  */
17
18 #include <realmode.h>
19 #include <biosint.h>
20 #include <basemem.h>
21 #include <gpxe/init.h>
22 #include <gpxe/hidemem.h>
23
24 /** Alignment for hidden memory regions */
25 #define ALIGN_HIDDEN 4096   /* 4kB page alignment should be enough */
26
27 /**
28  * A hidden region of Etherboot
29  *
30  * This represents a region that will be edited out of the system's
31  * memory map.
32  *
33  * This structure is accessed by assembly code, so must not be
34  * changed.
35  */
36 struct hidden_region {
37         /* Physical start address */
38         physaddr_t start;
39         /* Physical end address */
40         physaddr_t end;
41 };
42
43 /**
44  * List of hidden regions
45  *
46  * Must be terminated by a zero entry.
47  */
48 struct hidden_region __data16_array ( hidden_regions, [] ) = {
49         [TEXT] = { 0, 0 },
50         [BASEMEM] = { ( 640 * 1024 ), ( 640 * 1024 ) },
51         [EXTMEM] = { 0, 0 },
52         { 0, 0, } /* Terminator */
53 };
54 #define hidden_regions __use_data16 ( hidden_regions )
55
56 /** Assembly routine in e820mangler.S */
57 extern void int15();
58
59 /** Vector for storing original INT 15 handler */
60 extern struct segoff __text16 ( int15_vector );
61 #define int15_vector __use_text16 ( int15_vector )
62
63 /**
64  * Hide region of memory from system memory map
65  *
66  * @v start             Start of region
67  * @v end               End of region
68  */
69 void hide_region ( unsigned int region_id, physaddr_t start, physaddr_t end ) {
70         struct hidden_region *region = &hidden_regions[region_id];
71
72         /* Some operating systems get a nasty shock if a region of the
73          * E820 map seems to start on a non-page boundary.  Make life
74          * safer by rounding out our edited region.
75          */
76         region->start = ( start & ~( ALIGN_HIDDEN - 1 ) );
77         region->end = ( ( end + ALIGN_HIDDEN - 1 ) & ~( ALIGN_HIDDEN - 1 ) );
78
79         DBG ( "Hiding region %d [%lx,%lx)\n",
80               region_id, region->start, region->end );
81 }
82
83 /**
84  * Hide Etherboot text
85  *
86  */
87 static void hide_text ( void ) {
88
89         /* The linker defines these symbols for us */
90         extern char _text[];
91         extern char _end[];
92
93         hide_region ( TEXT, virt_to_phys ( _text ), virt_to_phys ( _end ) );
94 }
95
96 /**
97  * Hide used base memory
98  *
99  */
100 void hide_basemem ( void ) {
101         /* Hide from the top of free base memory to 640kB.  Don't use
102          * hide_region(), because we don't want this rounded to the
103          * nearest page boundary.
104          */
105         hidden_regions[BASEMEM].start = ( get_fbms() * 1024 );
106 }
107
108 /**
109  * Hide Etherboot
110  *
111  * Installs an INT 15 handler to edit Etherboot out of the memory map
112  * returned by the BIOS.
113  */
114 static void hide_etherboot ( void ) {
115
116         /* Initialise the hidden regions */
117         hide_text();
118         hide_basemem();
119
120         /* Hook INT 15 */
121         hook_bios_interrupt ( 0x15, ( unsigned int ) int15,
122                               &int15_vector );
123 }
124
125 /**
126  * Unhide Etherboot
127  *
128  * Uninstalls the INT 15 handler installed by hide_etherboot(), if
129  * possible.
130  */
131 static void unhide_etherboot ( int flags __unused ) {
132
133         /* If we have more than one hooked interrupt at this point, it
134          * means that some other vector is still hooked, in which case
135          * we can't safely unhook INT 15 because we need to keep our
136          * memory protected.  (We expect there to be at least one
137          * hooked interrupt, because INT 15 itself is still hooked).
138          */
139         if ( hooked_bios_interrupts > 1 ) {
140                 DBG ( "Cannot unhide: %d interrupt vectors still hooked\n",
141                       hooked_bios_interrupts );
142                 return;
143         }
144
145         /* Try to unhook INT 15.  If it fails, then just leave it
146          * hooked; it takes care of protecting itself.  :)
147          */
148         unhook_bios_interrupt ( 0x15, ( unsigned int ) int15,
149                                 &int15_vector );
150 }
151
152 /** Hide Etherboot startup function */
153 struct startup_fn hide_etherboot_startup_fn __startup_fn ( STARTUP_EARLY ) = {
154         .startup = hide_etherboot,
155         .shutdown = unhide_etherboot,
156 };