summaryrefslogtreecommitdiff
path: root/sys/arch/i386/stand/libsa/machdep.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/i386/stand/libsa/machdep.c')
-rw-r--r--sys/arch/i386/stand/libsa/machdep.c152
1 files changed, 102 insertions, 50 deletions
diff --git a/sys/arch/i386/stand/libsa/machdep.c b/sys/arch/i386/stand/libsa/machdep.c
index 8adffedd699..a2e9d907eae 100644
--- a/sys/arch/i386/stand/libsa/machdep.c
+++ b/sys/arch/i386/stand/libsa/machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: machdep.c,v 1.8 1997/08/12 21:51:30 mickey Exp $ */
+/* $OpenBSD: machdep.c,v 1.9 1997/08/13 03:23:39 mickey Exp $ */
/*
* Copyright (c) 1997 Michael Shalayeff
@@ -32,69 +32,120 @@
*
*/
+/*
+ * If you want to know the specification of APM BIOS, see the following
+ * documentations,
+ *
+ * [1] Intel Corporation and Microsoft Corporation, "Advanced Power
+ * Management, The Next Generation, Version 1.0", Feb.,1992.
+ *
+ * [2] Intel Corporation and Microsoft Corporation, "Advanced Power
+ * Management (APM) BIOS Interface Specification Revision 1.1",
+ * Sep.,1993, Intel Order Number: 241704-001, Microsoft Part
+ * Number: 781-110-X01
+ *
+ * or contact
+ *
+ * APM Support Desk (Intel Corporation, US)
+ * TEL: (800)628-8686
+ * FAX: (916)356-6100.
+ */
+
#include "libsa.h"
-#include <machine/biosvar.h>
#include <machine/apmvar.h>
+#undef APM_DISCONNECT /* XXX temp hack */
+#include <machine/biosvar.h>
#include "debug.h"
struct apm_connect_info apminfo;
-void
-machdep()
-{
#ifdef DEBUG
- *(u_int16_t*)0xb8148 = 0x4730;
+#define CKPT(c) (*(u_int16_t*)0xb8148 = 0x4700 + (c))
+#else
+#define CKPT(c) /* c */
#endif
- gateA20(1);
+
+#ifdef BOOT_APM
+static u_int
+apm_init()
+{
+ u_int detail;
+ u_int8_t f;
+ __asm __volatile(DOINT(0x15) "\n\t"
+ "setc %b0\n\t"
+ "shll $16, %%ecx\n\t"
+ "movzwl %%ax, %1\n\t"
+ "orl %%ecx, %1"
+ : "=d" (f), "=a" (detail)
+ : "1" (APM_INSTCHECK), "b" (PMDV_APMBIOS)
+ : "%ecx", "cc");
+ if (f || BIOS_regs.biosr_bx != 0x504d /* "PM" */ ) {
#ifdef DEBUG
- *(u_int16_t*)0xb8148 = 0x4731;
+ printf("apm_init: %x, %x, %x\n", f, BIOS_regs.biosr_bx, detail);
#endif
- debug_init();
-#ifdef DEBUG
- *(u_int16_t*)0xb8148 = 0x4732;
+ return 0;
+ } else
+ return detail;
+}
+
+static __inline int
+apm_disconnect() {
+ register u_int16_t rv;
+ __asm __volatile(DOINT(0x15) "\n\t"
+ "setc %b0"
+ : "=a" (rv)
+ : "a" (APM_DISCONNECT), "b" (PMDV_APMBIOS)
+ : "%ecx", "%edx", "cc");
+ return (rv & 0xff)? rv >> 8 : 0;
+}
+
+static __inline int
+apm_connect(struct apm_connect_info *apminfo)
+{
+ register u_int16_t f;
+ __asm __volatile (DOINT(0x15) "\n\t"
+ "setc %b0\n\t"
+ "movb %%ah, %h0\n\t"
+ "movzwl %%ax, %%eax\n\tshll $4, %1\n\t"
+ "movzwl %%cx, %%ecx\n\tshll $4, %2\n\t"
+ "movzwl %%dx, %%edx\n\tshll $4, %3\n\t"
+ : "=b" (f),
+ "=a" (apminfo->apm_code32_seg_base),
+ "=c" (apminfo->apm_code16_seg_base),
+ "=d" (apminfo->apm_data_seg_base)
+ : "a" (APM_PROT32CONNECT), "b" (PMDV_APMBIOS)
+ : "cc");
+ apminfo->apm_entrypt = BIOS_regs.biosr_bx;
+#if 0
+ apminfo->apm_code32_seg_len = BIOS_regs.biosr_si & 0xffff;
+ apminfo->apm_data_seg_len = BIOS_regs.biosr_di & 0xffff;
+#else
+ apminfo->apm_code32_seg_len = 0x10000;
+ apminfo->apm_data_seg_len = 0x10000;
#endif
- cninit(); /* call console init before doing any io */
-#ifdef DEBUG
- *(u_int16_t*)0xb8148 = 0x4733;
+ return (f & 0xff)? f >> 8 : 0;
+}
#endif
+
+void
+machdep()
+{
+ /* here */ CKPT('0');
+ gateA20(1); CKPT('1');
+ debug_init(); CKPT('2');
+ /* call console init before doing any io */
+ cninit(); CKPT('3');
#ifndef _TEST
- memprobe();
+ memprobe(); CKPT('4');
#endif
-#ifdef DEBUG
- *(u_int16_t*)0xb8148 = 0x4f34;
-#endif
-#ifdef BOOT_APM
- printf("apm_init: ");
- switch(apminfo.apm_detail = apm_init()) {
- case APMINI_CANTFIND:
- printf("not supported");
- break;
-
- case APMINI_NOT32BIT:
- printf("no 32 bit interface");
- break;
-
- case APMINI_CONNECTERR:
- printf("connect error");
- break;
- case APMINI_BADVER:
- printf("bad version");
- break;
+#ifdef BOOT_APM
+ if ((apminfo.apm_detail = apm_init())) {
- default:
- /* valid: detail, dx, bx */
- apminfo.apm_code32_seg_base = (BIOS_regs.biosr_ax & 0xffff)<< 4;
- apminfo.apm_code16_seg_base = (BIOS_regs.biosr_cx & 0xffff)<< 4;
- apminfo.apm_data_seg_base = (BIOS_regs.biosr_dx & 0xffff)<< 4;
-#if 0
- apminfo.apm_code32_seg_len = BIOS_regs.biosr_si & 0xffff;
- apminfo.apm_data_seg_len = BIOS_regs.biosr_di & 0xffff;
-#else
- apminfo.apm_code32_seg_len = 0x10000;
- apminfo.apm_data_seg_len = 0x10000;
-#endif
- apminfo.apm_entrypt = BIOS_regs.biosr_bx;
+ printf("apm: ");
+ apm_disconnect();
+ if (apm_connect(&apminfo) != 0)
+ printf("connect error\n");
#ifdef DEBUG
printf("%x text=%x/%x[%x] data=%x[%x] @ %x",
apminfo.apm_detail,
@@ -105,9 +156,10 @@ machdep()
apminfo.apm_data_seg_len,
apminfo.apm_entrypt);
#else
- printf("APM detected");
+ printf("present");
#endif
+ putchar('\n');
}
- putchar('\n');
#endif
+ CKPT('9');
}