Annotation of XNU/pexpert/i386/kd.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /*
                     23:  * @OSF_COPYRIGHT@
                     24:  */
                     25: /* 
                     26:  */
                     27:  
                     28: /* 
                     29:  *     Olivetti Mach Console driver v0.0
                     30:  *     Copyright Ing. C. Olivetti & C. S.p.A. 1988, 1989
                     31:  *     All rights reserved.
                     32:  *
                     33:  */ 
                     34: /*
                     35:  *   Copyright 1988, 1989 by Olivetti Advanced Technology Center, Inc.,
                     36:  * Cupertino, California.
                     37:  * 
                     38:  *             All Rights Reserved
                     39:  * 
                     40:  *   Permission to use, copy, modify, and distribute this software and
                     41:  * its documentation for any purpose and without fee is hereby
                     42:  * granted, provided that the above copyright notice appears in all
                     43:  * copies and that both the copyright notice and this permission notice
                     44:  * appear in supporting documentation, and that the name of Olivetti
                     45:  * not be used in advertising or publicity pertaining to distribution
                     46:  * of the software without specific, written prior permission.
                     47:  * 
                     48:  *   OLIVETTI DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
                     49:  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS,
                     50:  * IN NO EVENT SHALL OLIVETTI BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
                     51:  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
                     52:  * LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
                     53:  * NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUR OF OR IN CONNECTION
                     54:  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
                     55:  *
                     56:  * 
                     57:  *   Copyright 1988, 1989 by Intel Corporation, Santa Clara, California.
                     58:  * 
                     59:  *             All Rights Reserved
                     60:  * 
                     61:  * Permission to use, copy, modify, and distribute this software and
                     62:  * its documentation for any purpose and without fee is hereby
                     63:  * granted, provided that the above copyright notice appears in all
                     64:  * copies and that both the copyright notice and this permission notice
                     65:  * appear in supporting documentation, and that the name of Intel
                     66:  * not be used in advertising or publicity pertaining to distribution
                     67:  * of the software without specific, written prior permission.
                     68:  * 
                     69:  * INTEL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
                     70:  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS,
                     71:  * IN NO EVENT SHALL INTEL BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
                     72:  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
                     73:  * LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
                     74:  * NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
                     75:  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
                     76:  */
                     77: 
                     78: /* $ Header:  $ */
                     79: 
                     80: #include <string.h>
                     81: #include "kd.h"
                     82: 
                     83: #include       <mach/i386/vm_param.h>
                     84: 
                     85: #define at386_io_lock_state()  
                     86: #define at386_io_lock(op)      (TRUE)
                     87: #define at386_io_unlock()
                     88: 
                     89: 
                     90: typedef unsigned short i386_ioport_t;
                     91: 
                     92: /* read a byte */
                     93: extern unsigned char   inb(
                     94:                                i386_ioport_t   port);
                     95: /* write a longword */
                     96: extern void            outb(
                     97:                                i386_ioport_t   port,
                     98:                                unsigned char   datum);
                     99: 
                    100: extern __inline__ unsigned char inb(
                    101:                                i386_ioport_t port)
                    102: {
                    103:        unsigned char datum;
                    104:        __asm__ volatile("inb %1, %0" : "=a" (datum) : "d" (port));
                    105:        return(datum);
                    106: }
                    107: 
                    108: extern __inline__ void outb(
                    109:                                i386_ioport_t port,
                    110:                                unsigned char datum)
                    111: {
                    112:        __asm__ volatile("outb %0, %1" : : "a" (datum), "d" (port));
                    113: }
                    114: 
                    115: /* Forward */
                    116: 
                    117: extern void kd_sendcmd(unsigned char           ch);
                    118: extern void kdreboot(void);
                    119: extern int             kd_dogetc(int           wait);
                    120: extern void            kd_handle_ack(void);
                    121: extern void            kd_resend(void);
                    122: extern int             do_modifier(
                    123:                                int                     state,
                    124:                                Scancode                c,
                    125:                                int             up);
                    126: extern int     kdcheckmagic(
                    127:                                Scancode                sc,
                    128:                                int                     * regs);
                    129: extern int             kdstate2idx(
                    130:                                int                     state,
                    131:                                int             extended);
                    132: extern void            kdinit(void);
                    133: extern void            kd_belloff(void);
                    134: extern void            kd_bellon(void);
                    135: extern void            kd_senddata(unsigned char c);
                    136: extern unsigned char           kd_getdata(void);
                    137: extern unsigned char           kd_cmdreg_read(void);
                    138: extern void            set_kd_state(
                    139:                                int                     newstate);
                    140: extern unsigned char           state2leds(
                    141:                                int                     state);
                    142: extern void            kd_setleds1(
                    143:                                unsigned char                   val);
                    144: extern void            kd_setleds2(void);
                    145: extern void            cnsetleds(
                    146:                                unsigned char                   val);
                    147: extern int             kd_kbd_magic(
                    148:                                int                     scancode);
                    149: 
                    150: extern int             cngetc(void);
                    151: extern int             cnmaygetc(void);
                    152: extern void kdreboot(void);
                    153: extern int kd_dogetc(int       wait);
                    154: 
                    155: /* reboot on CTL-ALT-DEL ? */
                    156: extern int     rebootflag;
                    157: /* enter kernel debugger on CTR-ALT-d ? */
                    158: int            kbdkdbflag = 1;
                    159: /* allow keyboard mouse ? */
                    160: int            kbdmouseflag = 0;
                    161: 
                    162: /* 
                    163:  * kd_state shows the state of the modifier keys (ctrl, caps lock, 
                    164:  * etc.)  It should normally be changed by calling set_kd_state(), so
                    165:  * that the keyboard status LEDs are updated correctly.
                    166:  */
                    167: int    kd_state        = KS_NORMAL;
                    168: int    kb_mode         = KB_ASCII;     /* event/ascii */
                    169: 
                    170: int kd_kbd_mouse = 0;
                    171: int kd_kbd_magic_scale = 6;
                    172: int kd_kbd_magic_button  = 0;
                    173: 
                    174: /* 
                    175:  * Some keyboard commands work by sending a command, waiting for an 
                    176:  * ack (handled by kdintr), then sending data, which generates a 
                    177:  * second ack.  If we are in the middle of such a sequence, kd_ack
                    178:  * shows what the ack is for.
                    179:  * 
                    180:  * When a byte is sent to the keyboard, it is kept around in last_sent 
                    181:  * in case it needs to be resent.
                    182:  * 
                    183:  * The rest of the variables here hold the data required to complete
                    184:  * the sequence.
                    185:  * 
                    186:  * XXX - the System V driver keeps a command queue, I guess in case we
                    187:  * want to start a command while another is in progress.  Is this
                    188:  * something we should worry about?
                    189:  */
                    190: enum why_ack {NOT_WAITING, SET_LEDS, DATA_ACK};
                    191: enum why_ack   kd_ack  = NOT_WAITING;
                    192: 
                    193: unsigned char last_sent = 0;
                    194: 
                    195: unsigned char  kd_nextled      = 0;
                    196: 
                    197: /*
                    198:  * We don't provide any mutex protection for this flag because we know
                    199:  * that this module will have been initialized by the time multiple
                    200:  * threads are running.
                    201:  */
                    202: int kd_initialized     = FALSE;        /* driver initialized? */
                    203: int kd_extended        = FALSE;
                    204: 
                    205: /*
                    206:  * This array maps scancodes to Ascii characters (or character
                    207:  * sequences). 
                    208:  * Each row corresponds to one key.  There are NUMOUTPUT bytes per key
                    209:  * state.  The states are ordered: Normal, SHIFT, CTRL, ALT,
                    210:  * SHIFT/ALT.
                    211:  */
                    212: unsigned char  key_map[NUMKEYS][WIDTH_KMAP] = {
                    213: {NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC},
                    214: {K_ESC,NC,NC, K_ESC,NC,NC, K_ESC,NC,NC, K_ESC,NC,NC, K_ESC,NC,NC},
                    215: {K_ONE,NC,NC, K_BANG,NC,NC, K_ONE,NC,NC, 0x1b,0x4e,0x31, 0x1b,0x4e,0x21},
                    216: {K_TWO,NC,NC, K_ATSN,NC,NC, K_NUL,NC,NC, 0x1b,0x4e,0x32, 0x1b,0x4e,0x40},
                    217: {K_THREE,NC,NC, K_POUND,NC,NC, K_THREE,NC,NC, 0x1b,0x4e,0x33, 0x1b,0x4e,0x23},
                    218: {K_FOUR,NC,NC, K_DOLLAR,NC,NC, K_FOUR,NC,NC, 0x1b,0x4e,0x34, 0x1b,0x4e,0x24},
                    219: {K_FIVE,NC,NC, K_PERC,NC,NC, K_FIVE,NC,NC, 0x1b,0x4e,0x35, 0x1b,0x4e,0x25},
                    220: {K_SIX,NC,NC, K_CARET,NC,NC, K_RS,NC,NC, 0x1b,0x4e,0x36, 0x1b,0x4e,0x5e},
                    221: {K_SEVEN,NC,NC, K_AMPER,NC,NC, K_SEVEN,NC,NC, 0x1b,0x4e,0x37, 0x1b,0x4e,0x26},
                    222: {K_EIGHT,NC,NC, K_ASTER,NC,NC, K_EIGHT,NC,NC, 0x1b,0x4e,0x38, 0x1b,0x4e,0x2a},
                    223: {K_NINE,NC,NC, K_LPAREN,NC,NC, K_NINE,NC,NC, 0x1b,0x4e,0x39,0x1b,0x4e,0x28},
                    224: {K_ZERO,NC,NC, K_RPAREN,NC,NC, K_ZERO,NC,NC, 0x1b,0x4e,0x30,0x1b,0x4e,0x29},
                    225: {K_MINUS,NC,NC, K_UNDSC,NC,NC, K_US,NC,NC, 0x1b,0x4e,0x2d, 0x1b,0x4e,0x5f},
                    226: {K_EQL,NC,NC, K_PLUS,NC,NC, K_EQL,NC,NC, 0x1b,0x4e,0x3d, 0x1b,0x4e,0x2b},
                    227: {K_BS,NC,NC, K_BS,NC,NC, K_BS,NC,NC, K_BS,NC,NC, K_BS,NC,NC},
                    228: {K_HT,NC,NC, K_GS,NC,NC, K_HT,NC,NC, K_HT,NC,NC, K_GS,NC,NC},
                    229: {K_q,NC,NC, K_Q,NC,NC, K_DC1,NC,NC, 0x1b,0x4e,0x71, 0x1b,0x4e,0x51},
                    230: {K_w,NC,NC, K_W,NC,NC, K_ETB,NC,NC, 0x1b,0x4e,0x77, 0x1b,0x4e,0x57},
                    231: {K_e,NC,NC, K_E,NC,NC, K_ENQ,NC,NC, 0x1b,0x4e,0x65, 0x1b,0x4e,0x45},
                    232: {K_r,NC,NC, K_R,NC,NC, K_DC2,NC,NC, 0x1b,0x4e,0x72, 0x1b,0x4e,0x52},
                    233: {K_t,NC,NC, K_T,NC,NC, K_DC4,NC,NC, 0x1b,0x4e,0x74, 0x1b,0x4e,0x54},
                    234: {K_y,NC,NC, K_Y,NC,NC, K_EM,NC,NC, 0x1b,0x4e,0x79, 0x1b,0x4e,0x59},
                    235: {K_u,NC,NC, K_U,NC,NC, K_NAK,NC,NC, 0x1b,0x4e,0x75, 0x1b,0x4e,0x55},
                    236: {K_i,NC,NC, K_I,NC,NC, K_HT,NC,NC, 0x1b,0x4e,0x69, 0x1b,0x4e,0x49},
                    237: {K_o,NC,NC, K_O,NC,NC, K_SI,NC,NC, 0x1b,0x4e,0x6f, 0x1b,0x4e,0x4f},
                    238: {K_p,NC,NC, K_P,NC,NC, K_DLE,NC,NC, 0x1b,0x4e,0x70, 0x1b,0x4e,0x50},
                    239: {K_LBRKT,NC,NC, K_LBRACE,NC,NC, K_ESC,NC,NC, 0x1b,0x4e,0x5b, 0x1b,0x4e,0x7b},
                    240: {K_RBRKT,NC,NC, K_RBRACE,NC,NC, K_GS,NC,NC, 0x1b,0x4e,0x5d, 0x1b,0x4e,0x7d},
                    241: {K_CR,NC,NC, K_CR,NC,NC, K_CR,NC,NC, K_CR,NC,NC, K_CR,NC,NC},
                    242: {K_SCAN,K_CTLSC,NC, K_SCAN,K_CTLSC,NC, K_SCAN,K_CTLSC,NC, K_SCAN,K_CTLSC,NC, 
                    243:         K_SCAN,K_CTLSC,NC},
                    244: {K_a,NC,NC, K_A,NC,NC, K_SOH,NC,NC, 0x1b,0x4e,0x61, 0x1b,0x4e,0x41},
                    245: {K_s,NC,NC, K_S,NC,NC, K_DC3,NC,NC, 0x1b,0x4e,0x73, 0x1b,0x4e,0x53},
                    246: {K_d,NC,NC, K_D,NC,NC, K_EOT,NC,NC, 0x1b,0x4e,0x65, 0x1b,0x4e,0x45},
                    247: {K_f,NC,NC, K_F,NC,NC, K_ACK,NC,NC, 0x1b,0x4e,0x66, 0x1b,0x4e,0x46},
                    248: {K_g,NC,NC, K_G,NC,NC, K_BEL,NC,NC, 0x1b,0x4e,0x67, 0x1b,0x4e,0x47},
                    249: {K_h,NC,NC, K_H,NC,NC, K_BS,NC,NC, 0x1b,0x4e,0x68, 0x1b,0x4e,0x48},
                    250: {K_j,NC,NC, K_J,NC,NC, K_LF,NC,NC, 0x1b,0x4e,0x6a, 0x1b,0x4e,0x4a},
                    251: {K_k,NC,NC, K_K,NC,NC, K_VT,NC,NC, 0x1b,0x4e,0x6b, 0x1b,0x4e,0x4b},
                    252: {K_l,NC,NC, K_L,NC,NC, K_FF,NC,NC, 0x1b,0x4e,0x6c, 0x1b,0x4e,0x4c},
                    253: {K_SEMI,NC,NC, K_COLON,NC,NC, K_SEMI,NC,NC, 0x1b,0x4e,0x3b, 0x1b,0x4e,0x3a},
                    254: {K_SQUOTE,NC,NC,K_DQUOTE,NC,NC,K_SQUOTE,NC,NC,0x1b,0x4e,0x27,0x1b,0x4e,0x22},
                    255: {K_GRAV,NC,NC, K_TILDE,NC,NC, K_RS,NC,NC, 0x1b,0x4e,0x60, 0x1b,0x4e,0x7e},
                    256: {K_SCAN,K_LSHSC,NC, K_SCAN,K_LSHSC,NC, K_SCAN,K_LSHSC,NC, K_SCAN,K_LSHSC,NC,
                    257:         K_SCAN,K_LSHSC,NC},
                    258: {K_BSLSH,NC,NC, K_PIPE,NC,NC, K_FS,NC,NC, 0x1b,0x4e,0x5c, 0x1b,0x4e,0x7c},
                    259: {K_z,NC,NC, K_Z,NC,NC, K_SUB,NC,NC, 0x1b,0x4e,0x7a, 0x1b,0x4e,0x5a},
                    260: {K_x,NC,NC, K_X,NC,NC, K_CAN,NC,NC, 0x1b,0x4e,0x78, 0x1b,0x4e,0x58},
                    261: {K_c,NC,NC, K_C,NC,NC, K_ETX,NC,NC, 0x1b,0x4e,0x63, 0x1b,0x4e,0x43},
                    262: {K_v,NC,NC, K_V,NC,NC, K_SYN,NC,NC, 0x1b,0x4e,0x76, 0x1b,0x4e,0x56},
                    263: {K_b,NC,NC, K_B,NC,NC, K_STX,NC,NC, 0x1b,0x4e,0x62, 0x1b,0x4e,0x42},
                    264: {K_n,NC,NC, K_N,NC,NC, K_SO,NC,NC, 0x1b,0x4e,0x6e, 0x1b,0x4e,0x4e},
                    265: {K_m,NC,NC, K_M,NC,NC, K_CR,NC,NC, 0x1b,0x4e,0x6d, 0x1b,0x4e,0x4d},
                    266: {K_COMMA,NC,NC, K_LTHN,NC,NC, K_COMMA,NC,NC, 0x1b,0x4e,0x2c, 0x1b,0x4e,0x3c},
                    267: {K_PERIOD,NC,NC, K_GTHN,NC,NC, K_PERIOD,NC,NC,0x1b,0x4e,0x2e,0x1b,0x4e,0x3e},
                    268: {K_SLASH,NC,NC, K_QUES,NC,NC, K_SLASH,NC,NC, 0x1b,0x4e,0x2f, 0x1b,0x4e,0x3f},
                    269: {K_SCAN,K_RSHSC,NC, K_SCAN,K_RSHSC,NC, K_SCAN,K_RSHSC,NC, K_SCAN,K_RSHSC,NC, 
                    270:         K_SCAN,K_RSHSC,NC},
                    271: {K_ASTER,NC,NC, K_ASTER,NC,NC, K_ASTER,NC,NC, 0x1b,0x4e,0x2a,0x1b,0x4e,0x2a},
                    272: {K_SCAN,K_ALTSC,NC, K_SCAN,K_ALTSC,NC, K_SCAN,K_ALTSC,NC, K_SCAN,K_ALTSC,NC, 
                    273:         K_SCAN,K_ALTSC,NC},
                    274: {K_SPACE,NC,NC, K_SPACE,NC,NC, K_NUL,NC,NC, K_SPACE,NC,NC, K_SPACE,NC,NC},
                    275: {K_SCAN,K_CLCKSC,NC, K_SCAN,K_CLCKSC,NC, K_SCAN,K_CLCKSC,NC, 
                    276:         K_SCAN,K_CLCKSC,NC, K_SCAN,K_CLCKSC,NC},
                    277: {K_F1, K_F1S, K_F1, K_F1, K_F1S},
                    278: {K_F2, K_F2S, K_F2, K_F2, K_F2S},
                    279: {K_F3, K_F3S, K_F3, K_F3, K_F3S},
                    280: {K_F4, K_F4S, K_F4, K_F4, K_F4S},
                    281: {K_F5, K_F5S, K_F5, K_F5, K_F5S},
                    282: {K_F6, K_F6S, K_F6, K_F6, K_F6S},
                    283: {K_F7, K_F7S, K_F7, K_F7, K_F7S},
                    284: {K_F8, K_F8S, K_F8, K_F8, K_F8S},
                    285: {K_F9, K_F9S, K_F9, K_F9, K_F9S},
                    286: {K_F10, K_F10S, K_F10, K_F10, K_F10S},
                    287: {K_SCAN,K_NLCKSC,NC, K_SCAN,K_NLCKSC,NC, K_SCAN,K_NLCKSC,NC, 
                    288:         K_SCAN,K_NLCKSC,NC, K_SCAN,K_NLCKSC,NC},
                    289: {K_SCRL, K_NUL,NC,NC, K_SCRL, K_SCRL, K_NUL,NC,NC},
                    290: {K_HOME, K_SEVEN,NC,NC, K_HOME, K_HOME, 0x1b,0x4e,0x37},
                    291: {K_UA, K_EIGHT,NC,NC, K_UA, K_UA, 0x1b,0x4e,0x38},
                    292: {K_PUP, K_NINE,NC,NC, K_PUP, K_PUP, 0x1b,0x4e,0x39},
                    293: {0x1b,0x5b,0x53, K_MINUS,NC,NC, 0x1b,0x5b,0x53,0x1b,0x5b,0x53,0x1b,0x4e,0x2d},
                    294: {K_LA, K_FOUR,NC,NC, K_LA, K_LA, 0x1b,0x4e,0x34},
                    295: {0x1b,0x5b,0x47,K_FIVE,NC,NC,0x1b,0x5b,0x47, 0x1b,0x5b,0x47, 0x1b,0x4e,0x35},
                    296: {K_RA, K_SIX,NC,NC, K_RA, K_RA, 0x1b,0x4e,0x36},
                    297: {0x1b,0x5b,0x54,K_PLUS,NC,NC, 0x1b,0x5b,0x54, 0x1b,0x5b,0x54, 0x1b,0x4e,0x2b},
                    298: {K_END, K_ONE,NC,NC, K_END, K_END, 0x1b,0x4e,0x31},
                    299: {K_DA, K_TWO,NC,NC, K_DA, K_DA, 0x1b,0x4e,0x32},
                    300: {K_PDN, K_THREE,NC,NC, K_PDN, K_PDN, 0x1b,0x4e,0x33},
                    301: {K_INS, K_ZERO,NC,NC, K_INS, K_INS, 0x1b,0x4e,0x30},
                    302: {K_DEL,NC,NC, K_PERIOD,NC,NC, K_DEL,NC,NC, K_DEL,NC,NC, 0x1b,0x4e,0x2e},
                    303: {NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC},
                    304: {NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC},
                    305: {NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC,NC},
                    306: {K_F11, K_F11S, K_F11, K_F11, K_F11S},
                    307: {K_F12, K_F12S, K_F12, K_F12, K_F12S}
                    308: };
                    309: 
                    310: extern void cnputc(unsigned char ch);
                    311: 
                    312: /*
                    313:  * Switch for poll vs. interrupt.
                    314:  */
                    315: int    kd_pollc = 0;
                    316: 
                    317: int (*cgetc)(
                    318:        int     wait) = kd_dogetc;
                    319:                                        /* get a char. from console     */
                    320: void (*cputc)(
                    321:        char            ch) = cnputc;
                    322:                                        /* put a char. to console       */
                    323: 
                    324: /* 
                    325:  * cngetc:
                    326:  * 
                    327:  *     Get one character using polling, rather than interrupts.  Used 
                    328:  *     by the kernel debugger.  Note that Caps Lock is ignored.
                    329:  *     Normally this routine is called with interrupts already 
                    330:  *     disabled, but there is code in place so that it will be more 
                    331:  *     likely to work even if interrupts are turned on.
                    332:  */
                    333: 
                    334: int
                    335: cngetc(void)
                    336: {
                    337:        int ret;
                    338: 
                    339:        ret = (*cgetc)(TRUE);
                    340: 
                    341:        return ret;
                    342: }
                    343: 
                    344: int
                    345: cnmaygetc(void)
                    346: {
                    347:        int ret;
                    348: 
                    349:        ret = (*cgetc)(FALSE);
                    350: 
                    351:        return ret;
                    352: }
                    353: 
                    354: int
                    355: kd_dogetc(
                    356:        int     wait)
                    357: {
                    358:        unsigned char   c;
                    359:        unsigned char   scancode;
                    360:        unsigned int    char_idx;
                    361:        int     up;
                    362: 
                    363:        kdinit();
                    364:        kd_extended = FALSE;
                    365: 
                    366:        for ( ; ; ) {
                    367:                while (!(inb(K_STATUS) & K_OBUF_FUL))
                    368:                        if (!wait)
                    369:                                return (-1);
                    370:                up = FALSE;
                    371:                /*
                    372:                 * We'd come here for mouse events in debugger, if
                    373:                 * the mouse were on.
                    374:                 */
                    375:                if ((inb(K_STATUS) & 0x20) == 0x20) {
                    376:                        printf("M%xP", inb(K_RDWR));
                    377:                        continue;
                    378:                }
                    379:                scancode = inb(K_RDWR);
                    380:                /*
                    381:                 * Handle extend modifier and
                    382:                 * ack/resend, otherwise we may never receive
                    383:                 * a key.
                    384:                 */
                    385:                if (scancode == K_EXTEND) {
                    386:                        kd_extended = TRUE;
                    387:                        continue;
                    388:                } else if (scancode == K_RESEND) {
                    389: /*  printf("kd_getc: resend"); */
                    390:                        kd_resend();
                    391:                        continue;
                    392:                } else if (scancode == K_ACKSC) {
                    393: /*  printf("kd_getc: handle_ack");     */
                    394:                        kd_handle_ack();
                    395:                        continue;
                    396:                }
                    397:                if (scancode & K_UP) {
                    398:                        up = TRUE;
                    399:                        scancode &= ~K_UP;
                    400:                }
                    401:                if (kd_kbd_mouse)
                    402:                        kd_kbd_magic(scancode);
                    403:                if (scancode < NUMKEYS) {
                    404:                        /* Lookup in map, then process. */
                    405:                        char_idx = kdstate2idx(kd_state, kd_extended);
                    406:                        c = key_map[scancode][char_idx];
                    407:                        if (c == K_SCAN) {
                    408:                                c = key_map[scancode][++char_idx];
                    409:                                kd_state = do_modifier(kd_state, c, up);
                    410: #ifdef notdef
                    411:                                cnsetleds(state2leds(kd_state));
                    412: #endif
                    413:                        } else if (!up) {
                    414:                                /* regular key-down */
                    415:                                if (c == K_CR)
                    416:                                        c = K_LF;
                    417: #ifdef notdef
                    418:                                splx(o_pri);
                    419: #endif
                    420:                                return(c & 0177);
                    421:                        }
                    422:                }
                    423:        }
                    424: }
                    425: 
                    426: 
                    427: int            old_kb_mode;
                    428: 
                    429: #if    MACH_KDB
                    430: #define        poll_spl()      db_splhigh()    /* prevent race w/ kdintr() */
                    431: #define poll_splx(s)   db_splx(s)
                    432: #else  /* MACH_KDB */
                    433: #define        poll_spl()      SPLKD()
                    434: #define poll_splx(s)   splx(s)
                    435: #endif /* MACH_KDB */
                    436: 
                    437: 
                    438: void
                    439: cnpollc(
                    440:        int     on)
                    441: {
                    442:        int     old_spl;                /* spl we're called at... */
                    443: 
                    444:        if (cpu_number()) {
                    445:                return;
                    446:        }
                    447:        if (on) {
                    448:                old_spl = poll_spl();
                    449: 
                    450:                old_kb_mode = kb_mode;
                    451:                kb_mode = KB_ASCII;
                    452:                poll_splx(old_spl);
                    453: 
                    454:                kd_pollc++;
                    455:        } else {
                    456:                --kd_pollc;
                    457: 
                    458:                old_spl = poll_spl();
                    459:                kb_mode = old_kb_mode;
                    460:                poll_splx(old_spl);
                    461: 
                    462: 
                    463:        }
                    464: }
                    465: 
                    466: /* 
                    467:  * kd_handle_ack:
                    468:  * 
                    469:  *     For pending commands, complete the command.  For data bytes, 
                    470:  *     drop the ack on the floor.
                    471:  */
                    472: 
                    473: void
                    474: kd_handle_ack(void)
                    475: {
                    476:        switch (kd_ack) {
                    477:        case SET_LEDS:
                    478:                kd_setleds2();
                    479:                kd_ack = DATA_ACK;
                    480:                break;
                    481:        case DATA_ACK:
                    482:                kd_ack = NOT_WAITING;
                    483:                break;
                    484:        case NOT_WAITING:
                    485:                printf("unexpected ACK from keyboard\n");
                    486:                break;
                    487:        default:
                    488:                panic("bogus kd_ack\n");
                    489:                break;
                    490:        }
                    491: }
                    492: 
                    493: /* 
                    494:  * kd_resend:
                    495:  *
                    496:  *     Resend a missed keyboard command or data byte.
                    497:  */
                    498: 
                    499: void
                    500: kd_resend(void)
                    501: {
                    502:        if (kd_ack == NOT_WAITING) 
                    503:                printf("unexpected RESEND from keyboard\n");
                    504:        else
                    505:                kd_senddata(last_sent);
                    506: }
                    507: 
                    508: 
                    509: /*
                    510:  * do_modifier:
                    511:  *
                    512:  *     Change keyboard state according to which modifier key and
                    513:  *     whether it went down or up.
                    514:  *
                    515:  * input:      the current state, the key, and the key's direction.  
                    516:  *             The key can be any key, not just a modifier key.
                    517:  * 
                    518:  * output:     the new state
                    519:  */
                    520: 
                    521: int
                    522: do_modifier(
                    523:        int             state,
                    524:        Scancode        c,
                    525:        int     up)
                    526: {
                    527:        switch (c) {
                    528:        case (K_ALTSC):
                    529:                if (up)
                    530:                        state &= ~KS_ALTED;
                    531:                else
                    532:                        state |= KS_ALTED;
                    533:                kd_extended = FALSE;
                    534:                break;
                    535: #ifndef        ORC
                    536:        case (K_CLCKSC):
                    537: #endif /* ORC */
                    538:        case (K_CTLSC):
                    539:                if (up)
                    540:                        state &= ~KS_CTLED;
                    541:                else
                    542:                        state |= KS_CTLED;
                    543:                kd_extended = FALSE;
                    544:                break;
                    545: #ifdef ORC
                    546:        case (K_CLCKSC):
                    547:                if (!up)
                    548:                        state ^= KS_CLKED;
                    549:                break;
                    550: #endif /* ORC */
                    551:        case (K_NLCKSC):
                    552:                if (!up)
                    553:                        state ^= KS_NLKED;
                    554:                break;
                    555:        case (K_LSHSC):
                    556:        case (K_RSHSC):
                    557:                if (up)
                    558:                        state &= ~KS_SHIFTED;
                    559:                else
                    560:                        state |= KS_SHIFTED;
                    561:                kd_extended = FALSE;
                    562:                break;
                    563:        }
                    564: 
                    565:        return(state);
                    566: }
                    567: 
                    568: 
                    569: /* 
                    570:  * kdcheckmagic:
                    571:  * 
                    572:  *     Check for magic keystrokes for invoking the debugger or 
                    573:  *     rebooting or ...
                    574:  *
                    575:  * input:      an unprocessed scancode
                    576:  * 
                    577:  * output:     TRUE if a magic key combination was recognized and 
                    578:  *             processed.  FALSE otherwise.
                    579:  *
                    580:  * side effects:       
                    581:  *             various actions possible, depending on which keys are
                    582:  *             pressed.  If the debugger is called, steps are taken 
                    583:  *             to ensure that the system doesn't think the magic keys 
                    584:  *             are still held down.
                    585:  */
                    586: 
                    587: int
                    588: kdcheckmagic(
                    589:        Scancode        scancode,
                    590:        int             *regs)
                    591: {
                    592:        static int magic_state = KS_NORMAL; /* like kd_state */
                    593:        int up = FALSE;
                    594:        extern  int     rebootflag;
                    595: 
                    596:        if (scancode == 0x46 && kbdmouseflag)   /* scroll lock */
                    597:        {
                    598:                kd_kbd_mouse = !kd_kbd_mouse;
                    599:                kd_kbd_magic_button = 0;
                    600:                return(TRUE);
                    601:        }
                    602:        if (scancode & K_UP) {
                    603:                up = TRUE;
                    604:                scancode &= ~K_UP;
                    605:        }
                    606:        magic_state = do_modifier(magic_state, scancode, up);
                    607: 
                    608:        if ((magic_state&(KS_CTLED|KS_ALTED)) == (KS_CTLED|KS_ALTED)) {
                    609:                switch (scancode) {
                    610: #if    MACH_KDB
                    611:                case K_dSC:             /*  ctl-alt-d */
                    612:                        if (!kbdkdbflag)
                    613:                                return(FALSE);
                    614: 
                    615:                        kdb_kintr();    /* invoke debugger */
                    616: 
                    617:                        /* Returned from debugger, so reset kbd state. */
                    618:                        (void)SPLKD();
                    619:                        magic_state = KS_NORMAL;
                    620:                        if (kb_mode == KB_ASCII)
                    621:                                kd_state = KS_NORMAL;
                    622:                                /* setting leds kills kbd */
                    623: 
                    624:                        return(TRUE);
                    625:                        break;
                    626: #endif /* MACH_KDB */
                    627:                case K_DELSC:           /* ctl-alt-del */
                    628:                        /* if rebootflag is on, reboot the system */
                    629:                        if (rebootflag)
                    630:                                kdreboot();
                    631:                        break;
                    632:                }
                    633:        }
                    634:        return(FALSE);
                    635: }
                    636: 
                    637: 
                    638: /*
                    639:  * kdstate2idx:
                    640:  *
                    641:  *     Return the value for the 2nd index into key_map that 
                    642:  *     corresponds to the given state.
                    643:  */
                    644: 
                    645: int
                    646: kdstate2idx(
                    647:        int             state,          /* bit vector, not a state index */
                    648:        int     extended)
                    649: {
                    650:        int state_idx = NORM_STATE;
                    651: 
                    652:        if ((!extended) && state != KS_NORMAL) {
                    653:                if ((state&(KS_SHIFTED|KS_ALTED)) == (KS_SHIFTED|KS_ALTED))
                    654:                        state_idx = SHIFT_ALT;
                    655:                else if (state&KS_SHIFTED)
                    656:                        state_idx = SHIFT_STATE;
                    657:                else if (state&KS_ALTED)
                    658:                        state_idx = ALT_STATE;
                    659:                else if (state&KS_CTLED)
                    660:                        state_idx = CTRL_STATE;
                    661:        }
                    662: 
                    663:        return (CHARIDX(state_idx));
                    664: }
                    665: 
                    666: /*
                    667:  * kdinit:
                    668:  *
                    669:  *     This code initializes the structures and sets up the port registers
                    670:  *     for the console driver.  
                    671:  *
                    672:  *     Each bitmap-based graphics card is likely to require a unique
                    673:  *     way to determine the card's presence.  The driver runs through
                    674:  *     each "special" card that it knows about and uses the first one
                    675:  *     that it finds.  If it doesn't find any, it assumes that an
                    676:  *     EGA-like card is installed.
                    677:  *
                    678:  * input       : None. Interrupts are assumed to be disabled
                    679:  * output      : Driver is initialized
                    680:  *
                    681:  */
                    682: 
                    683: void
                    684: kdinit(void)
                    685: {
                    686:        unsigned char   k_comm;         /* keyboard command byte */
                    687:        unsigned char kd_stat;
                    688: 
                    689:        if (kd_initialized)
                    690:                return;
                    691:        kd_initialized = TRUE;
                    692: 
                    693:        /* get rid of any garbage in output buffer */
                    694:        if (inb(K_STATUS) & K_OBUF_FUL)
                    695:                (void)inb(K_RDWR);
                    696: 
                    697:        cnsetleds(kd_state = KS_NORMAL);
                    698: 
                    699:        kd_sendcmd(KC_CMD_READ);        /* ask for the ctlr command byte */
                    700:        k_comm = kd_getdata();
                    701:        k_comm &= ~K_CB_DISBLE;         /* clear keyboard disable bit */
                    702:        k_comm |= K_CB_ENBLIRQ;         /* enable interrupt */
                    703:        kd_sendcmd(KC_CMD_WRITE);       /* write new ctlr command byte */
                    704:        kd_senddata(k_comm);
                    705: 
                    706: /*     set_kd_state(KS_NORMAL);        does only HALF of set-leds sequence -
                    707:                                        leaves kbd dead */
                    708: 
                    709:        /* get rid of any garbage in output buffer */
                    710:        (void)inb(K_RDWR);
                    711: }
                    712: 
                    713: /*
                    714:  * kd_belloff:
                    715:  *
                    716:  *     This routine shuts the bell off, by sending the appropriate code
                    717:  *     to the speaker port.
                    718:  *
                    719:  * input       : None
                    720:  * output      : bell is turned off
                    721:  *
                    722:  */
                    723: 
                    724: void
                    725: kd_belloff(void)
                    726: {
                    727:        unsigned char status;
                    728: 
                    729:        status = (inb(K_PORTB) & ~(K_SPKRDATA | K_ENABLETMR2));
                    730:        outb(K_PORTB, status);
                    731: }
                    732: 
                    733: 
                    734: /*
                    735:  * kd_bellon:
                    736:  *
                    737:  *     This routine turns the bell on.
                    738:  *
                    739:  * input       : None
                    740:  * output      : bell is turned on
                    741:  *
                    742:  */
                    743: 
                    744: void
                    745: kd_bellon(void)
                    746: {
                    747:        unsigned char   status;
                    748: 
                    749:        /* program timer 2 */
                    750:        outb(K_TMRCTL, K_SELTMR2 | K_RDLDTWORD | K_TSQRWAVE | K_TBINARY);
                    751:        outb(K_TMR2, 1500 & 0xff);      /* LSB */
                    752:        outb(K_TMR2, (int)1500 >> 8);   /* MSB */
                    753: 
                    754:        /* start speaker - why must we turn on K_SPKRDATA? */
                    755:        status = (inb(K_PORTB)| K_ENABLETMR2 | K_SPKRDATA);
                    756:        outb(K_PORTB, status);
                    757:        return;
                    758: }
                    759: 
                    760: /*
                    761:  * kd_senddata:
                    762:  *
                    763:  *     This function sends a byte to the keyboard RDWR port, but
                    764:  *     first waits until the input/output data buffer is clear before
                    765:  *     sending the data.  Note that this byte can be either data or a
                    766:  *     keyboard command.
                    767:  *
                    768:  */
                    769: 
                    770: void
                    771: kd_senddata(
                    772:        unsigned char           ch)
                    773: {
                    774:        while (inb(K_STATUS) & K_IBUF_FUL);
                    775:        outb(K_RDWR, ch);
                    776:        last_sent = ch;
                    777: }
                    778: 
                    779: /*
                    780:  * kd_sendcmd:
                    781:  *
                    782:  *     This function sends a command byte to the keyboard command
                    783:  *     port, but first waits until the input/output data buffer is
                    784:  *     clear before sending the data.
                    785:  *
                    786:  */
                    787: 
                    788: void
                    789: kd_sendcmd(
                    790:        unsigned char           ch)
                    791: {
                    792:        while (inb(K_STATUS) & K_IBUF_FUL);
                    793:        outb(K_CMD, ch);
                    794: }
                    795: 
                    796: 
                    797: /* 
                    798:  * kd_getdata:
                    799:  * 
                    800:  *     This function returns a data byte from the keyboard RDWR port, 
                    801:  *     after waiting until the port is flagged as having something to 
                    802:  *     read. 
                    803:  */
                    804: 
                    805: unsigned char
                    806: kd_getdata(void)
                    807: {
                    808:        while ((inb(K_STATUS) & K_OBUF_FUL) == 0);
                    809:        return(inb(K_RDWR));
                    810: }
                    811: 
                    812: unsigned char
                    813: kd_cmdreg_read(void)
                    814: {
                    815:        int ch=KC_CMD_READ;
                    816: 
                    817:        while (inb(K_STATUS) & (K_IBUF_FUL | K_OBUF_FUL));
                    818:        outb(K_CMD, ch);
                    819: 
                    820:        while ((inb(K_STATUS) & K_OBUF_FUL) == 0);
                    821:        return(inb(K_RDWR));
                    822: }
                    823: 
                    824: void
                    825: kd_cmdreg_write(
                    826:        unsigned char           val)
                    827: {
                    828:        int ch=KC_CMD_WRITE;
                    829: 
                    830:        while (inb(K_STATUS) & K_IBUF_FUL);
                    831:        outb(K_CMD, ch);
                    832: 
                    833:        while (inb(K_STATUS) & K_IBUF_FUL);
                    834:        outb(K_RDWR, val);
                    835: }
                    836: 
                    837: int kd_mouse_write_no_ack = 0;
                    838: 
                    839: int
                    840: kd_mouse_write(
                    841:        unsigned char           val)
                    842: {
                    843:        int ch=0xd4;            /* output byte to aux device (i.e. mouse) */
                    844:        int ret = 0;
                    845: 
                    846:        while (inb(K_STATUS) & K_IBUF_FUL);
                    847:        outb(K_CMD, ch);
                    848: 
                    849:        while (inb(K_STATUS) & K_IBUF_FUL);
                    850:        outb(K_RDWR, val);
                    851: 
                    852:        if (kd_mouse_write_no_ack) goto done;
                    853: 
                    854:        while ((inb(K_STATUS) & K_OBUF_FUL) == 0);
                    855:        if ((inb(K_STATUS) & 0x20) == 0x20) {
                    856:                switch (ret = inb(K_RDWR)) {
                    857:                case 0xfa:
                    858:                        break;
                    859:                case 0xfe:
                    860:                case 0xfc:
                    861:                default:
                    862:                        printf("kd_mouse_write: saw %x for %x\n",
                    863:                                ret, val);
                    864:                }
                    865:        } else  {               /* abort */
                    866:                printf("kd_mouse_write: sync error ??? on %x\n", val);
                    867:        }
                    868: 
                    869: done:  
                    870:        return ret;
                    871: }
                    872: 
                    873: void
                    874: kd_mouse_read(
                    875:        int             no,
                    876:        char            *buf)
                    877: {
                    878: 
                    879:        while (no-- > 0) {
                    880:                while ((inb(K_STATUS) & K_OBUF_FUL) == 0);
                    881:                /*
                    882:                 * We may have seen a mouse event.
                    883:                 */
                    884:                if ((inb(K_STATUS) & 0x20) == 0x20) {
                    885:                        *buf++ = (unsigned char)inb(K_RDWR);
                    886:                } else {        /* abort */
                    887:                        int     junk = inb(K_RDWR);
                    888:                        printf("kd_mouse_read: sync error, received: 0x%x\n",
                    889:                                junk);
                    890:                        break;
                    891:                }
                    892:        }
                    893: }
                    894: 
                    895: void
                    896: kd_mouse_drain(void)
                    897: {
                    898:        int i;
                    899:        while(inb(K_STATUS) & K_IBUF_FUL);
                    900:        while((i = inb(K_STATUS)) & K_OBUF_FUL)
                    901:                printf("kbd: S = %x D = %x\n", i, inb(K_RDWR));
                    902: }
                    903: 
                    904: /* 
                    905:  * set_kd_state:
                    906:  * 
                    907:  *     Set kd_state and update the keyboard status LEDs.
                    908:  */
                    909: 
                    910: void
                    911: set_kd_state(
                    912:        int     newstate)
                    913: {
                    914:        kd_state = newstate;
                    915:        kd_setleds1(state2leds(newstate));
                    916: }
                    917: 
                    918: /* 
                    919:  * state2leds:
                    920:  * 
                    921:  *     Return a byte containing LED settings for the keyboard, given 
                    922:  *     a state vector.
                    923:  */
                    924: 
                    925: unsigned char
                    926: state2leds(
                    927:        int     state)
                    928: {
                    929:        unsigned char result = 0;
                    930: 
                    931:        if (state & KS_NLKED)
                    932:                result |= K_LED_NUMLK;
                    933:        if (state & KS_CLKED)
                    934:                result |= K_LED_CAPSLK;
                    935:        return(result);
                    936: }
                    937: 
                    938: /* 
                    939:  * kd_setleds[12]:
                    940:  * 
                    941:  *     Set the keyboard LEDs according to the given byte.  
                    942:  */
                    943: 
                    944: void
                    945: kd_setleds1(
                    946:        unsigned char           val)
                    947: {
                    948:        if (kd_ack != NOT_WAITING) {
                    949:                printf("kd_setleds1: unexpected state (%d)\n", kd_ack);
                    950:                return;
                    951:        }
                    952: 
                    953:        kd_ack = SET_LEDS;
                    954:        kd_nextled = val;
                    955:        kd_senddata(K_CMD_LEDS);
                    956: }
                    957: 
                    958: void
                    959: kd_setleds2(void)
                    960: {
                    961:        kd_senddata(kd_nextled);
                    962: }
                    963: 
                    964: 
                    965: /* 
                    966:  * cnsetleds:
                    967:  * 
                    968:  *     like kd_setleds[12], but not interrupt-based.
                    969:  *     Currently disabled because cngetc ignores caps lock and num 
                    970:  *     lock anyway.
                    971:  */
                    972: 
                    973: void
                    974: cnsetleds(
                    975:        unsigned char           val)
                    976: {
                    977:        kd_senddata(K_CMD_LEDS);
                    978:        (void)kd_getdata();             /* XXX - assume is ACK */
                    979:        kd_senddata(val);
                    980:        (void)kd_getdata();             /* XXX - assume is ACK */
                    981: }
                    982: 
                    983: void
                    984: kdreboot(void)
                    985: {
                    986:        kd_sendcmd(0xFE);               /* XXX - magic # */
                    987:         /*
                    988:          * DRAT.  We're still here.  Let's try a "CPU shutdown", which consists
                    989:          * of clearing the IDTR and causing an exception.  It's in locore.s
                    990:          */
                    991:         cpu_shutdown();
                    992:         /*NOTREACHED*/
                    993: }
                    994: 
                    995: int
                    996: kd_kbd_magic(
                    997:        int     scancode)
                    998: {
                    999: int new_button = 0;
                   1000: 
                   1001:        if (kd_kbd_mouse == 2)
                   1002:                printf("sc = %x\n", scancode);
                   1003: 
                   1004:        switch (scancode) {
                   1005: /* f1 f2 f3 */
                   1006:        case 0x3d:
                   1007:                new_button++;
                   1008:        case 0x3c:
                   1009:                new_button++;
                   1010:        case 0x3b:
                   1011:                new_button++;
                   1012:                if (kd_kbd_magic_button && (new_button != kd_kbd_magic_button)) {
                   1013:                                /* down w/o up */
                   1014:                }
                   1015:                                /* normal */
                   1016:                if (kd_kbd_magic_button == new_button) {
                   1017:                        kd_kbd_magic_button = 0;
                   1018:                } else {
                   1019:                        kd_kbd_magic_button = new_button;
                   1020:                }
                   1021:                break;
                   1022:        default:
                   1023:                return 0;
                   1024:        }
                   1025:        return 1;
                   1026: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.