blob: ee274834ea8ba35cf0a804d8be7cf07fa0077993 [file] [log] [blame]
H. Peter Anvin49df18f2007-07-11 12:18:43 -07001/* -*- linux-c -*- ------------------------------------------------------- *
2 *
3 * Copyright (C) 1991, 1992 Linus Torvalds
4 * Copyright 2007 rPath, Inc. - All Rights Reserved
H. Peter Anvind54ea252009-04-01 18:14:26 -07005 * Copyright 2009 Intel Corporation; author H. Peter Anvin
H. Peter Anvin49df18f2007-07-11 12:18:43 -07006 *
7 * Original APM BIOS checking by Stephen Rothwell, May 1994
8 * (sfr@canb.auug.org.au)
9 *
10 * This file is part of the Linux kernel, and is made available under
11 * the terms of the GNU General Public License version 2.
12 *
13 * ----------------------------------------------------------------------- */
14
15/*
H. Peter Anvin49df18f2007-07-11 12:18:43 -070016 * Get APM BIOS information
17 */
18
19#include "boot.h"
20
H. Peter Anvin49df18f2007-07-11 12:18:43 -070021int query_apm_bios(void)
22{
H. Peter Anvind54ea252009-04-01 18:14:26 -070023 struct biosregs ireg, oreg;
H. Peter Anvin49df18f2007-07-11 12:18:43 -070024
25 /* APM BIOS installation check */
H. Peter Anvind54ea252009-04-01 18:14:26 -070026 initregs(&ireg);
27 ireg.ah = 0x53;
28 intcall(0x15, &ireg, &oreg);
H. Peter Anvin49df18f2007-07-11 12:18:43 -070029
H. Peter Anvind54ea252009-04-01 18:14:26 -070030 if (oreg.flags & X86_EFLAGS_CF)
H. Peter Anvin49df18f2007-07-11 12:18:43 -070031 return -1; /* No APM BIOS */
32
H. Peter Anvind54ea252009-04-01 18:14:26 -070033 if (oreg.bx != 0x504d) /* "PM" signature */
H. Peter Anvin49df18f2007-07-11 12:18:43 -070034 return -1;
35
H. Peter Anvind54ea252009-04-01 18:14:26 -070036 if (!(oreg.cx & 0x02)) /* 32 bits supported? */
H. Peter Anvin49df18f2007-07-11 12:18:43 -070037 return -1;
38
39 /* Disconnect first, just in case */
H. Peter Anvind54ea252009-04-01 18:14:26 -070040 ireg.al = 0x04;
41 intcall(0x15, &ireg, NULL);
H. Peter Anvin49df18f2007-07-11 12:18:43 -070042
43 /* 32-bit connect */
H. Peter Anvind54ea252009-04-01 18:14:26 -070044 ireg.al = 0x03;
45 intcall(0x15, &ireg, &oreg);
H. Peter Anvin49df18f2007-07-11 12:18:43 -070046
H. Peter Anvind54ea252009-04-01 18:14:26 -070047 boot_params.apm_bios_info.cseg = oreg.ax;
48 boot_params.apm_bios_info.offset = oreg.ebx;
49 boot_params.apm_bios_info.cseg_16 = oreg.cx;
50 boot_params.apm_bios_info.dseg = oreg.dx;
51 boot_params.apm_bios_info.cseg_len = oreg.si;
52 boot_params.apm_bios_info.cseg_16_len = oreg.hsi;
53 boot_params.apm_bios_info.dseg_len = oreg.di;
H. Peter Anvin49df18f2007-07-11 12:18:43 -070054
H. Peter Anvind54ea252009-04-01 18:14:26 -070055 if (oreg.flags & X86_EFLAGS_CF)
H. Peter Anvin49df18f2007-07-11 12:18:43 -070056 return -1;
57
58 /* Redo the installation check as the 32-bit connect;
59 some BIOSes return different flags this way... */
60
H. Peter Anvind54ea252009-04-01 18:14:26 -070061 ireg.al = 0x00;
62 intcall(0x15, &ireg, &oreg);
H. Peter Anvin49df18f2007-07-11 12:18:43 -070063
H. Peter Anvind54ea252009-04-01 18:14:26 -070064 if ((oreg.eflags & X86_EFLAGS_CF) || oreg.bx != 0x504d) {
H. Peter Anvin49df18f2007-07-11 12:18:43 -070065 /* Failure with 32-bit connect, try to disconect and ignore */
H. Peter Anvind54ea252009-04-01 18:14:26 -070066 ireg.al = 0x04;
67 intcall(0x15, &ireg, NULL);
H. Peter Anvin49df18f2007-07-11 12:18:43 -070068 return -1;
69 }
70
H. Peter Anvind54ea252009-04-01 18:14:26 -070071 boot_params.apm_bios_info.version = oreg.ax;
72 boot_params.apm_bios_info.flags = oreg.cx;
H. Peter Anvin49df18f2007-07-11 12:18:43 -070073 return 0;
74}
75