Merge mainline into mbivid
This commit is contained in:
		
						commit
						9ba27423f5
					
				
					 931 changed files with 122070 additions and 40781 deletions
				
			
		
							
								
								
									
										59
									
								
								grub-core/commands/i386/cmostest.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								grub-core/commands/i386/cmostest.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | |||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/command.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/cmos.h> | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_cmostest (struct grub_command *cmd __attribute__ ((unused)), | ||||
| 		   int argc, char *argv[]) | ||||
| { | ||||
|   int byte, bit; | ||||
|   char *rest; | ||||
| 
 | ||||
|   if (argc != 1) | ||||
|     return grub_error (GRUB_ERR_BAD_ARGUMENT, "Address required."); | ||||
| 
 | ||||
|   byte = grub_strtoul (argv[0], &rest, 0); | ||||
|   if (*rest != ':') | ||||
|     return grub_error (GRUB_ERR_BAD_ARGUMENT, "Address required."); | ||||
| 
 | ||||
|   bit = grub_strtoul (rest + 1, 0, 0); | ||||
| 
 | ||||
|   if (grub_cmos_read (byte) & (1 << bit)) | ||||
|     return GRUB_ERR_NONE; | ||||
| 
 | ||||
|   return grub_error (GRUB_ERR_TEST_FAILURE, "false"); | ||||
| } | ||||
| 
 | ||||
| static grub_command_t cmd; | ||||
| 
 | ||||
|  | ||||
| GRUB_MOD_INIT(cmostest) | ||||
| { | ||||
|   cmd = grub_register_command ("cmostest", grub_cmd_cmostest, | ||||
| 			       "cmostest BYTE:BIT", | ||||
| 			       "Test bit at BYTE:BIT in CMOS."); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(cmostest) | ||||
| { | ||||
|   grub_unregister_command (cmd); | ||||
| } | ||||
							
								
								
									
										98
									
								
								grub-core/commands/i386/cpuid.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								grub-core/commands/i386/cpuid.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,98 @@ | |||
| /* cpuid.c - test for CPU features */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2006, 2007, 2009  Free Software Foundation, Inc. | ||||
|  *  Based on gcc/gcc/config/i386/driver-i386.c | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/mm.h> | ||||
| #include <grub/env.h> | ||||
| #include <grub/command.h> | ||||
| #include <grub/extcmd.h> | ||||
| #include <grub/i386/cpuid.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| #define cpuid(num,a,b,c,d) \ | ||||
|   asm volatile ("xchgl %%ebx, %1; cpuid; xchgl %%ebx, %1" \ | ||||
| 		: "=a" (a), "=r" (b), "=c" (c), "=d" (d)  \ | ||||
| 		: "0" (num)) | ||||
| 
 | ||||
| static const struct grub_arg_option options[] = | ||||
|   { | ||||
|     {"long-mode", 'l', 0, N_("Check for long mode flag (default)."), 0, 0}, | ||||
|     {0, 0, 0, 0, 0, 0} | ||||
|   }; | ||||
| 
 | ||||
| #define bit_LM (1 << 29) | ||||
| 
 | ||||
| unsigned char grub_cpuid_has_longmode = 0; | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_cpuid (grub_extcmd_context_t ctxt __attribute__ ((unused)), | ||||
| 		int argc __attribute__ ((unused)), | ||||
| 		char **args __attribute__ ((unused))) | ||||
| { | ||||
|   return grub_cpuid_has_longmode ? GRUB_ERR_NONE | ||||
|     : grub_error (GRUB_ERR_TEST_FAILURE, "false"); | ||||
| } | ||||
| 
 | ||||
| static grub_extcmd_t cmd; | ||||
| 
 | ||||
| GRUB_MOD_INIT(cpuid) | ||||
| { | ||||
| #ifdef __x86_64__ | ||||
|   /* grub-emu */ | ||||
|   grub_cpuid_has_longmode = 1; | ||||
| #else | ||||
|   unsigned int eax, ebx, ecx, edx; | ||||
|   unsigned int max_level; | ||||
|   unsigned int ext_level; | ||||
| 
 | ||||
|   /* See if we can use cpuid.  */ | ||||
|   asm volatile ("pushfl; pushfl; popl %0; movl %0,%1; xorl %2,%0;" | ||||
| 		"pushl %0; popfl; pushfl; popl %0; popfl" | ||||
| 		: "=&r" (eax), "=&r" (ebx) | ||||
| 		: "i" (0x00200000)); | ||||
|   if (((eax ^ ebx) & 0x00200000) == 0) | ||||
|     goto done; | ||||
| 
 | ||||
|   /* Check the highest input value for eax.  */ | ||||
|   cpuid (0, eax, ebx, ecx, edx); | ||||
|   /* We only look at the first four characters.  */ | ||||
|   max_level = eax; | ||||
|   if (max_level == 0) | ||||
|     goto done; | ||||
| 
 | ||||
|   cpuid (0x80000000, eax, ebx, ecx, edx); | ||||
|   ext_level = eax; | ||||
|   if (ext_level < 0x80000000) | ||||
|     goto done; | ||||
| 
 | ||||
|   cpuid (0x80000001, eax, ebx, ecx, edx); | ||||
|   grub_cpuid_has_longmode = !!(edx & bit_LM); | ||||
| done: | ||||
| #endif | ||||
| 
 | ||||
|   cmd = grub_register_extcmd ("cpuid", grub_cmd_cpuid, 0, | ||||
| 			      "[-l]", N_("Check for CPU features."), options); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(cpuid) | ||||
| { | ||||
|   grub_unregister_extcmd (cmd); | ||||
| } | ||||
							
								
								
									
										81
									
								
								grub-core/commands/i386/pc/acpi.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								grub-core/commands/i386/pc/acpi.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,81 @@ | |||
| /* acpi.c - get acpi tables. */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/acpi.h> | ||||
| #include <grub/misc.h> | ||||
| 
 | ||||
| struct grub_acpi_rsdp_v10 * | ||||
| grub_machine_acpi_get_rsdpv1 (void) | ||||
| { | ||||
|   int ebda_len; | ||||
|   grub_uint8_t *ebda, *ptr; | ||||
| 
 | ||||
|   grub_dprintf ("acpi", "Looking for RSDP. Scanning EBDA\n"); | ||||
|   ebda = (grub_uint8_t *) ((* ((grub_uint16_t *) 0x40e)) << 4); | ||||
|   ebda_len = * (grub_uint16_t *) ebda; | ||||
|   if (! ebda_len) | ||||
|     return 0; | ||||
|   for (ptr = ebda; ptr < ebda + 0x400; ptr += 16) | ||||
|     if (grub_memcmp (ptr, "RSD PTR ", 8) == 0 | ||||
| 	&& grub_byte_checksum (ptr, sizeof (struct grub_acpi_rsdp_v10)) == 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v10 *) ptr)->revision == 0) | ||||
|       return (struct grub_acpi_rsdp_v10 *) ptr; | ||||
| 
 | ||||
|   grub_dprintf ("acpi", "Looking for RSDP. Scanning BIOS\n"); | ||||
|   for (ptr = (grub_uint8_t *) 0xe0000; ptr < (grub_uint8_t *) 0x100000; | ||||
|        ptr += 16) | ||||
|     if (grub_memcmp (ptr, "RSD PTR ", 8) == 0 | ||||
| 	&& grub_byte_checksum (ptr, sizeof (struct grub_acpi_rsdp_v10)) == 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v10 *) ptr)->revision == 0) | ||||
|       return (struct grub_acpi_rsdp_v10 *) ptr; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| struct grub_acpi_rsdp_v20 * | ||||
| grub_machine_acpi_get_rsdpv2 (void) | ||||
| { | ||||
|   int ebda_len; | ||||
|   grub_uint8_t *ebda, *ptr; | ||||
| 
 | ||||
|   grub_dprintf ("acpi", "Looking for RSDP. Scanning EBDA\n"); | ||||
|   ebda = (grub_uint8_t *) ((* ((grub_uint16_t *) 0x40e)) << 4); | ||||
|   ebda_len = * (grub_uint16_t *) ebda; | ||||
|   if (! ebda_len) | ||||
|     return 0; | ||||
|   for (ptr = ebda; ptr < ebda + 0x400; ptr += 16) | ||||
|     if (grub_memcmp (ptr, "RSD PTR ", 8) == 0 | ||||
| 	&& grub_byte_checksum (ptr, sizeof (struct grub_acpi_rsdp_v10)) == 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v10 *) ptr)->revision != 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v20 *) ptr)->length < 1024 | ||||
| 	&& grub_byte_checksum (ptr, ((struct grub_acpi_rsdp_v20 *) ptr)->length) | ||||
| 	== 0) | ||||
|       return (struct grub_acpi_rsdp_v20 *) ptr; | ||||
| 
 | ||||
|   grub_dprintf ("acpi", "Looking for RSDP. Scanning BIOS\n"); | ||||
|   for (ptr = (grub_uint8_t *) 0xe0000; ptr < (grub_uint8_t *) 0x100000; | ||||
|        ptr += 16) | ||||
|     if (grub_memcmp (ptr, "RSD PTR ", 8) == 0 | ||||
| 	&& grub_byte_checksum (ptr, sizeof (struct grub_acpi_rsdp_v10)) == 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v10 *) ptr)->revision != 0 | ||||
| 	&& ((struct grub_acpi_rsdp_v20 *) ptr)->length < 1024 | ||||
| 	&& grub_byte_checksum (ptr, ((struct grub_acpi_rsdp_v20 *) ptr)->length) | ||||
| 	== 0) | ||||
|       return (struct grub_acpi_rsdp_v20 *) ptr; | ||||
|   return 0; | ||||
| } | ||||
							
								
								
									
										420
									
								
								grub-core/commands/i386/pc/drivemap.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										420
									
								
								grub-core/commands/i386/pc/drivemap.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,420 @@ | |||
| /* drivemap.c - command to manage the BIOS drive mappings.  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2008, 2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/extcmd.h> | ||||
| #include <grub/dl.h> | ||||
| #include <grub/mm.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/disk.h> | ||||
| #include <grub/loader.h> | ||||
| #include <grub/env.h> | ||||
| #include <grub/machine/memory.h> | ||||
| #include <grub/machine/biosnum.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| 
 | ||||
| /* Real mode IVT slot (seg:off far pointer) for interrupt 0x13.  */ | ||||
| static grub_uint32_t *const int13slot = UINT_TO_PTR (4 * 0x13); | ||||
| 
 | ||||
| /* Remember to update enum opt_idxs accordingly.  */ | ||||
| static const struct grub_arg_option options[] = { | ||||
|   {"list", 'l', 0, N_("Show the current mappings."), 0, 0}, | ||||
|   {"reset", 'r', 0, N_("Reset all mappings to the default values."), 0, 0}, | ||||
|   {"swap", 's', 0, N_("Perform both direct and reverse mappings."), 0, 0}, | ||||
|   {0, 0, 0, 0, 0, 0} | ||||
| }; | ||||
| 
 | ||||
| /* Remember to update options[] accordingly.  */ | ||||
| enum opt_idxs | ||||
| { | ||||
|   OPTIDX_LIST = 0, | ||||
|   OPTIDX_RESET, | ||||
|   OPTIDX_SWAP, | ||||
| }; | ||||
| 
 | ||||
| /* Realmode far ptr (2 * 16b) to the previous INT13h handler.  */ | ||||
| extern grub_uint32_t grub_drivemap_oldhandler; | ||||
| 
 | ||||
| /* The type "void" is used for imported assembly labels, takes no storage and
 | ||||
|    serves just to take the address with &label.  */ | ||||
| 
 | ||||
| /* The assembly function to replace the old INT13h handler. It does not follow
 | ||||
|    any C callspecs and returns with IRET.  */ | ||||
| extern const void grub_drivemap_handler; | ||||
| 
 | ||||
| /* Start of the drive mappings area (space reserved at runtime).  */ | ||||
| extern const void grub_drivemap_mapstart; | ||||
| 
 | ||||
| typedef struct drivemap_node | ||||
| { | ||||
|   struct drivemap_node *next; | ||||
|   grub_uint8_t newdrive; | ||||
|   grub_uint8_t redirto; | ||||
| } drivemap_node_t; | ||||
| 
 | ||||
| typedef struct __attribute__ ((packed)) int13map_node | ||||
| { | ||||
|   grub_uint8_t disknum; | ||||
|   grub_uint8_t mapto; | ||||
| } int13map_node_t; | ||||
| 
 | ||||
| #define INT13H_OFFSET(x) \ | ||||
| 	(((grub_uint8_t *)(x)) - ((grub_uint8_t *)&grub_drivemap_handler)) | ||||
| 
 | ||||
| static drivemap_node_t *map_head; | ||||
| static void *drivemap_hook; | ||||
| static int drivemap_mmap; | ||||
| 
 | ||||
| /* Puts the specified mapping into the table, replacing an existing mapping
 | ||||
|    for newdrive or adding a new one if required.  */ | ||||
| static grub_err_t | ||||
| drivemap_set (grub_uint8_t newdrive, grub_uint8_t redirto) | ||||
| { | ||||
|   drivemap_node_t *mapping = 0; | ||||
|   drivemap_node_t *search = map_head; | ||||
|   while (search) | ||||
|     { | ||||
|       if (search->newdrive == newdrive) | ||||
| 	{ | ||||
| 	  mapping = search; | ||||
| 	  break; | ||||
| 	} | ||||
|       search = search->next; | ||||
|     } | ||||
| 
 | ||||
|   /* Check for pre-existing mappings to modify before creating a new one.  */ | ||||
|   if (mapping) | ||||
|     mapping->redirto = redirto; | ||||
|   else | ||||
|     { | ||||
|       mapping = grub_malloc (sizeof (drivemap_node_t)); | ||||
|       if (! mapping) | ||||
| 	return grub_error (GRUB_ERR_OUT_OF_MEMORY, | ||||
| 			   "cannot allocate map entry, not enough memory"); | ||||
|       mapping->newdrive = newdrive; | ||||
|       mapping->redirto = redirto; | ||||
|       mapping->next = map_head; | ||||
|       map_head = mapping; | ||||
|     } | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| /* Removes the mapping for newdrive from the table.  If there is no mapping,
 | ||||
|    then this function behaves like a no-op on the map.  */ | ||||
| static void | ||||
| drivemap_remove (grub_uint8_t newdrive) | ||||
| { | ||||
|   drivemap_node_t *mapping = 0; | ||||
|   drivemap_node_t *search = map_head; | ||||
|   drivemap_node_t *previous = 0; | ||||
| 
 | ||||
|   while (search) | ||||
|     { | ||||
|       if (search->newdrive == newdrive) | ||||
| 	{ | ||||
| 	  mapping = search; | ||||
| 	  break; | ||||
| 	} | ||||
|       previous = search; | ||||
|       search = search->next; | ||||
|     } | ||||
| 
 | ||||
|   if (mapping) | ||||
|     { | ||||
|       if (previous) | ||||
| 	previous->next = mapping->next; | ||||
|       else | ||||
| 	map_head = mapping->next; | ||||
|       grub_free (mapping); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /* Given a GRUB-like device name and a convenient location, stores the
 | ||||
|    related BIOS disk number.  Accepts devices like \((f|h)dN\), with | ||||
|    0 <= N < 128.  */ | ||||
| static grub_err_t | ||||
| tryparse_diskstring (const char *str, grub_uint8_t *output) | ||||
| { | ||||
|   /* Skip opening paren in order to allow both (hd0) and hd0.  */ | ||||
|   if (*str == '(') | ||||
|     str++; | ||||
|   if ((str[0] == 'f' || str[0] == 'h') && str[1] == 'd') | ||||
|     { | ||||
|       grub_uint8_t bios_num = (str[0] == 'h') ? 0x80 : 0x00; | ||||
|       unsigned long drivenum = grub_strtoul (str + 2, 0, 0); | ||||
|       if (grub_errno == GRUB_ERR_NONE && drivenum < 128) | ||||
| 	{ | ||||
| 	  bios_num |= drivenum; | ||||
| 	  if (output) | ||||
| 	    *output = bios_num; | ||||
| 	  return GRUB_ERR_NONE; | ||||
| 	} | ||||
|     } | ||||
|   return grub_error (GRUB_ERR_BAD_ARGUMENT, "device format \"%s\" " | ||||
| 		     "invalid: must be (f|h)dN, with 0 <= N < 128", str); | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| list_mappings (void) | ||||
| { | ||||
|   /* Show: list mappings.  */ | ||||
|   if (! map_head) | ||||
|     { | ||||
|       grub_printf ("No drives have been remapped\n"); | ||||
|       return GRUB_ERR_NONE; | ||||
|     } | ||||
| 
 | ||||
|   grub_printf ("OS disk #num ------> GRUB/BIOS device\n"); | ||||
|   drivemap_node_t *curnode = map_head; | ||||
|   while (curnode) | ||||
|     { | ||||
|       grub_printf ("%cD #%-3u (0x%02x)       %cd%d\n", | ||||
| 		   (curnode->newdrive & 0x80) ? 'H' : 'F', | ||||
| 		   curnode->newdrive & 0x7F, curnode->newdrive, | ||||
| 		   (curnode->redirto & 0x80) ? 'h' : 'f', | ||||
| 		   curnode->redirto & 0x7F | ||||
| 		   ); | ||||
|       curnode = curnode->next; | ||||
|     } | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_drivemap (struct grub_extcmd_context *ctxt, int argc, char **args) | ||||
| { | ||||
|   if (ctxt->state[OPTIDX_LIST].set) | ||||
|     { | ||||
|       return list_mappings (); | ||||
|     } | ||||
|   else if (ctxt->state[OPTIDX_RESET].set) | ||||
|     { | ||||
|       /* Reset: just delete all mappings, freeing their memory.  */ | ||||
|       drivemap_node_t *curnode = map_head; | ||||
|       drivemap_node_t *prevnode = 0; | ||||
|       while (curnode) | ||||
| 	{ | ||||
| 	  prevnode = curnode; | ||||
| 	  curnode = curnode->next; | ||||
| 	  grub_free (prevnode); | ||||
| 	} | ||||
|       map_head = 0; | ||||
|       return GRUB_ERR_NONE; | ||||
|     } | ||||
|   else if (!ctxt->state[OPTIDX_SWAP].set && argc == 0) | ||||
|     { | ||||
|       /* No arguments */ | ||||
|       return list_mappings (); | ||||
|     } | ||||
| 
 | ||||
|   /* Neither flag: put mapping.  */ | ||||
|   grub_uint8_t mapfrom = 0; | ||||
|   grub_uint8_t mapto = 0xFF; | ||||
|   grub_err_t err; | ||||
| 
 | ||||
|   if (argc != 2) | ||||
|     return grub_error (GRUB_ERR_BAD_ARGUMENT, "two arguments required"); | ||||
| 
 | ||||
|   err = tryparse_diskstring (args[0], &mapfrom); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   err = tryparse_diskstring (args[1], &mapto); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   if (mapto == mapfrom) | ||||
|     { | ||||
|       /* Reset to default.  */ | ||||
|       grub_dprintf ("drivemap", "Removing mapping for %s (%02x)\n", | ||||
| 		    args[0], mapfrom); | ||||
|       drivemap_remove (mapfrom); | ||||
|       return GRUB_ERR_NONE; | ||||
|     } | ||||
|   /* Set the mapping for the disk (overwrites any existing mapping).  */ | ||||
|   grub_dprintf ("drivemap", "%s %s (%02x) = %s (%02x)\n", | ||||
| 		ctxt->state[OPTIDX_SWAP].set ? "Swapping" : "Mapping", | ||||
| 		args[1], mapto, args[0], mapfrom); | ||||
|   err = drivemap_set (mapto, mapfrom); | ||||
|   /* If -s, perform the reverse mapping too (only if the first was OK).  */ | ||||
|   if (ctxt->state[OPTIDX_SWAP].set && err == GRUB_ERR_NONE) | ||||
|     err = drivemap_set (mapfrom, mapto); | ||||
|   return err; | ||||
| } | ||||
| 
 | ||||
| /* Int13h handler installer - reserves conventional memory for the handler,
 | ||||
|    copies it over and sets the IVT entry for int13h. | ||||
|    This code rests on the assumption that GRUB does not activate any kind | ||||
|    of memory mapping apart from identity paging, since it accesses | ||||
|    realmode structures by their absolute addresses, like the IVT at 0; | ||||
|    and transforms a pmode pointer into a rmode seg:off far ptr.  */ | ||||
| static grub_err_t | ||||
| install_int13_handler (int noret __attribute__ ((unused))) | ||||
| { | ||||
|   /* Size of the full int13 handler "bundle", including code and map.  */ | ||||
|   grub_uint32_t total_size; | ||||
|   /* Base address of the space reserved for the handler bundle.  */ | ||||
|   grub_uint8_t *handler_base = 0; | ||||
|   /* Address of the map within the deployed bundle.  */ | ||||
|   int13map_node_t *handler_map; | ||||
| 
 | ||||
|   int i; | ||||
|   int entries = 0; | ||||
|   drivemap_node_t *curentry = map_head; | ||||
| 
 | ||||
|   /* Count entries to prepare a contiguous map block.  */ | ||||
|   while (curentry) | ||||
|     { | ||||
|       entries++; | ||||
|       curentry = curentry->next; | ||||
|     } | ||||
|   if (entries == 0) | ||||
|     { | ||||
|       /* No need to install the int13h handler.  */ | ||||
|       grub_dprintf ("drivemap", "No drives marked as remapped, not installing " | ||||
| 		    "our int13h handler.\n"); | ||||
|       return GRUB_ERR_NONE; | ||||
|     } | ||||
| 
 | ||||
|   grub_dprintf ("drivemap", "Installing our int13h handler\n"); | ||||
| 
 | ||||
|   /* Save the pointer to the old handler.  */ | ||||
|   grub_drivemap_oldhandler = *int13slot; | ||||
|   grub_dprintf ("drivemap", "Original int13 handler: %04x:%04x\n", | ||||
| 		(grub_drivemap_oldhandler >> 16) & 0x0ffff, | ||||
| 		grub_drivemap_oldhandler & 0x0ffff); | ||||
| 
 | ||||
|   /* Find a rmode-segment-aligned zone in conventional memory big
 | ||||
|      enough to hold the handler and its data.  */ | ||||
|   total_size = INT13H_OFFSET (&grub_drivemap_mapstart) | ||||
|     + (entries + 1) * sizeof (int13map_node_t); | ||||
|   grub_dprintf ("drivemap", "Payload is %u bytes long\n", total_size); | ||||
|   handler_base = grub_mmap_malign_and_register (16, total_size, | ||||
| 						&drivemap_mmap, | ||||
| 						GRUB_MACHINE_MEMORY_RESERVED, | ||||
| 						GRUB_MMAP_MALLOC_LOW); | ||||
|   if (! handler_base) | ||||
|     return grub_error (GRUB_ERR_OUT_OF_MEMORY, "couldn't reserve " | ||||
| 		       "memory for the int13h handler"); | ||||
| 
 | ||||
|   /* Copy int13h handler bundle to reserved area.  */ | ||||
|   grub_dprintf ("drivemap", "Reserved memory at %p, copying handler\n", | ||||
| 		handler_base); | ||||
|   grub_memcpy (handler_base, &grub_drivemap_handler, | ||||
| 	       INT13H_OFFSET (&grub_drivemap_mapstart)); | ||||
| 
 | ||||
|   /* Copy the mappings to the reserved area.  */ | ||||
|   curentry = map_head; | ||||
|   handler_map = (int13map_node_t *) (handler_base + | ||||
| 				     INT13H_OFFSET (&grub_drivemap_mapstart)); | ||||
|   grub_dprintf ("drivemap", "Target map at %p, copying mappings\n", handler_map); | ||||
|   for (i = 0; i < entries; ++i, curentry = curentry->next) | ||||
|     { | ||||
|       handler_map[i].disknum = curentry->newdrive; | ||||
|       handler_map[i].mapto = curentry->redirto; | ||||
|       grub_dprintf ("drivemap", "\t#%d: 0x%02x <- 0x%02x\n", i, | ||||
| 		    handler_map[i].disknum, handler_map[i].mapto); | ||||
|     } | ||||
|   /* Signal end-of-map.  */ | ||||
|   handler_map[i].disknum = 0; | ||||
|   handler_map[i].mapto = 0; | ||||
|   grub_dprintf ("drivemap", "\t#%d: 0x00 <- 0x00 (end)\n", i); | ||||
| 
 | ||||
|   /* Install our function as the int13h handler in the IVT.  */ | ||||
|   *int13slot = ((grub_uint32_t) handler_base) << 12;	/* Segment address.  */ | ||||
|   grub_dprintf ("drivemap", "New int13 handler: %04x:%04x\n", | ||||
| 		(*int13slot >> 16) & 0x0ffff, *int13slot & 0x0ffff); | ||||
| 
 | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| uninstall_int13_handler (void) | ||||
| { | ||||
|   if (! grub_drivemap_oldhandler) | ||||
|     return GRUB_ERR_NONE; | ||||
| 
 | ||||
|   *int13slot = grub_drivemap_oldhandler; | ||||
|   grub_mmap_free_and_unregister (drivemap_mmap); | ||||
|   grub_drivemap_oldhandler = 0; | ||||
|   grub_dprintf ("drivemap", "Restored int13 handler: %04x:%04x\n", | ||||
| 		(*int13slot >> 16) & 0x0ffff, *int13slot & 0x0ffff); | ||||
| 
 | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| grub_get_root_biosnumber_drivemap (void) | ||||
| { | ||||
|   char *biosnum; | ||||
|   int ret = -1; | ||||
|   grub_device_t dev; | ||||
| 
 | ||||
|   biosnum = grub_env_get ("biosnum"); | ||||
| 
 | ||||
|   if (biosnum) | ||||
|     return grub_strtoul (biosnum, 0, 0); | ||||
| 
 | ||||
|   dev = grub_device_open (0); | ||||
|   if (dev && dev->disk && dev->disk->dev | ||||
|       && dev->disk->dev->id == GRUB_DISK_DEVICE_BIOSDISK_ID) | ||||
|     { | ||||
|       drivemap_node_t *curnode = map_head; | ||||
|       ret = (int) dev->disk->id; | ||||
|       while (curnode) | ||||
| 	{ | ||||
| 	  if (curnode->redirto == ret) | ||||
| 	    { | ||||
| 	      ret = curnode->newdrive; | ||||
| 	      break; | ||||
| 	    } | ||||
| 	  curnode = curnode->next; | ||||
| 	} | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|   if (dev) | ||||
|     grub_device_close (dev); | ||||
| 
 | ||||
|   return ret; | ||||
| } | ||||
| 
 | ||||
| static grub_extcmd_t cmd; | ||||
| static int (*grub_get_root_biosnumber_saved) (void); | ||||
| 
 | ||||
| GRUB_MOD_INIT (drivemap) | ||||
| { | ||||
|   grub_get_root_biosnumber_saved = grub_get_root_biosnumber; | ||||
|   grub_get_root_biosnumber = grub_get_root_biosnumber_drivemap; | ||||
|   cmd = grub_register_extcmd ("drivemap", grub_cmd_drivemap, 0, | ||||
| 			      N_("-l | -r | [-s] grubdev osdisk."), | ||||
| 			      N_("Manage the BIOS drive mappings."), | ||||
| 			      options); | ||||
|   drivemap_hook = | ||||
|     grub_loader_register_preboot_hook (&install_int13_handler, | ||||
| 				       &uninstall_int13_handler, | ||||
| 				       GRUB_LOADER_PREBOOT_HOOK_PRIO_NORMAL); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI (drivemap) | ||||
| { | ||||
|   grub_get_root_biosnumber = grub_get_root_biosnumber_saved; | ||||
|   grub_loader_unregister_preboot_hook (drivemap_hook); | ||||
|   drivemap_hook = 0; | ||||
|   grub_unregister_extcmd (cmd); | ||||
| } | ||||
							
								
								
									
										110
									
								
								grub-core/commands/i386/pc/drivemap_int13h.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								grub-core/commands/i386/pc/drivemap_int13h.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,110 @@ | |||
| /* drivemap_int13h.S - interrupt handler for the BIOS drive remapper */ | ||||
| /* | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2008, 2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/symbol.h> | ||||
| 
 | ||||
| #define INT13H_OFFSET(x) ((x) - LOCAL (base)) | ||||
| 
 | ||||
| .code16 | ||||
| 
 | ||||
| /* Copy starts here.  When deployed, this code must be segment-aligned.  */ | ||||
| 
 | ||||
| /* The replacement int13 handler.   Preserve all registers.  */ | ||||
| FUNCTION(grub_drivemap_handler) | ||||
| LOCAL (base): | ||||
| 	/* Save %dx for future restore. */ | ||||
| 	push	%dx | ||||
| 	/* Push flags. Used to simulate interrupt with original flags. */ | ||||
| 	pushf | ||||
| 
 | ||||
| 	/* Map the drive number (always in DL).  */ | ||||
| 	push	%ax | ||||
| 	push	%bx | ||||
| 	movw	$INT13H_OFFSET(LOCAL (mapstart)), %bx | ||||
| 
 | ||||
| more_remaining: | ||||
| 	movw	%cs:(%bx), %ax | ||||
| 	cmpb	%ah, %al | ||||
| 	jz	not_found /* DRV=DST => map end - drive not remapped, keep DL.  */ | ||||
| 	inc	%bx | ||||
| 	inc	%bx | ||||
| 	cmpb	%dl, %al | ||||
| 	jnz	more_remaining /* Not found, but more remaining, loop.  */ | ||||
| 	movb	%ah, %dl /* Found - drive remapped, modify DL.  */ | ||||
| 
 | ||||
| not_found: | ||||
| 	pop	%bx | ||||
| 	pop	%ax | ||||
| 
 | ||||
| 	/* If the call isn't ah=0x8 or ah=0x15 we must restore %dx.  */ | ||||
| 	cmpb	$0x8, %ah | ||||
| 	jz	norestore | ||||
| 	cmpb	$0x15, %ah | ||||
| 	jz	norestore | ||||
| 
 | ||||
| 	/* Restore flags.  */ | ||||
| 	popf | ||||
| 	pushf | ||||
| 
 | ||||
| 	lcall *%cs:INT13H_OFFSET (LOCAL (oldhandler)) | ||||
| 
 | ||||
| 	push	%bp | ||||
| 	mov	%sp, %bp | ||||
| 
 | ||||
| tail: | ||||
| 	/* Save new flags below %esp so the caller will recieve new flags.  */ | ||||
| 	pushf | ||||
| 	pop	%dx | ||||
| 	mov	%dx, 8(%bp) | ||||
| 
 | ||||
| 	pop	%bp | ||||
| 
 | ||||
| 	/* Restore %dx.  */ | ||||
| 	pop	%dx | ||||
| 	iret | ||||
| 
 | ||||
| norestore: | ||||
| 
 | ||||
| 	/* Restore flags.  */ | ||||
| 	popf | ||||
| 	pushf | ||||
| 
 | ||||
| 	lcall *%cs:INT13H_OFFSET (LOCAL (oldhandler)) | ||||
| 
 | ||||
| 	push	%bp | ||||
| 	mov	%sp, %bp | ||||
| 
 | ||||
| 	/* Save %dx. So it won't be restored to original value.  */ | ||||
| 	mov	%dx, 2(%bp) | ||||
| 
 | ||||
| 	jmp tail | ||||
| 
 | ||||
| /* Far pointer to the old handler.  Stored as a CS:IP in the style of real-mode | ||||
|    IVT entries (thus PI:SC in mem).  */ | ||||
| VARIABLE(grub_drivemap_oldhandler) | ||||
| LOCAL (oldhandler): | ||||
| 	.word 0x0, 0x0 | ||||
| 
 | ||||
| /* This label MUST be at the end of the copied block, since the installer code | ||||
|    reserves additional space for mappings at runtime and copies them over it.  */ | ||||
| 	.align 2
 | ||||
| 	 | ||||
| VARIABLE(grub_drivemap_mapstart) | ||||
| LOCAL (mapstart): | ||||
| 	.byte 0
 | ||||
							
								
								
									
										125
									
								
								grub-core/commands/i386/pc/halt.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								grub-core/commands/i386/pc/halt.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,125 @@ | |||
| /* halt.c - command to halt the computer.  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2005,2007,2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/extcmd.h> | ||||
| #include <grub/i18n.h> | ||||
| #include <grub/machine/int.h> | ||||
| #include <grub/acpi.h> | ||||
| 
 | ||||
| static const struct grub_arg_option options[] = | ||||
|   { | ||||
|     {"no-apm", 'n', 0, N_("Do not use APM to halt the computer."), 0, 0}, | ||||
|     {0, 0, 0, 0, 0, 0} | ||||
|   }; | ||||
| 
 | ||||
| static inline void __attribute__ ((noreturn)) | ||||
| stop (void) | ||||
| { | ||||
|   while (1) | ||||
|     { | ||||
|       asm volatile ("hlt"); | ||||
|     } | ||||
| } | ||||
| /*
 | ||||
|  * Halt the system, using APM if possible. If NO_APM is true, don't use | ||||
|  * APM even if it is available. | ||||
|  */ | ||||
| void  | ||||
| grub_halt (int no_apm) | ||||
| { | ||||
|   struct grub_bios_int_registers regs; | ||||
| 
 | ||||
|   grub_acpi_halt (); | ||||
| 
 | ||||
|   if (no_apm) | ||||
|     stop (); | ||||
| 
 | ||||
|   /* detect APM */ | ||||
|   regs.eax = 0x5300; | ||||
|   regs.ebx = 0; | ||||
|   regs.flags = GRUB_CPU_INT_FLAGS_DEFAULT; | ||||
|   grub_bios_interrupt (0x15, ®s); | ||||
|    | ||||
|   if (regs.flags & GRUB_CPU_INT_FLAGS_CARRY) | ||||
|     stop (); | ||||
| 
 | ||||
|   /* disconnect APM first */ | ||||
|   regs.eax = 0x5304; | ||||
|   regs.ebx = 0; | ||||
|   regs.flags = GRUB_CPU_INT_FLAGS_DEFAULT; | ||||
|   grub_bios_interrupt (0x15, ®s); | ||||
| 
 | ||||
|   /* connect APM */ | ||||
|   regs.eax = 0x5301; | ||||
|   regs.ebx = 0; | ||||
|   regs.flags = GRUB_CPU_INT_FLAGS_DEFAULT; | ||||
|   grub_bios_interrupt (0x15, ®s); | ||||
|   if (regs.flags & GRUB_CPU_INT_FLAGS_CARRY) | ||||
|     stop (); | ||||
| 
 | ||||
|   /* set APM protocol level - 1.1 or bust. (this covers APM 1.2 also) */ | ||||
|   regs.eax = 0x530E; | ||||
|   regs.ebx = 0; | ||||
|   regs.ecx = 0x0101; | ||||
|   regs.flags = GRUB_CPU_INT_FLAGS_DEFAULT; | ||||
|   grub_bios_interrupt (0x15, ®s); | ||||
|   if (regs.flags & GRUB_CPU_INT_FLAGS_CARRY) | ||||
|     stop (); | ||||
| 
 | ||||
|   /* set the power state to off */ | ||||
|   regs.eax = 0x5307; | ||||
|   regs.ebx = 1; | ||||
|   regs.ecx = 3; | ||||
|   regs.flags = GRUB_CPU_INT_FLAGS_DEFAULT; | ||||
|   grub_bios_interrupt (0x15, ®s); | ||||
| 
 | ||||
|   /* shouldn't reach here */ | ||||
|   stop (); | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_halt (grub_extcmd_context_t ctxt, | ||||
| 	       int argc __attribute__ ((unused)), | ||||
| 	       char **args __attribute__ ((unused))) | ||||
| 
 | ||||
| { | ||||
|   struct grub_arg_list *state = ctxt->state; | ||||
|   int no_apm = 0; | ||||
| 
 | ||||
|   if (state[0].set) | ||||
|     no_apm = 1; | ||||
|   grub_halt (no_apm); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static grub_extcmd_t cmd; | ||||
|  | ||||
| GRUB_MOD_INIT(halt) | ||||
| { | ||||
|   cmd = grub_register_extcmd ("halt", grub_cmd_halt, 0, "[-n]", | ||||
| 			      N_("Halt the system, if possible using APM."), | ||||
| 			      options); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(halt) | ||||
| { | ||||
|   grub_unregister_extcmd (cmd); | ||||
| } | ||||
							
								
								
									
										273
									
								
								grub-core/commands/i386/pc/play.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										273
									
								
								grub-core/commands/i386/pc/play.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,273 @@ | |||
| /* play.c - command to play a tune  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2005,2007,2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| /* Lots of this file is borrowed from GNU/Hurd generic-speaker driver.  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/file.h> | ||||
| #include <grub/disk.h> | ||||
| #include <grub/term.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/machine/time.h> | ||||
| #include <grub/cpu/io.h> | ||||
| #include <grub/command.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| #define BASE_TEMPO (60 * GRUB_TICKS_PER_SECOND) | ||||
| 
 | ||||
| /* The speaker port.  */ | ||||
| #define SPEAKER			0x61 | ||||
| 
 | ||||
| /* If 0, follow state of SPEAKER_DATA bit, otherwise enable output
 | ||||
|    from timer 2.  */ | ||||
| #define SPEAKER_TMR2		0x01 | ||||
| 
 | ||||
| /* If SPEAKER_TMR2 is not set, this provides direct input into the
 | ||||
|    speaker.  Otherwise, this enables or disables the output from the | ||||
|    timer.  */ | ||||
| #define SPEAKER_DATA		0x02 | ||||
| 
 | ||||
| /* The PIT channel value ports.  You can write to and read from them.
 | ||||
|    Do not mess with timer 0 or 1.  */ | ||||
| #define PIT_COUNTER_0		0x40 | ||||
| #define PIT_COUNTER_1		0x41 | ||||
| #define PIT_COUNTER_2		0x42 | ||||
| 
 | ||||
| /* The frequency of the PIT clock.  */ | ||||
| #define PIT_FREQUENCY		0x1234dd | ||||
| 
 | ||||
| /* The PIT control port.  You can only write to it.  Do not mess with
 | ||||
|    timer 0 or 1.  */ | ||||
| #define PIT_CTRL		0x43 | ||||
| #define PIT_CTRL_SELECT_MASK	0xc0 | ||||
| #define PIT_CTRL_SELECT_0	0x00 | ||||
| #define PIT_CTRL_SELECT_1	0x40 | ||||
| #define PIT_CTRL_SELECT_2	0x80 | ||||
| 
 | ||||
| /* Read and load control.  */ | ||||
| #define PIT_CTRL_READLOAD_MASK	0x30 | ||||
| #define PIT_CTRL_COUNTER_LATCH	0x00	/* Hold timer value until read.  */ | ||||
| #define PIT_CTRL_READLOAD_LSB	0x10	/* Read/load the LSB.  */ | ||||
| #define PIT_CTRL_READLOAD_MSB	0x20	/* Read/load the MSB.  */ | ||||
| #define PIT_CTRL_READLOAD_WORD	0x30	/* Read/load the LSB then the MSB.  */ | ||||
| 
 | ||||
| /* Mode control.  */ | ||||
| #define PIT_CTRL_MODE_MASK	0x0e | ||||
| 
 | ||||
| /* Interrupt on terminal count.  Setting the mode sets output to low.
 | ||||
|    When counter is set and terminated, output is set to high.  */ | ||||
| #define PIT_CTRL_INTR_ON_TERM	0x00 | ||||
| 
 | ||||
| /* Programmable one-shot.  When loading counter, output is set to
 | ||||
|    high.  When counter terminated, output is set to low.  Can be | ||||
|    triggered again from that point on by setting the gate pin to | ||||
|    high.  */ | ||||
| #define PIT_CTRL_PROGR_ONE_SHOT	0x02 | ||||
| 
 | ||||
| /* Rate generator.  Output is low for one period of the counter, and
 | ||||
|    high for the other.  */ | ||||
| #define PIT_CTRL_RATE_GEN	0x04 | ||||
| 
 | ||||
| /* Square wave generator.  Output is low for one half of the period,
 | ||||
|    and high for the other half.  */ | ||||
| #define PIT_CTRL_SQUAREWAVE_GEN	0x06 | ||||
| 
 | ||||
| /* Software triggered strobe.  Setting the mode sets output to high.
 | ||||
|    When counter is set and terminated, output is set to low.  */ | ||||
| #define PIT_CTRL_SOFTSTROBE	0x08 | ||||
| 
 | ||||
| /* Hardware triggered strobe.  Like software triggered strobe, but
 | ||||
|    only starts the counter when the gate pin is set to high.  */ | ||||
| #define PIT_CTRL_HARDSTROBE	0x0a | ||||
| 
 | ||||
| /* Count mode.  */ | ||||
| #define PIT_CTRL_COUNT_MASK	0x01 | ||||
| #define PIT_CTRL_COUNT_BINARY	0x00	/* 16-bit binary counter.  */ | ||||
| #define PIT_CTRL_COUNT_BCD	0x01	/* 4-decade BCD counter.  */ | ||||
| 
 | ||||
| #define T_REST			((grub_uint16_t) 0) | ||||
| #define T_FINE			((grub_uint16_t) -1) | ||||
| 
 | ||||
| struct note | ||||
| { | ||||
|   grub_uint16_t pitch; | ||||
|   grub_uint16_t duration; | ||||
| }; | ||||
| 
 | ||||
| static void | ||||
| beep_off (void) | ||||
| { | ||||
|   unsigned char status; | ||||
| 
 | ||||
|   status = grub_inb (SPEAKER); | ||||
|   grub_outb (status & ~(SPEAKER_TMR2 | SPEAKER_DATA), SPEAKER); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| beep_on (grub_uint16_t pitch) | ||||
| { | ||||
|   unsigned char status; | ||||
|   unsigned int counter; | ||||
| 
 | ||||
|   if (pitch < 20) | ||||
|     pitch = 20; | ||||
|   else if (pitch > 20000) | ||||
|     pitch = 20000; | ||||
| 
 | ||||
|   counter = PIT_FREQUENCY / pitch; | ||||
| 
 | ||||
|   /* Program timer 2.  */ | ||||
|   grub_outb (PIT_CTRL_SELECT_2 | PIT_CTRL_READLOAD_WORD | ||||
| 	| PIT_CTRL_SQUAREWAVE_GEN | PIT_CTRL_COUNT_BINARY, PIT_CTRL); | ||||
|   grub_outb (counter & 0xff, PIT_COUNTER_2);		/* LSB */ | ||||
|   grub_outb ((counter >> 8) & 0xff, PIT_COUNTER_2);	/* MSB */ | ||||
| 
 | ||||
|   /* Start speaker.  */ | ||||
|   status = grub_inb (SPEAKER); | ||||
|   grub_outb (status | SPEAKER_TMR2 | SPEAKER_DATA, SPEAKER); | ||||
| } | ||||
| 
 | ||||
| /* Returns whether playing should continue.  */ | ||||
| static int | ||||
| play (unsigned tempo, struct note *note) | ||||
| { | ||||
|   unsigned int to; | ||||
| 
 | ||||
|   if (note->pitch == T_FINE || grub_checkkey () >= 0) | ||||
|     return 1; | ||||
| 
 | ||||
|   grub_dprintf ("play", "pitch = %d, duration = %d\n", note->pitch, | ||||
|                 note->duration); | ||||
| 
 | ||||
|   switch (note->pitch) | ||||
|     { | ||||
|       case T_REST: | ||||
|         beep_off (); | ||||
|         break; | ||||
| 
 | ||||
|       default: | ||||
|         beep_on (note->pitch); | ||||
|         break; | ||||
|     } | ||||
| 
 | ||||
|   to = grub_get_rtc () + BASE_TEMPO * note->duration / tempo; | ||||
|   while (((unsigned int) grub_get_rtc () <= to) && (grub_checkkey () < 0)) | ||||
|     ; | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_play (grub_command_t cmd __attribute__ ((unused)), | ||||
| 	       int argc, char **args) | ||||
| { | ||||
| 
 | ||||
|   if (argc < 1) | ||||
|     return grub_error (GRUB_ERR_BAD_ARGUMENT, "file name or tempo and notes required"); | ||||
| 
 | ||||
|   if (argc == 1) | ||||
|     { | ||||
|       struct note buf; | ||||
|       grub_uint32_t tempo; | ||||
|       grub_file_t file; | ||||
| 
 | ||||
|       file = grub_file_open (args[0]); | ||||
| 
 | ||||
|       if (! file) | ||||
|         return grub_error (GRUB_ERR_FILE_NOT_FOUND, "file not found"); | ||||
| 
 | ||||
|       if (grub_file_read (file, &tempo, sizeof (tempo)) != sizeof (tempo)) | ||||
|         { | ||||
|           grub_file_close (file); | ||||
|           return grub_error (GRUB_ERR_FILE_READ_ERROR, | ||||
|                              "file doesn't even contains a full tempo record"); | ||||
|         } | ||||
| 
 | ||||
|       tempo = grub_le_to_cpu32 (tempo); | ||||
|       grub_dprintf ("play","tempo = %d\n", tempo); | ||||
| 
 | ||||
|       while (grub_file_read (file, &buf, | ||||
|                              sizeof (struct note)) == sizeof (struct note)) | ||||
|         { | ||||
|           buf.pitch = grub_le_to_cpu16 (buf.pitch); | ||||
|           buf.duration = grub_le_to_cpu16 (buf.duration); | ||||
| 
 | ||||
|           if (play (tempo, &buf)) | ||||
|             break; | ||||
|         } | ||||
| 
 | ||||
|       grub_file_close (file); | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       char *end; | ||||
|       unsigned tempo; | ||||
|       struct note note; | ||||
|       int i; | ||||
| 
 | ||||
|       tempo = grub_strtoul (args[0], &end, 0); | ||||
| 
 | ||||
|       if (*end) | ||||
|         /* Was not a number either, assume it was supposed to be a file name.  */ | ||||
|         return grub_error (GRUB_ERR_FILE_NOT_FOUND, "file not found"); | ||||
| 
 | ||||
|       grub_dprintf ("play","tempo = %d\n", tempo); | ||||
| 
 | ||||
|       for (i = 1; i + 1 < argc; i += 2) | ||||
|         { | ||||
|           note.pitch = grub_strtoul (args[i], &end, 0); | ||||
|           if (*end) | ||||
|             { | ||||
|               grub_error (GRUB_ERR_BAD_NUMBER, "bogus pitch number"); | ||||
|               break; | ||||
|             } | ||||
| 
 | ||||
|           note.duration = grub_strtoul (args[i + 1], &end, 0); | ||||
|           if (*end) | ||||
|             { | ||||
|               grub_error (GRUB_ERR_BAD_NUMBER, "bogus duration number"); | ||||
|               break; | ||||
|             } | ||||
| 
 | ||||
|           if (play (tempo, ¬e)) | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|   beep_off (); | ||||
| 
 | ||||
|   while (grub_checkkey () > 0) | ||||
|     grub_getkey (); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static grub_command_t cmd; | ||||
|  | ||||
| GRUB_MOD_INIT(play) | ||||
| { | ||||
|   cmd = grub_register_command ("play", grub_cmd_play, | ||||
| 			       N_("FILE | TEMPO [PITCH1 DURATION1] [PITCH2 DURATION2] ... "), | ||||
| 			       N_("Play a tune.")); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(play) | ||||
| { | ||||
|   grub_unregister_command (cmd); | ||||
| } | ||||
							
								
								
									
										52
									
								
								grub-core/commands/i386/pc/pxecmd.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								grub-core/commands/i386/pc/pxecmd.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,52 @@ | |||
| /* pxe.c - command to control the pxe driver  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2008,2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/err.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/machine/pxe.h> | ||||
| #include <grub/command.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_pxe_unload (grub_command_t cmd __attribute__ ((unused)), | ||||
| 		     int argc __attribute__ ((unused)), | ||||
| 		     char **args __attribute__ ((unused))) | ||||
| { | ||||
|   if (! grub_pxe_pxenv) | ||||
|     return grub_error (GRUB_ERR_FILE_NOT_FOUND, "no pxe environment"); | ||||
| 
 | ||||
|   grub_pxe_unload (); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static grub_command_t cmd; | ||||
| 
 | ||||
| GRUB_MOD_INIT(pxecmd) | ||||
| { | ||||
|   cmd = grub_register_command ("pxe_unload", grub_cmd_pxe_unload, | ||||
| 			       0, | ||||
| 			       N_("Unload PXE environment.")); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(pxecmd) | ||||
| { | ||||
|   grub_unregister_command (cmd); | ||||
| } | ||||
							
								
								
									
										383
									
								
								grub-core/commands/i386/pc/sendkey.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										383
									
								
								grub-core/commands/i386/pc/sendkey.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,383 @@ | |||
| /* sendkey.c - fake keystroke. */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  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. | ||||
|  * | ||||
|  *  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, write to the Free Software | ||||
|  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/types.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/mm.h> | ||||
| #include <grub/err.h> | ||||
| #include <grub/dl.h> | ||||
| #include <grub/extcmd.h> | ||||
| #include <grub/cpu/io.h> | ||||
| #include <grub/loader.h> | ||||
| 
 | ||||
| static char sendkey[0x20]; | ||||
| /* Length of sendkey.  */ | ||||
| static int keylen = 0; | ||||
| static int noled = 0; | ||||
| static const struct grub_arg_option options[] = | ||||
|   { | ||||
|     {"num", 'n', 0, "set numlock mode", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"caps", 'c', 0, "set capslock mode", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"scroll", 's', 0, "set scrolllock mode", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"insert", 0, 0, "set insert mode", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"pause", 0, 0, "set pause mode", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"left-shift", 0, 0, "press left shift", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"right-shift", 0, 0, "press right shift", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"sysrq", 0, 0, "press SysRq", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"numkey", 0, 0, "press NumLock key", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"capskey", 0, 0, "press CapsLock key", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"scrollkey", 0, 0, "press ScrollLock key", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"insertkey", 0, 0, "press Insert key", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"left-alt", 0, 0, "press left alt", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"right-alt", 0, 0, "press right alt", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"left-ctrl", 0, 0, "press left ctrl", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"right-ctrl", 0, 0, "press right ctrl", "[on|off]", ARG_TYPE_STRING}, | ||||
|     {"no-led", 0, 0, "don't update LED state", 0, 0}, | ||||
|     {0, 0, 0, 0, 0, 0} | ||||
|   }; | ||||
| static int simple_flag_offsets[]  | ||||
| = {5, 6, 4, 7, 11, 1, 0, 10, 13, 14, 12, 15, 9, 3, 8, 2}; | ||||
| 
 | ||||
| static grub_uint32_t andmask = 0xffffffff, ormask = 0; | ||||
| 
 | ||||
| struct  | ||||
| keysym | ||||
| { | ||||
|   char *unshifted_name;			/* the name in unshifted state */ | ||||
|   char *shifted_name;			/* the name in shifted state */ | ||||
|   unsigned char unshifted_ascii;	/* the ascii code in unshifted state */ | ||||
|   unsigned char shifted_ascii;		/* the ascii code in shifted state */ | ||||
|   unsigned char keycode;		/* keyboard scancode */ | ||||
| }; | ||||
| 
 | ||||
| /* The table for key symbols. If the "shifted" member of an entry is
 | ||||
|    NULL, the entry does not have shifted state. Copied from GRUB Legacy setkey fuction  */ | ||||
| static struct keysym keysym_table[] = | ||||
| { | ||||
|   {"escape",		0,		0x1b,	0,	0x01}, | ||||
|   {"1",			"exclam",	'1',	'!',	0x02}, | ||||
|   {"2",			"at",		'2',	'@',	0x03}, | ||||
|   {"3",			"numbersign",	'3',	'#',	0x04}, | ||||
|   {"4",			"dollar",	'4',	'$',	0x05}, | ||||
|   {"5",			"percent",	'5',	'%',	0x06}, | ||||
|   {"6",			"caret",	'6',	'^',	0x07}, | ||||
|   {"7",			"ampersand",	'7',	'&',	0x08}, | ||||
|   {"8",			"asterisk",	'8',	'*',	0x09}, | ||||
|   {"9",			"parenleft",	'9',	'(',	0x0a}, | ||||
|   {"0",			"parenright",	'0',	')',	0x0b}, | ||||
|   {"minus",		"underscore",	'-',	'_',	0x0c}, | ||||
|   {"equal",		"plus",		'=',	'+',	0x0d}, | ||||
|   {"backspace",		0,		'\b',	0,	0x0e}, | ||||
|   {"tab",		0,		'\t',	0,	0x0f}, | ||||
|   {"q",			"Q",		'q',	'Q',	0x10}, | ||||
|   {"w",			"W",		'w',	'W',	0x11}, | ||||
|   {"e",			"E",		'e',	'E',	0x12}, | ||||
|   {"r",			"R",		'r',	'R',	0x13}, | ||||
|   {"t",			"T",		't',	'T',	0x14}, | ||||
|   {"y",			"Y",		'y',	'Y',	0x15}, | ||||
|   {"u",			"U",		'u',	'U',	0x16}, | ||||
|   {"i",			"I",		'i',	'I',	0x17}, | ||||
|   {"o",			"O",		'o',	'O',	0x18}, | ||||
|   {"p",			"P",		'p',	'P',	0x19}, | ||||
|   {"bracketleft",	"braceleft",	'[',	'{',	0x1a}, | ||||
|   {"bracketright",	"braceright",	']',	'}',	0x1b}, | ||||
|   {"enter",		0,		'\r',	0,	0x1c}, | ||||
|   {"control",		0,		0,	0,	0x1d}, | ||||
|   {"a",			"A",		'a',	'A',	0x1e}, | ||||
|   {"s",			"S",		's',	'S',	0x1f}, | ||||
|   {"d",			"D",		'd',	'D',	0x20}, | ||||
|   {"f",			"F",		'f',	'F',	0x21}, | ||||
|   {"g",			"G",		'g',	'G',	0x22}, | ||||
|   {"h",			"H",		'h',	'H',	0x23}, | ||||
|   {"j",			"J",		'j',	'J',	0x24}, | ||||
|   {"k",			"K",		'k',	'K',	0x25}, | ||||
|   {"l",			"L",		'l',	'L',	0x26}, | ||||
|   {"semicolon",		"colon",	';',	':',	0x27}, | ||||
|   {"quote",		"doublequote",	'\'',	'"',	0x28}, | ||||
|   {"backquote",		"tilde",	'`',	'~',	0x29}, | ||||
|   {"shift",		0,		0,	0,	0x2a}, | ||||
|   {"backslash",		"bar",		'\\',	'|',	0x2b}, | ||||
|   {"z",			"Z",		'z',	'Z',	0x2c}, | ||||
|   {"x",			"X",		'x',	'X',	0x2d}, | ||||
|   {"c",			"C",		'c',	'C',	0x2e}, | ||||
|   {"v",			"V",		'v',	'V',	0x2f}, | ||||
|   {"b",			"B",		'b',	'B',	0x30}, | ||||
|   {"n",			"N",		'n',	'N',	0x31}, | ||||
|   {"m",			"M",		'm',	'M',	0x32}, | ||||
|   {"comma",		"less",		',',	'<',	0x33}, | ||||
|   {"period",		"greater",	'.',	'>',	0x34}, | ||||
|   {"slash",		"question",	'/',	'?',	0x35}, | ||||
|   {"rshift",		0,		0,	0,	0x36}, | ||||
|   {"numasterisk",		0,		'*',	0,	0x37}, | ||||
|   {"alt",		0,		0,	0,	0x38}, | ||||
|   {"space",		0,		' ',	0,	0x39}, | ||||
|   {"capslock",		0,		0,	0,	0x3a}, | ||||
|   {"F1",		0,		0,	0,	0x3b}, | ||||
|   {"F2",		0,		0,	0,	0x3c}, | ||||
|   {"F3",		0,		0,	0,	0x3d}, | ||||
|   {"F4",		0,		0,	0,	0x3e}, | ||||
|   {"F5",		0,		0,	0,	0x3f}, | ||||
|   {"F6",	 	0,		0,	0,	0x40}, | ||||
|   {"F7",		0,		0,	0,	0x41}, | ||||
|   {"F8",		0,		0,	0,	0x42}, | ||||
|   {"F9",		0,		0,	0,	0x43}, | ||||
|   {"F10",		0,		0,	0,	0x44}, | ||||
|   {"num7",		"numhome",		'7',	0,	0x47}, | ||||
|   {"num8",		"numup",		'8',	0,	0x48}, | ||||
|   {"num9",		"numpgup",		'9',	0,	0x49}, | ||||
|   {"numminus",		0,		'-',	0,	0x4a}, | ||||
|   {"num4",		"numleft",		'4',	0,	0x4b}, | ||||
|   {"num5",		"numcenter",		'5',	0,	0x4c}, | ||||
|   {"num6",		"numright",		'6',	0,	0x4d}, | ||||
|   {"numplus",		0,		'-',	0,	0x4e}, | ||||
|   {"num1",		"numend",		'1',	0,	0x4f}, | ||||
|   {"num2",		"numdown",		'2',	0,	0x50}, | ||||
|   {"num3",		"numpgdown",		'3',	0,	0x51}, | ||||
|   {"num0",		"numinsert",		'0',	0,	0x52}, | ||||
|   {"numperiod",	"numdelete", 0,	0x7f,		0x53}, | ||||
|   {"F11",		0,		0,	0,	0x57}, | ||||
|   {"F12",		0,		0,	0,	0x58}, | ||||
|   {"numenter",		0,		'\r',	0,	0xe0}, | ||||
|   {"numslash",		0,		'/',	0,	0xe0}, | ||||
|   {"delete",		0,		0x7f,	0,	0xe0}, | ||||
|   {"insert",		0,		0xe0,	0,	0x52}, | ||||
|   {"home",		0,		0xe0,	0,	0x47}, | ||||
|   {"end",		0,		0xe0,	0,	0x4f}, | ||||
|   {"pgdown",		0,		0xe0,	0,	0x51}, | ||||
|   {"pgup",		0,		0xe0,	0,	0x49}, | ||||
|   {"down",		0,		0xe0,	0,	0x50}, | ||||
|   {"up",		0,		0xe0,	0,	0x48}, | ||||
|   {"left",		0,		0xe0,	0,	0x4b}, | ||||
|   {"right",		0,		0xe0,	0,	0x4d} | ||||
| }; | ||||
| 
 | ||||
| /* Set a simple flag in flags variable  
 | ||||
|    OUTOFFSET - offset of flag in FLAGS, | ||||
|    OP - action id | ||||
| */ | ||||
| static void | ||||
| grub_sendkey_set_simple_flag (int outoffset, int op) | ||||
| {       | ||||
|   if (op == 2) | ||||
|     { | ||||
|       andmask |= (1 << outoffset); | ||||
|       ormask &= ~(1 << outoffset); | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       andmask &= (~(1 << outoffset)); | ||||
|       if (op == 1) | ||||
| 	ormask |= (1 << outoffset); | ||||
|       else | ||||
| 	ormask &= ~(1 << outoffset); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| grub_sendkey_parse_op (struct grub_arg_list state) | ||||
| { | ||||
|   if (! state.set) | ||||
|     return 2; | ||||
| 
 | ||||
|   if (grub_strcmp (state.arg, "off") == 0 || grub_strcmp (state.arg, "0") == 0  | ||||
|       || grub_strcmp (state.arg, "unpress") == 0) | ||||
|     return 0; | ||||
| 
 | ||||
|   if (grub_strcmp (state.arg, "on")  == 0 || grub_strcmp (state.arg, "1")  == 0 | ||||
|       || grub_strcmp (state.arg, "press") == 0) | ||||
|     return 1; | ||||
| 
 | ||||
|   return 2; | ||||
| } | ||||
| 
 | ||||
| static grub_uint32_t oldflags; | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_sendkey_postboot (void) | ||||
| { | ||||
|   /* For convention: pointer to flags.  */ | ||||
|   grub_uint32_t *flags = (grub_uint32_t *) 0x417; | ||||
| 
 | ||||
|   *flags = oldflags; | ||||
| 
 | ||||
|   *((char *) 0x41a) = 0x1e; | ||||
|   *((char *) 0x41c) = 0x1e; | ||||
| 
 | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| /* Set keyboard buffer to our sendkey  */ | ||||
| static grub_err_t | ||||
| grub_sendkey_preboot (int noret __attribute__ ((unused))) | ||||
| { | ||||
|   /* For convention: pointer to flags.  */ | ||||
|   grub_uint32_t *flags = (grub_uint32_t *) 0x417; | ||||
| 
 | ||||
|   oldflags = *flags; | ||||
|    | ||||
|   /* Set the sendkey.  */ | ||||
|   *((char *) 0x41a) = 0x1e; | ||||
|   *((char *) 0x41c) = keylen + 0x1e; | ||||
|   grub_memcpy ((char *) 0x41e, sendkey, 0x20); | ||||
| 
 | ||||
|   /* Transform "any ctrl" to "right ctrl" flag.  */ | ||||
|   if (*flags & (1 << 8)) | ||||
|     *flags &= ~(1 << 2); | ||||
| 
 | ||||
|   /* Transform "any alt" to "right alt" flag.  */ | ||||
|   if (*flags & (1 << 9)) | ||||
|     *flags &= ~(1 << 3); | ||||
|    | ||||
|   *flags = (*flags & andmask) | ormask; | ||||
| 
 | ||||
|   /* Transform "right ctrl" to "any ctrl" flag.  */ | ||||
|   if (*flags & (1 << 8)) | ||||
|     *flags |= (1 << 2); | ||||
| 
 | ||||
|   /* Transform "right alt" to "any alt" flag.  */ | ||||
|   if (*flags & (1 << 9)) | ||||
|     *flags |= (1 << 3); | ||||
| 
 | ||||
|   /* Write new LED state  */ | ||||
|   if (!noled) | ||||
|     { | ||||
|       int value = 0; | ||||
|       int failed; | ||||
|       /* Try 5 times  */ | ||||
|       for (failed = 0; failed < 5; failed++) | ||||
| 	{ | ||||
| 	  value = 0; | ||||
| 	  /* Send command change LEDs  */ | ||||
| 	  grub_outb (0xed, 0x60); | ||||
| 
 | ||||
| 	  /* Wait */ | ||||
| 	  do | ||||
| 	    value = grub_inb (0x60); | ||||
| 	  while ((value != 0xfa) && (value != 0xfe)); | ||||
| 
 | ||||
| 	  if (value == 0xfa) | ||||
| 	    { | ||||
| 	      /* Set new LEDs*/ | ||||
| 	      grub_outb ((*flags >> 4) & 7, 0x60); | ||||
| 	      break; | ||||
| 	    } | ||||
| 	} | ||||
|     } | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_sendkey (grub_extcmd_context_t ctxt, int argc, char **args) | ||||
| { | ||||
|   struct grub_arg_list *state = ctxt->state; | ||||
| 
 | ||||
|   auto int find_key_code (char *key);  | ||||
|   auto int find_ascii_code (char *key); | ||||
| 
 | ||||
|   int find_key_code (char *key) | ||||
|     { | ||||
|       unsigned i; | ||||
| 
 | ||||
|       for (i = 0; i < sizeof (keysym_table) / sizeof (keysym_table[0]); i++) | ||||
| 	{ | ||||
| 	  if (keysym_table[i].unshifted_name  | ||||
| 	      && grub_strcmp (key, keysym_table[i].unshifted_name) == 0) | ||||
| 	    return keysym_table[i].keycode; | ||||
| 	  else if (keysym_table[i].shifted_name  | ||||
| 		   && grub_strcmp (key, keysym_table[i].shifted_name) == 0) | ||||
| 	    return keysym_table[i].keycode; | ||||
| 	} | ||||
| 
 | ||||
|       return 0; | ||||
|     } | ||||
| 
 | ||||
|   int find_ascii_code (char *key) | ||||
|     { | ||||
|       unsigned i; | ||||
| 
 | ||||
|       for (i = 0; i < sizeof (keysym_table) / sizeof (keysym_table[0]); i++) | ||||
| 	{ | ||||
| 	  if (keysym_table[i].unshifted_name  | ||||
| 	      && grub_strcmp (key, keysym_table[i].unshifted_name) == 0) | ||||
| 	    return keysym_table[i].unshifted_ascii; | ||||
| 	  else if (keysym_table[i].shifted_name  | ||||
| 		   && grub_strcmp (key, keysym_table[i].shifted_name) == 0) | ||||
| 	    return keysym_table[i].shifted_ascii; | ||||
| 	} | ||||
| 
 | ||||
|       return 0; | ||||
|     } | ||||
| 
 | ||||
|   andmask = 0xffffffff; | ||||
|   ormask = 0; | ||||
| 
 | ||||
|   { | ||||
|     int i; | ||||
| 
 | ||||
|     keylen = 0; | ||||
| 
 | ||||
|     for (i = 0; i < argc && keylen < 0x20; i++) | ||||
|       { | ||||
| 	int key_code; | ||||
| 	 | ||||
| 	key_code = find_key_code (args[i]); | ||||
| 	if (key_code) | ||||
| 	  { | ||||
| 	    sendkey[keylen++] = find_ascii_code (args[i]); | ||||
| 	    sendkey[keylen++] = key_code; | ||||
| 	  } | ||||
|       } | ||||
|   } | ||||
| 
 | ||||
|   { | ||||
|     unsigned i; | ||||
|     for (i = 0; i < sizeof (simple_flag_offsets)  | ||||
| 	   / sizeof (simple_flag_offsets[0]); i++) | ||||
|       grub_sendkey_set_simple_flag (simple_flag_offsets[i],  | ||||
| 				    grub_sendkey_parse_op(state[i])); | ||||
|   } | ||||
| 
 | ||||
|   /* Set noled. */ | ||||
|   noled = (state[sizeof (simple_flag_offsets)  | ||||
| 		 / sizeof (simple_flag_offsets[0])].set); | ||||
| 
 | ||||
|   return GRUB_ERR_NONE; | ||||
| } | ||||
| 
 | ||||
| static grub_extcmd_t cmd; | ||||
| static void *preboot_hook; | ||||
| 
 | ||||
| GRUB_MOD_INIT (sendkey) | ||||
| { | ||||
|   cmd = grub_register_extcmd ("sendkey", grub_cmd_sendkey, 0, | ||||
| 			      "sendkey [KEYSTROKE1] [KEYSTROKE2] ...", | ||||
| 			      "Emulate a keystroke", options); | ||||
| 
 | ||||
|   preboot_hook  | ||||
|     = grub_loader_register_preboot_hook (grub_sendkey_preboot,  | ||||
| 					 grub_sendkey_postboot, | ||||
| 					 GRUB_LOADER_PREBOOT_HOOK_PRIO_CONSOLE); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI (sendkey) | ||||
| { | ||||
|   grub_unregister_extcmd (cmd); | ||||
|   grub_loader_unregister_preboot_hook (preboot_hook); | ||||
| } | ||||
							
								
								
									
										185
									
								
								grub-core/commands/i386/pc/vbeinfo.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										185
									
								
								grub-core/commands/i386/pc/vbeinfo.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,185 @@ | |||
| /* vbeinfo.c - command to list compatible VBE video modes.  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2005,2007,2008,2009  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/dl.h> | ||||
| #include <grub/env.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/machine/init.h> | ||||
| #include <grub/machine/vbe.h> | ||||
| #include <grub/mm.h> | ||||
| #include <grub/command.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| static void * | ||||
| real2pm (grub_vbe_farptr_t ptr) | ||||
| { | ||||
|   return (void *) ((((unsigned long) ptr & 0xFFFF0000) >> 12UL) | ||||
| 		   + ((unsigned long) ptr & 0x0000FFFF)); | ||||
| } | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_vbeinfo (grub_command_t cmd __attribute__ ((unused)), | ||||
| 		  int argc __attribute__ ((unused)), | ||||
| 		  char **args __attribute__ ((unused))) | ||||
| { | ||||
|   struct grub_vbe_info_block controller_info; | ||||
|   struct grub_vbe_mode_info_block mode_info_tmp; | ||||
|   grub_uint32_t use_mode = GRUB_VBE_DEFAULT_VIDEO_MODE; | ||||
|   grub_uint16_t *video_mode_list; | ||||
|   grub_uint16_t *p; | ||||
|   grub_uint16_t *saved_video_mode_list; | ||||
|   grub_size_t video_mode_list_size; | ||||
|   grub_err_t err; | ||||
|   char *modevar; | ||||
| 
 | ||||
|   err = grub_vbe_probe (&controller_info); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   grub_printf ("VBE info:   version: %d.%d  OEM software rev: %d.%d\n", | ||||
| 	       controller_info.version >> 8, | ||||
|                controller_info.version & 0xFF, | ||||
|                controller_info.oem_software_rev >> 8, | ||||
|                controller_info.oem_software_rev & 0xFF); | ||||
| 
 | ||||
|   /* The total_memory field is in 64 KiB units.  */ | ||||
|   grub_printf ("            total memory: %d KiB\n", | ||||
|                (controller_info.total_memory << 16) / 1024); | ||||
| 
 | ||||
|   /* Because the information on video modes is stored in a temporary place,
 | ||||
|      it is better to copy it to somewhere safe.  */ | ||||
|   p = video_mode_list = real2pm (controller_info.video_mode_ptr); | ||||
|   while (*p++ != 0xFFFF) | ||||
|     ; | ||||
| 
 | ||||
|   video_mode_list_size = (grub_addr_t) p - (grub_addr_t) video_mode_list; | ||||
|   saved_video_mode_list = grub_malloc (video_mode_list_size); | ||||
|   if (! saved_video_mode_list) | ||||
|     return grub_errno; | ||||
| 
 | ||||
|   grub_memcpy (saved_video_mode_list, video_mode_list, video_mode_list_size); | ||||
| 
 | ||||
|   grub_printf ("List of compatible video modes:\n"); | ||||
|   grub_printf ("Legend: P=Packed pixel, D=Direct color, " | ||||
| 	       "mask/pos=R/G/B/reserved\n"); | ||||
| 
 | ||||
|   /* Walk through all video modes listed.  */ | ||||
|   for (p = saved_video_mode_list; *p != 0xFFFF; p++) | ||||
|     { | ||||
|       const char *memory_model = 0; | ||||
|       grub_uint32_t mode = (grub_uint32_t) *p; | ||||
| 
 | ||||
|       err = grub_vbe_get_video_mode_info (mode, &mode_info_tmp); | ||||
|       if (err != GRUB_ERR_NONE) | ||||
| 	{ | ||||
| 	  grub_errno = GRUB_ERR_NONE; | ||||
| 	  continue; | ||||
| 	} | ||||
| 
 | ||||
|       if ((mode_info_tmp.mode_attributes & GRUB_VBE_MODEATTR_SUPPORTED) == 0) | ||||
| 	/* If not available, skip it.  */ | ||||
| 	continue; | ||||
| 
 | ||||
|       if ((mode_info_tmp.mode_attributes & GRUB_VBE_MODEATTR_RESERVED_1) == 0) | ||||
| 	/* Not enough information.  */ | ||||
| 	continue; | ||||
| 
 | ||||
|       if ((mode_info_tmp.mode_attributes & GRUB_VBE_MODEATTR_COLOR) == 0) | ||||
| 	/* Monochrome is unusable.  */ | ||||
| 	continue; | ||||
| 
 | ||||
|       if ((mode_info_tmp.mode_attributes & GRUB_VBE_MODEATTR_LFB_AVAIL) == 0) | ||||
| 	/* We support only linear frame buffer modes.  */ | ||||
| 	continue; | ||||
| 
 | ||||
|       if ((mode_info_tmp.mode_attributes & GRUB_VBE_MODEATTR_GRAPHICS) == 0) | ||||
| 	/* We allow only graphical modes.  */ | ||||
| 	continue; | ||||
| 
 | ||||
|       switch (mode_info_tmp.memory_model) | ||||
| 	{ | ||||
| 	case GRUB_VBE_MEMORY_MODEL_PACKED_PIXEL: | ||||
| 	  memory_model = "Packed"; | ||||
| 	  break; | ||||
| 	case GRUB_VBE_MEMORY_MODEL_DIRECT_COLOR: | ||||
| 	  memory_model = "Direct"; | ||||
| 	  break; | ||||
| 
 | ||||
| 	default: | ||||
| 	  break; | ||||
| 	} | ||||
| 
 | ||||
|       if (! memory_model) | ||||
| 	continue; | ||||
| 
 | ||||
|       grub_printf ("0x%03x:  %4d x %4d x %2d  %s", | ||||
|                    mode, | ||||
|                    mode_info_tmp.x_resolution, | ||||
|                    mode_info_tmp.y_resolution, | ||||
|                    mode_info_tmp.bits_per_pixel, | ||||
|                    memory_model); | ||||
| 
 | ||||
|       /* Show mask and position details for direct color modes.  */ | ||||
|       if (mode_info_tmp.memory_model == GRUB_VBE_MEMORY_MODEL_DIRECT_COLOR) | ||||
|         grub_printf (", mask: %d/%d/%d/%d  pos: %d/%d/%d/%d", | ||||
|                      mode_info_tmp.red_mask_size, | ||||
|                      mode_info_tmp.green_mask_size, | ||||
|                      mode_info_tmp.blue_mask_size, | ||||
|                      mode_info_tmp.rsvd_mask_size, | ||||
|                      mode_info_tmp.red_field_position, | ||||
|                      mode_info_tmp.green_field_position, | ||||
|                      mode_info_tmp.blue_field_position, | ||||
|                      mode_info_tmp.rsvd_field_position); | ||||
|       grub_printf ("\n"); | ||||
|     } | ||||
| 
 | ||||
|   grub_free (saved_video_mode_list); | ||||
| 
 | ||||
|   /* Check existence of vbe_mode environment variable.  */ | ||||
|   modevar = grub_env_get ("vbe_mode"); | ||||
| 
 | ||||
|   if (modevar != 0) | ||||
|     { | ||||
|       unsigned long value; | ||||
| 
 | ||||
|       value = grub_strtoul (modevar, 0, 0); | ||||
|       if (grub_errno == GRUB_ERR_NONE) | ||||
| 	use_mode = value; | ||||
|       else | ||||
| 	grub_errno = GRUB_ERR_NONE; | ||||
|     } | ||||
| 
 | ||||
|   grub_printf ("Configured VBE mode (vbe_mode) = 0x%03x\n", use_mode); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static grub_command_t cmd; | ||||
| 
 | ||||
| GRUB_MOD_INIT(vbeinfo) | ||||
| { | ||||
|   cmd = | ||||
|     grub_register_command ("vbeinfo", grub_cmd_vbeinfo, 0, | ||||
| 			   N_("List compatible VESA BIOS extension video modes.")); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(vbeinfo) | ||||
| { | ||||
|   grub_unregister_command (cmd); | ||||
| } | ||||
							
								
								
									
										178
									
								
								grub-core/commands/i386/pc/vbetest.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								grub-core/commands/i386/pc/vbetest.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,178 @@ | |||
| /* vbetest.c - command to test VESA BIOS Extension 2.0+ support.  */ | ||||
| /*
 | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 2005,2007  Free Software Foundation, Inc. | ||||
|  * | ||||
|  *  GRUB 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 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <grub/normal.h> | ||||
| #include <grub/dl.h> | ||||
| #include <grub/env.h> | ||||
| #include <grub/misc.h> | ||||
| #include <grub/term.h> | ||||
| #include <grub/machine/init.h> | ||||
| #include <grub/machine/vbe.h> | ||||
| #include <grub/err.h> | ||||
| #include <grub/i18n.h> | ||||
| 
 | ||||
| static grub_err_t | ||||
| grub_cmd_vbetest (grub_command_t cmd __attribute__ ((unused)), | ||||
| 		  int argc __attribute__ ((unused)), | ||||
| 		  char **args __attribute__ ((unused))) | ||||
| { | ||||
|   grub_err_t err; | ||||
|   char *modevar; | ||||
|   struct grub_vbe_mode_info_block mode_info; | ||||
|   struct grub_vbe_info_block controller_info; | ||||
|   grub_uint32_t use_mode = GRUB_VBE_DEFAULT_VIDEO_MODE; | ||||
|   grub_uint32_t old_mode; | ||||
|   grub_uint8_t *framebuffer = 0; | ||||
|   grub_uint32_t bytes_per_scan_line = 0; | ||||
|   unsigned char *ptr; | ||||
|   int i; | ||||
| 
 | ||||
|   grub_printf ("Probing for VESA BIOS Extension ... "); | ||||
| 
 | ||||
|   /* Check if VESA BIOS exists.  */ | ||||
|   err = grub_vbe_probe (&controller_info); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   grub_printf ("found!\n"); | ||||
| 
 | ||||
|   /* Dump out controller information.  */ | ||||
|   grub_printf ("VBE signature = %c%c%c%c\n", | ||||
| 	       controller_info.signature[0], | ||||
| 	       controller_info.signature[1], | ||||
| 	       controller_info.signature[2], | ||||
| 	       controller_info.signature[3]); | ||||
| 
 | ||||
|   grub_printf ("VBE version = %d.%d\n", | ||||
| 	       controller_info.version >> 8, | ||||
| 	       controller_info.version & 0xFF); | ||||
|   grub_printf ("OEM string ptr = %08x\n", | ||||
| 	       controller_info.oem_string_ptr); | ||||
|   grub_printf ("Total memory = %d\n", | ||||
| 	       controller_info.total_memory); | ||||
| 
 | ||||
|   err = grub_vbe_get_video_mode (&old_mode); | ||||
|   grub_printf ("Get video mode err = %04x\n", err); | ||||
| 
 | ||||
|   if (err == GRUB_ERR_NONE) | ||||
|     grub_printf ("Old video mode = %04x\n", old_mode); | ||||
|   else | ||||
|     grub_errno = GRUB_ERR_NONE; | ||||
| 
 | ||||
|   /* Check existence of vbe_mode environment variable.  */ | ||||
|   modevar = grub_env_get ("vbe_mode"); | ||||
|   if (modevar != 0) | ||||
|     { | ||||
|       unsigned long value; | ||||
| 
 | ||||
|       value = grub_strtoul (modevar, 0, 0); | ||||
|       if (grub_errno == GRUB_ERR_NONE) | ||||
| 	use_mode = value; | ||||
|       else | ||||
| 	grub_errno = GRUB_ERR_NONE; | ||||
|     } | ||||
| 
 | ||||
|   err = grub_vbe_get_video_mode_info (use_mode, &mode_info); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   /* Dump out details about the mode being tested.  */ | ||||
|   grub_printf ("mode: 0x%03x\n", | ||||
|                use_mode); | ||||
|   grub_printf ("width : %d\n", | ||||
|                mode_info.x_resolution); | ||||
|   grub_printf ("height: %d\n", | ||||
|                mode_info.y_resolution); | ||||
|   grub_printf ("memory model: %02x\n", | ||||
|                mode_info.memory_model); | ||||
|   grub_printf ("bytes/scanline: %d\n", | ||||
|                mode_info.bytes_per_scan_line); | ||||
|   grub_printf ("bytes/scanline (lin): %d\n", | ||||
|                mode_info.lin_bytes_per_scan_line); | ||||
|   grub_printf ("base address: %08x\n", | ||||
|                mode_info.phys_base_addr); | ||||
|   grub_printf ("red mask/pos: %d/%d\n", | ||||
|                mode_info.red_mask_size, | ||||
|                mode_info.red_field_position); | ||||
|   grub_printf ("green mask/pos: %d/%d\n", | ||||
|                mode_info.green_mask_size, | ||||
|                mode_info.green_field_position); | ||||
|   grub_printf ("blue mask/pos: %d/%d\n", | ||||
|                mode_info.blue_mask_size, | ||||
|                mode_info.blue_field_position); | ||||
| 
 | ||||
|   grub_printf ("Press any key to continue.\n"); | ||||
| 
 | ||||
|   grub_getkey (); | ||||
| 
 | ||||
|   /* Setup GFX mode.  */ | ||||
|   err = grub_vbe_set_video_mode (use_mode, &mode_info); | ||||
|   if (err != GRUB_ERR_NONE) | ||||
|     return err; | ||||
| 
 | ||||
|   /* Determine framebuffer address and how many bytes are in scan line.  */ | ||||
|   framebuffer = (grub_uint8_t *) mode_info.phys_base_addr; | ||||
|   ptr = framebuffer; | ||||
| 
 | ||||
|   if (controller_info.version >= 0x300) | ||||
|     { | ||||
|       bytes_per_scan_line = mode_info.lin_bytes_per_scan_line; | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       bytes_per_scan_line = mode_info.bytes_per_scan_line; | ||||
|     } | ||||
| 
 | ||||
|   /* Draw some random data to screen.  */ | ||||
|   for (i = 0; i < 256 * 256; i++) | ||||
|     { | ||||
|       ptr[i] = i & 0x0F; | ||||
|     } | ||||
| 
 | ||||
|   /* Draw white line to screen.  */ | ||||
|   for (i = 0; i < 100; i++) | ||||
|     { | ||||
|       ptr[mode_info.bytes_per_scan_line * 50 + i] = 0x0F; | ||||
|     } | ||||
| 
 | ||||
|   /* Draw another white line to screen.  */ | ||||
|   grub_memset (ptr + bytes_per_scan_line * 51, 0x0f, bytes_per_scan_line); | ||||
| 
 | ||||
|   grub_getkey (); | ||||
| 
 | ||||
|   grub_video_restore (); | ||||
| 
 | ||||
|   /* Restore old video mode.  */ | ||||
|   grub_vbe_set_video_mode (old_mode, 0); | ||||
| 
 | ||||
|   return grub_errno; | ||||
| } | ||||
| 
 | ||||
| static grub_command_t cmd; | ||||
| 
 | ||||
| GRUB_MOD_INIT(vbetest) | ||||
| { | ||||
|   cmd = grub_register_command ("vbetest", grub_cmd_vbetest, | ||||
| 			       0, N_("Test VESA BIOS Extension 2.0+ support.")); | ||||
| } | ||||
| 
 | ||||
| GRUB_MOD_FINI(vbetest) | ||||
| { | ||||
|   grub_unregister_command (cmd); | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue