kbd.c revision d8c870b0de5416eb7c3dbc6e97d93a26a5a5f299
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2007 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
/*
* Usage: kbd [-r] [-t] [-l] [-i] [-c on|off] [-a enable|disable|alternate]
* [-d keyboard device]
* -r reset the keyboard as if power-up
* -t return the type of the keyboard being used
* -l return the layout of the keyboard being used,
* and the Autorepeat settings
* -i read in the default configuration file
* -c on|off turn on|off clicking
* -a enable|disable|alternate sets abort sequence
* -D autorepeat delay sets autorepeat dealy, unit in ms
* -R autorepeat rate sets autorepeat rate, unit in ms
* -d keyboard device chooses the kbd device, default /dev/kbd.
* -s keyboard layout sets keyboard layout
*/
#include <sys/types.h>
#include <sys/ioctl.h>
#include <sys/kbio.h>
#include <sys/kbd.h>
#include <stdio.h>
#include <fcntl.h>
#include <deflt.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <stropts.h>
#include <libintl.h>
#include <locale.h>
#define KBD_DEVICE "/dev/kbd" /* default keyboard device */
#define DEF_FILE "/etc/default/kbd" /* kbd defaults file */
#define DEF_ABORT "KEYBOARD_ABORT="
#define DEF_CLICK "KEYCLICK="
#define DEF_RPTDELAY "REPEAT_DELAY="
#define DEF_RPTRATE "REPEAT_RATE="
#define DEF_LAYOUT "LAYOUT="
#define KBD_LAYOUT_FILE "/usr/share/lib/keytables/type_6/kbd_layouts"
#define MAX_LAYOUT_NUM 128
#define MAX_LINE_SIZE 256
#define DEFAULT_KBD_LAYOUT 33
char *layout_names[MAX_LAYOUT_NUM];
int layout_numbers[MAX_LAYOUT_NUM];
static int layout_count;
static int default_layout_number = 0;
static void reset(int);
static void get_type(int);
static void get_layout(int);
static void kbd_defaults(int);
static void usage(void);
static int click(char *, int);
static int abort_enable(char *, int);
static int set_repeat_delay(char *, int);
static int set_repeat_rate(char *, int);
static int get_layout_number(char *);
static int set_layout(int, int);
static int get_layouts(void);
static int set_kbd_layout(int, char *);
int
main(int argc, char **argv)
{
int c, error;
int rflag, tflag, lflag, cflag, dflag, aflag, iflag, errflag,
Dflag, Rflag, rtlacDRflag, sflag;
char *copt, *aopt, *delay, *rate, *layout_name;
char *kbdname = KBD_DEVICE;
int kbd;
extern char *optarg;
extern int optind;
rflag = tflag = cflag = dflag = aflag = iflag = errflag = lflag =
Dflag = Rflag = sflag = 0;
copt = aopt = (char *)0;
(void) setlocale(LC_ALL, "");
#if !defined(TEXT_DOMAIN)
#define TEXT_DOMAIN "SYS_TEST"
#endif
(void) textdomain(TEXT_DOMAIN);
while ((c = getopt(argc, argv, "rtlisc:a:d:D:R:")) != EOF) {
switch (c) {
case 'r':
rflag++;
break;
case 't':
tflag++;
break;
case 'l':
lflag++;
break;
case 'i':
iflag++;
break;
case 's':
sflag++;
break;
case 'c':
copt = optarg;
cflag++;
break;
case 'a':
aopt = optarg;
aflag++;
break;
case 'd':
kbdname = optarg;
dflag++;
break;
case 'D':
delay = optarg;
Dflag++;
break;
case 'R':
rate = optarg;
Rflag++;
break;
case '?':
errflag++;
break;
}
}
/*
* Check for valid arguments:
*
* If argument parsing failed or if there are left-over
* command line arguments(except -s option), then we're done now.
*/
if (errflag != 0 || ((sflag == 0) && (argc != optind))) {
usage();
exit(1);
}
/*
* kbd requires that the user specify either "-i" or "-s" or at
* least one of -[rtlacDR]. The "-d" option is, well, optional.
* We don't care if it's there or not.
*/
rtlacDRflag = rflag + tflag + lflag + aflag + cflag + Dflag + Rflag;
if (!((iflag != 0 && sflag == 0 && rtlacDRflag == 0) ||
(iflag == 0 && sflag != 0 && dflag == 0 && rtlacDRflag == 0) ||
(iflag == 0 && sflag == 0 && rtlacDRflag != 0))) {
usage();
exit(1);
}
if (Dflag && atoi(delay) <= 0) {
(void) fprintf(stderr, "Invalid arguments: -D %s\n", delay);
usage();
exit(1);
}
if (Rflag && atoi(rate) <= 0) {
(void) fprintf(stderr, "Invalid arguments: -R %s\n", rate);
usage();
exit(1);
}
/*
* Open the keyboard device
*/
if ((kbd = open(kbdname, O_RDWR)) < 0) {
perror("opening the keyboard");
(void) fprintf(stderr, "kbd: Cannot open %s\n", kbdname);
exit(1);
}
if (iflag) {
kbd_defaults(kbd);
exit(0); /* A mutually exclusive option */
/*NOTREACHED*/
}
if (tflag)
get_type(kbd);
if (lflag)
get_layout(kbd);
if (cflag && (error = click(copt, kbd)) != 0)
exit(error);
if (rflag)
reset(kbd);
if (aflag && (error = abort_enable(aopt, kbd)) != 0)
exit(error);
if (Dflag && (error = set_repeat_delay(delay, kbd)) != 0)
exit(error);
if (Rflag && (error = set_repeat_rate(rate, kbd)) != 0)
exit(error);
if (sflag) {
if (argc == optind) {
layout_name = NULL;
} else if (argc == (optind + 1)) {
layout_name = argv[optind];
} else {
usage();
exit(1);
}
if ((error = set_kbd_layout(kbd, layout_name)) != 0)
exit(error);
}
return (0);
}
/*
* this routine gets the type of the keyboard being used
*/
static int
set_kbd_layout(int kbd, char *layout_name)
{
int layout_num;
int error = 1;
/* get the language info from the layouts file */
if (get_layouts() != 0)
return (error);
if (layout_name != NULL) {
if ((layout_num = get_layout_number(layout_name)) == -1) {
(void) fprintf(stderr, "%s: unknown layout name\n"
"Please refer to 'kbd -s' to get the "
"supported layouts.\n", layout_name);
return (error);
}
} else {
int i, j, print_cnt, input_num;
boolean_t input_right = B_TRUE;
boolean_t default_input = B_FALSE;
char input[8]; /* 8 chars is enough for numbers */
print_cnt = (layout_count % 2) ?
layout_count/2+1 : layout_count/2;
for (i = 1; i <= print_cnt; i++) {
(void) printf("%2d. %-30s", i,
layout_names[i-1]);
j = i + print_cnt;
if (j <= layout_count) {
(void) printf("%-2d. %-30s\n", j,
layout_names[j-1]);
}
}
(void) printf(gettext("\nTo select the keyboard layout,"
" enter a number [default %d]:"),
default_layout_number+1);
for (;;) {
if (input_right == B_FALSE)
(void) printf(gettext("Invalid input. "
"Please input a number "
"(1,2,...):"));
(void) memset(input, 0, 8);
(void) fflush(stdin);
(void) fgets(input, 8, stdin);
if (strlen(input) > 4) {
input_right = B_FALSE;
continue;
}
if (input[0] == '\n') {
default_input = B_TRUE;
break;
}
input_right = B_TRUE;
/* check if the inputs are numbers 0~9 */
for (i = 0; i < (strlen(input) - 1); i++) {
if ((input[i] < '0') ||
(input[i] > '9')) {
input_right = B_FALSE;
break;
}
}
if (input_right == B_FALSE)
continue;
input_num = atoi(input);
if ((input_num > 0) &&
(input_num <= layout_count))
break;
else
input_right = B_FALSE;
}
if (default_input == B_TRUE)
layout_num = DEFAULT_KBD_LAYOUT;
else
layout_num = layout_numbers[--input_num];
}
if ((error = set_layout(kbd, layout_num)) != 0)
return (error);
return (0);
}
/*
* this routine resets the state of the keyboard as if power-up
*/
static void
reset(int kbd)
{
int cmd;
cmd = KBD_CMD_RESET;
if (ioctl(kbd, KIOCCMD, &cmd)) {
perror("kbd: ioctl error");
exit(1);
}
}
/*
* this routine gets the type of the keyboard being used
*/
static void
get_type(int kbd)
{
int kbd_type;
if (ioctl(kbd, KIOCTYPE, &kbd_type)) {
perror("ioctl (kbd type)");
exit(1);
}
switch (kbd_type) {
case KB_SUN3:
(void) printf("Type 3 Sun keyboard\n");
break;
case KB_SUN4:
(void) printf("Type 4 Sun keyboard\n");
break;
case KB_ASCII:
(void) printf("ASCII\n");
break;
case KB_PC:
(void) printf("PC\n");
break;
case KB_USB:
(void) printf("USB keyboard\n");
break;
default:
(void) printf("Unknown keyboard type\n");
break;
}
}
/*
* this routine gets the layout of the keyboard being used
* also, included the autorepeat delay and rate being used
*/
static void
get_layout(int kbd)
{
int kbd_type;
int kbd_layout;
/* these two variables are used for getting delay&rate */
int delay, rate;
delay = rate = 0;
if (ioctl(kbd, KIOCTYPE, &kbd_type)) {
perror("ioctl (kbd type)");
exit(1);
}
if (ioctl(kbd, KIOCLAYOUT, &kbd_layout)) {
perror("ioctl (kbd layout)");
exit(1);
}
(void) printf("type=%d\nlayout=%d (0x%.2x)\n",
kbd_type, kbd_layout, kbd_layout);
/* below code is used to get the autorepeat delay and rate */
if (ioctl(kbd, KIOCGRPTDELAY, &delay)) {
perror("ioctl (kbd get repeat delay)");
exit(1);
}
if (ioctl(kbd, KIOCGRPTRATE, &rate)) {
perror("ioctl (kbd get repeat rate)");
exit(1);
}
(void) printf("delay(ms)=%d\n", delay);
(void) printf("rate(ms)=%d\n", rate);
}
/*
* this routine enables or disables clicking of the keyboard
*/
static int
click(char *copt, int kbd)
{
int cmd;
if (strcmp(copt, "on") == 0)
cmd = KBD_CMD_CLICK;
else if (strcmp(copt, "off") == 0)
cmd = KBD_CMD_NOCLICK;
else {
(void) fprintf(stderr, "wrong option -- %s\n", copt);
usage();
return (1);
}
if (ioctl(kbd, KIOCCMD, &cmd)) {
perror("kbd ioctl (keyclick)");
return (1);
}
return (0);
}
/*
* this routine enables/disables/sets BRK or abort sequence feature
*/
static int
abort_enable(char *aopt, int kbd)
{
int enable;
if (strcmp(aopt, "alternate") == 0)
enable = KIOCABORTALTERNATE;
else if (strcmp(aopt, "enable") == 0)
enable = KIOCABORTENABLE;
else if (strcmp(aopt, "disable") == 0)
enable = KIOCABORTDISABLE;
else {
(void) fprintf(stderr, "wrong option -- %s\n", aopt);
usage();
return (1);
}
if (ioctl(kbd, KIOCSKABORTEN, &enable)) {
perror("kbd ioctl (abort enable)");
return (1);
}
return (0);
}
/*
* this routine set autorepeat delay
*/
static int
set_repeat_delay(char *delay_str, int kbd)
{
int delay = atoi(delay_str);
/*
* The error message depends on the different inputs.
* a. the input is a invalid integer(unit in ms)
* b. the input is a integer less than the minimal delay setting.
* The condition (a) has been covered by main function and set_default
* function.
*/
if (ioctl(kbd, KIOCSRPTDELAY, &delay) == -1) {
if (delay < KIOCRPTDELAY_MIN)
(void) fprintf(stderr, "kbd: specified delay %d is "
"less than minimum %d\n", delay, KIOCRPTDELAY_MIN);
else
perror("kbd: set repeat delay");
return (1);
}
return (0);
}
/*
* this routine set autorepeat rate
*/
static int
set_repeat_rate(char *rate_str, int kbd)
{
int rate = atoi(rate_str);
/*
* The error message depends on the different inputs.
* a. the input is a invalid integer(unit in ms)
* b. the input is a integer less than the minimal rate setting.
* The condition (a) has been covered by main function and set_default
* function.
*/
if (ioctl(kbd, KIOCSRPTRATE, &rate) == -1) {
if (rate < KIOCRPTRATE_MIN)
(void) fprintf(stderr, "kbd: specified rate %d is "
"less than minimum %d\n", rate, KIOCRPTRATE_MIN);
else
perror("kbd: set repeat rate");
return (1);
}
return (0);
}
#define BAD_DEFAULT "kbd: bad default value for %s: %s\n"
static void
kbd_defaults(int kbd)
{
char *p;
int layout_num;
if (defopen(DEF_FILE) != 0) {
(void) fprintf(stderr, "Can't open default file: %s\n",
DEF_FILE);
exit(1);
}
p = defread(DEF_CLICK);
if (p != NULL) {
/*
* KEYCLICK must equal "on" or "off"
*/
if ((strcmp(p, "on") == 0) || (strcmp(p, "off") == 0))
(void) click(p, kbd);
else
(void) fprintf(stderr, BAD_DEFAULT, DEF_CLICK, p);
}
p = defread(DEF_ABORT);
if (p != NULL) {
/*
* ABORT must equal "enable", "disable" or "alternate"
*/
if ((strcmp(p, "enable") == 0) ||
(strcmp(p, "alternate") == 0) ||
(strcmp(p, "disable") == 0))
(void) abort_enable(p, kbd);
else
(void) fprintf(stderr, BAD_DEFAULT, DEF_ABORT, p);
}
p = defread(DEF_RPTDELAY);
if (p != NULL) {
/*
* REPEAT_DELAY unit in ms
*/
if (atoi(p) > 0)
(void) set_repeat_delay(p, kbd);
else
(void) fprintf(stderr, BAD_DEFAULT, DEF_RPTDELAY, p);
}
p = defread(DEF_RPTRATE);
if (p != NULL) {
/*
* REPEAT_RATE unit in ms
*/
if (atoi(p) > 0)
(void) set_repeat_rate(p, kbd);
else
(void) fprintf(stderr, BAD_DEFAULT, DEF_RPTRATE, p);
}
p = defread(DEF_LAYOUT);
if (p != NULL) {
/*
* LAYOUT must be one of the layouts supported in kbd_layouts
*/
if (get_layouts() != 0)
return;
if ((layout_num = get_layout_number(p)) == -1) {
(void) fprintf(stderr, BAD_DEFAULT, DEF_LAYOUT, p);
return;
}
(void) set_layout(kbd, layout_num);
}
}
static int
get_layout_number(char *layout)
{
int i;
int layout_number = -1;
for (i = 0; i < layout_count; i ++) {
if (strcmp(layout, layout_names[i]) == 0) {
layout_number = layout_numbers[i];
break;
}
}
return (layout_number);
}
static int
get_layouts()
{
FILE *stream;
char buffer[MAX_LINE_SIZE];
char *result = NULL;
int i = 0;
char *tmpbuf;
if ((stream = fopen(KBD_LAYOUT_FILE, "r")) == 0) {
perror(KBD_LAYOUT_FILE);
return (1);
}
while ((fgets(buffer, MAX_LINE_SIZE, stream) != NULL) &&
(i < MAX_LAYOUT_NUM)) {
if (buffer[0] == '#')
continue;
if ((result = strtok(buffer, "=")) == NULL)
continue;
if ((tmpbuf = strdup(result)) != NULL) {
layout_names[i] = tmpbuf;
} else {
perror("out of memory getting layout names");
return (1);
}
if ((result = strtok(NULL, "\n")) == NULL)
continue;
layout_numbers[i] = atoi(result);
if (strcmp(tmpbuf, "US-English") == 0)
default_layout_number = i;
i++;
}
layout_count = i;
return (0);
}
/*
* this routine sets the layout of the keyboard being used
*/
static int
set_layout(int kbd, int layout_num)
{
if (ioctl(kbd, KIOCSLAYOUT, layout_num)) {
perror("ioctl (set kbd layout)");
return (1);
}
return (0);
}
static char *usage1 = "kbd [-r] [-t] [-l] [-a enable|disable|alternate]";
static char *usage2 = " [-c on|off][-D delay][-R rate][-d keyboard device]";
static char *usage3 = "kbd -i [-d keyboard device]";
static char *usage4 = "kbd -s [language]";
static void
usage(void)
{
(void) fprintf(stderr, "Usage:\t%s\n\t%s\n\t%s\n\t%s\n", usage1, usage2,
usage3, usage4);
}