| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 1 | /* -*- linux-c -*- ------------------------------------------------------- * | 
|  | 2 | * | 
|  | 3 | *   Copyright (C) 1991, 1992 Linus Torvalds | 
|  | 4 | *   Copyright 2007 rPath, Inc. - All Rights Reserved | 
|  | 5 | * | 
|  | 6 | *   Original APM BIOS checking by Stephen Rothwell, May 1994 | 
|  | 7 | *   (sfr@canb.auug.org.au) | 
|  | 8 | * | 
|  | 9 | *   This file is part of the Linux kernel, and is made available under | 
|  | 10 | *   the terms of the GNU General Public License version 2. | 
|  | 11 | * | 
|  | 12 | * ----------------------------------------------------------------------- */ | 
|  | 13 |  | 
|  | 14 | /* | 
| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 15 | * Get APM BIOS information | 
|  | 16 | */ | 
|  | 17 |  | 
|  | 18 | #include "boot.h" | 
|  | 19 |  | 
| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 20 | int query_apm_bios(void) | 
|  | 21 | { | 
|  | 22 | u16 ax, bx, cx, dx, di; | 
|  | 23 | u32 ebx, esi; | 
|  | 24 | u8 err; | 
|  | 25 |  | 
|  | 26 | /* APM BIOS installation check */ | 
|  | 27 | ax = 0x5300; | 
|  | 28 | bx = cx = 0; | 
|  | 29 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp ; setc %0" | 
|  | 30 | : "=d" (err), "+a" (ax), "+b" (bx), "+c" (cx) | 
|  | 31 | : : "esi", "edi"); | 
|  | 32 |  | 
|  | 33 | if (err) | 
|  | 34 | return -1;		/* No APM BIOS */ | 
|  | 35 |  | 
|  | 36 | if (bx != 0x504d)	/* "PM" signature */ | 
|  | 37 | return -1; | 
|  | 38 |  | 
| Mikael Pettersson | 1514ab0 | 2007-07-24 00:25:59 +0200 | [diff] [blame] | 39 | if (!(cx & 0x02))		/* 32 bits supported? */ | 
| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 40 | return -1; | 
|  | 41 |  | 
|  | 42 | /* Disconnect first, just in case */ | 
|  | 43 | ax = 0x5304; | 
| H. Peter Anvin | 1a13286 | 2007-07-23 15:37:14 -0700 | [diff] [blame] | 44 | bx = 0; | 
| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 45 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp" | 
| H. Peter Anvin | 1a13286 | 2007-07-23 15:37:14 -0700 | [diff] [blame] | 46 | : "+a" (ax), "+b" (bx) | 
|  | 47 | : : "ecx", "edx", "esi", "edi"); | 
| H. Peter Anvin | 49df18f | 2007-07-11 12:18:43 -0700 | [diff] [blame] | 48 |  | 
|  | 49 | /* Paranoia */ | 
|  | 50 | ebx = esi = 0; | 
|  | 51 | cx = dx = di = 0; | 
|  | 52 |  | 
|  | 53 | /* 32-bit connect */ | 
|  | 54 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp ; setc %6" | 
|  | 55 | : "=a" (ax), "+b" (ebx), "+c" (cx), "+d" (dx), | 
|  | 56 | "+S" (esi), "+D" (di), "=m" (err) | 
|  | 57 | : "a" (0x5303)); | 
|  | 58 |  | 
|  | 59 | boot_params.apm_bios_info.cseg = ax; | 
|  | 60 | boot_params.apm_bios_info.offset = ebx; | 
|  | 61 | boot_params.apm_bios_info.cseg_16 = cx; | 
|  | 62 | boot_params.apm_bios_info.dseg = dx; | 
|  | 63 | boot_params.apm_bios_info.cseg_len = (u16)esi; | 
|  | 64 | boot_params.apm_bios_info.cseg_16_len = esi >> 16; | 
|  | 65 | boot_params.apm_bios_info.dseg_len = di; | 
|  | 66 |  | 
|  | 67 | if (err) | 
|  | 68 | return -1; | 
|  | 69 |  | 
|  | 70 | /* Redo the installation check as the 32-bit connect; | 
|  | 71 | some BIOSes return different flags this way... */ | 
|  | 72 |  | 
|  | 73 | ax = 0x5300; | 
|  | 74 | bx = cx = 0; | 
|  | 75 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp ; setc %0" | 
|  | 76 | : "=d" (err), "+a" (ax), "+b" (bx), "+c" (cx) | 
|  | 77 | : : "esi", "edi"); | 
|  | 78 |  | 
|  | 79 | if (err || bx != 0x504d) { | 
|  | 80 | /* Failure with 32-bit connect, try to disconect and ignore */ | 
|  | 81 | ax = 0x5304; | 
|  | 82 | bx = 0; | 
|  | 83 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp" | 
|  | 84 | : "+a" (ax), "+b" (bx) | 
|  | 85 | : : "ecx", "edx", "esi", "edi"); | 
|  | 86 | return -1; | 
|  | 87 | } | 
|  | 88 |  | 
|  | 89 | boot_params.apm_bios_info.version = ax; | 
|  | 90 | boot_params.apm_bios_info.flags = cx; | 
|  | 91 | return 0; | 
|  | 92 | } | 
|  | 93 |  |