diff options
Diffstat (limited to 'sys')
-rw-r--r-- | sys/conf/files.alpha | 1 | ||||
-rw-r--r-- | sys/conf/files.i386 | 1 | ||||
-rw-r--r-- | sys/conf/files.pc98 | 1 | ||||
-rw-r--r-- | sys/conf/options.i386 | 5 | ||||
-rw-r--r-- | sys/conf/options.pc98 | 5 | ||||
-rw-r--r-- | sys/i386/i386/userconfig.c | 3148 | ||||
-rw-r--r-- | sys/pc98/i386/userconfig.c | 3178 |
7 files changed, 0 insertions, 6339 deletions
diff --git a/sys/conf/files.alpha b/sys/conf/files.alpha index ca80a9e..5f12a5e 100644 --- a/sys/conf/files.alpha +++ b/sys/conf/files.alpha @@ -81,7 +81,6 @@ alpha/alpha/support.s standard alpha/alpha/swtch.s standard alpha/alpha/sys_machdep.c standard alpha/alpha/trap.c standard -#alpha/alpha/userconfig.c optional userconfig alpha/alpha/vm_machdep.c standard alpha/isa/isa.c optional isa alpha/isa/isa_dma.c optional isa diff --git a/sys/conf/files.i386 b/sys/conf/files.i386 index 819a881..5b8796b 100644 --- a/sys/conf/files.i386 +++ b/sys/conf/files.i386 @@ -212,7 +212,6 @@ i386/i386/support.s standard i386/i386/swtch.s standard i386/i386/sys_machdep.c standard i386/i386/trap.c standard -#i386/i386/userconfig.c optional userconfig i386/i386/vm86.c standard i386/i386/vm_machdep.c standard i386/ibcs2/ibcs2_errno.c optional ibcs2 diff --git a/sys/conf/files.pc98 b/sys/conf/files.pc98 index ed4805a..f063998 100644 --- a/sys/conf/files.pc98 +++ b/sys/conf/files.pc98 @@ -382,7 +382,6 @@ pc98/apm/apm_bioscall.s optional apm pc98/i386/busio.s standard pc98/i386/busiosubr.c standard pc98/i386/machdep.c standard -#pc98/i386/userconfig.c optional userconfig pc98/pc98/atapi.c optional wdc pc98/pc98/clock.c standard pc98/pc98/diskslice_machdep.c standard diff --git a/sys/conf/options.i386 b/sys/conf/options.i386 index 9513141..588171a 100644 --- a/sys/conf/options.i386 +++ b/sys/conf/options.i386 @@ -120,11 +120,6 @@ KBD_MAXWAIT opt_kbd.h KBD_RESETDELAY opt_kbd.h KBDIO_DEBUG opt_kbd.h -#USERCONFIG opt_userconfig.h -#VISUAL_USERCONFIG opt_userconfig.h -#INTRO_USERCONFIG opt_userconfig.h -#DEV_EISA opt_userconfig.h - EISA_SLOTS opt_eisa.h # pcvt(4) has a bunch of options diff --git a/sys/conf/options.pc98 b/sys/conf/options.pc98 index 7d9df0f..f2c4093 100644 --- a/sys/conf/options.pc98 +++ b/sys/conf/options.pc98 @@ -108,11 +108,6 @@ KBD_MAXWAIT opt_kbd.h KBD_RESETDELAY opt_kbd.h KBDIO_DEBUG opt_kbd.h -#USERCONFIG opt_userconfig.h -#VISUAL_USERCONFIG opt_userconfig.h -#INTRO_USERCONFIG opt_userconfig.h -#DEV_EISA opt_userconfig.h - EISA_SLOTS opt_eisa.h # pcvt(4) has a bunch of options diff --git a/sys/i386/i386/userconfig.c b/sys/i386/i386/userconfig.c deleted file mode 100644 index 034334e..0000000 --- a/sys/i386/i386/userconfig.c +++ /dev/null @@ -1,3148 +0,0 @@ -/** - ** Copyright (c) 1995 - ** Michael Smith, msmith@freebsd.org. All rights reserved. - ** - ** This code contains a module marked : - - * Copyright (c) 1991 Regents of the University of California. - * All rights reserved. - * Copyright (c) 1994 Jordan K. Hubbard - * All rights reserved. - * Copyright (c) 1994 David Greenman - * All rights reserved. - * - * Many additional changes by Bruce Evans - * - * This code is derived from software contributed by the - * University of California Berkeley, Jordan K. Hubbard, - * David Greenman and Bruce Evans. - - ** As such, it contains code subject to the above copyrights. - ** The module and its copyright can be found below. - ** - ** Redistribution and use in source and binary forms, with or without - ** modification, are permitted provided that the following conditions - ** are met: - ** 1. Redistributions of source code must retain the above copyright - ** notice, this list of conditions and the following disclaimer as - ** the first lines of this file unmodified. - ** 2. Redistributions in binary form must reproduce the above copyright - ** notice, this list of conditions and the following disclaimer in the - ** documentation and/or other materials provided with the distribution. - ** 3. All advertising materials mentioning features or use of this software - ** must display the following acknowledgment: - ** This product includes software developed by Michael Smith. - ** 4. The name of the author may not be used to endorse or promote products - ** derived from this software without specific prior written permission. - ** - ** THIS SOFTWARE IS PROVIDED BY MICHAEL SMITH ``AS IS'' AND ANY EXPRESS OR - ** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES - ** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. - ** IN NO EVENT SHALL MICHAEL SMITH BE LIABLE FOR ANY DIRECT, INDIRECT, - ** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT - ** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, - ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY - ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF - ** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - ** - ** $FreeBSD$ - **/ - -/** - ** USERCONFIG - ** - ** Kernel boot-time configuration manipulation tool for FreeBSD. - ** - ** Two modes of operation are supported : the default is the line-editor mode, - ** the command "visual" invokes the fullscreen mode. - ** - ** The line-editor mode is the old favorite from FreeBSD 2.0/20.05 &c., the - ** fullscreen mode requires syscons or a minimal-ansi serial console. - **/ - -/** - ** USERCONFIG, visual mode. - ** - ** msmith@freebsd.org - ** - ** Look for "EDIT THIS LIST" to add to the list of known devices - ** - ** - ** There are a number of assumptions made in this code. - ** - ** - That the console supports a minimal set of ANSI escape sequences - ** (See the screen manipulation section for a summary) - ** and has at least 24 rows. - ** - That values less than or equal to zero for any of the device - ** parameters indicate that the driver does not use the parameter. - ** - That flags are _always_ editable. - ** - ** Devices marked as disabled are imported as such. - ** - ** For this tool to be useful, the list of devices below _MUST_ be updated - ** when a new driver is brought into the kernel. It is not possible to - ** extract this information from the drivers in the kernel. - ** - ** XXX - TODO: - ** - ** - Display _what_ a device conflicts with. - ** - Implement page up/down (as what?) - ** - Wizard mode (no restrictions) - ** - Find out how to put syscons back into low-intensity mode so that the - ** !b escape is useful on the console. (It seems to be that it actually - ** gets low/high intensity backwards. That looks OK.) - ** - ** - Only display headings with devices under them. (difficult) - **/ - -#include "opt_userconfig.h" -#define COMPAT_OLDISA /* get the definitions */ - -#include <sys/param.h> -#include <sys/systm.h> -#include <sys/kernel.h> -#include <sys/malloc.h> -#include <sys/reboot.h> -#include <sys/linker.h> -#include <sys/sysctl.h> -#include <sys/bus.h> -#include <sys/cons.h> - -#include <machine/md_var.h> -#include <machine/limits.h> - -#define _I386_ISA_ISA_DEVICE_H_ - -static MALLOC_DEFINE(M_DEVL, "uc_devlist", "uc_device lists in userconfig()"); - -#include <machine/uc_device.h> -static struct uc_device *uc_devlist; /* list read by kget to extract changes */ -static struct uc_device *uc_devtab; /* fake uc_device table */ - -static int userconfig_boot_parsing; /* set if we are reading from the boot instructions */ - -#define putchar(x) cnputc(x) - -static void load_devtab(void); -static void free_devtab(void); -static void save_resource(struct uc_device *); - -static int -sysctl_machdep_uc_devlist(SYSCTL_HANDLER_ARGS) -{ - struct uc_device *id; - int error=0; - char name[8]; - - if(!req->oldptr) { - /* Only sizing */ - id=uc_devlist; - while(id) { - error+=sizeof(struct uc_device)+8; - id=id->id_next; - } - return(SYSCTL_OUT(req,0,error)); - } else { - /* Output the data. The buffer is filled with consecutive - * struct uc_device and char buf[8], containing the name - * (not guaranteed to end with '\0'). - */ - id=uc_devlist; - while(id) { - error=sysctl_handle_opaque(oidp,id, - sizeof(struct uc_device),req); - if(error) return(error); - strncpy(name,id->id_name,8); - error=sysctl_handle_opaque(oidp,name, - 8,req); - if(error) return(error); - id=id->id_next; - } - return(0); - } -} - -SYSCTL_PROC( _machdep, OID_AUTO, uc_devlist, CTLFLAG_RD, - 0, 0, sysctl_machdep_uc_devlist, "A", - "List of ISA devices changed in UserConfig"); - -/* -** Obtain command input. -** -** Initially, input is read from a possibly-loaded script. -** At the end of the script, or if no script is supplied, -** behaviour is determined by the RB_CONFIG (-c) flag. If -** the flag is set, user input is read from the console; if -** unset, the 'quit' command is invoked and userconfig -** will exit. -** -** Note that quit commands encountered in the script will be -** ignored if the RB_CONFIG flag is supplied. -*/ -static const char *config_script; -static int config_script_size; /* use of int for -ve magic value */ - -#define has_config_script() (config_script_size > 0) - -static int -init_config_script(void) -{ - caddr_t autoentry, autoattr; - - /* Look for loaded userconfig script */ - autoentry = preload_search_by_type("userconfig_script"); - if (autoentry != NULL) { - /* We have one, get size and data */ - config_script_size = 0; - if ((autoattr = preload_search_info(autoentry, MODINFO_SIZE)) != NULL) - config_script_size = (size_t)*(u_int32_t *)autoattr; - config_script = NULL; - if ((autoattr = preload_search_info(autoentry, MODINFO_ADDR)) != NULL) - config_script = *(const char **)autoattr; - /* sanity check */ - if ((config_script_size == 0) || (config_script == NULL)) { - config_script_size = 0; - config_script = NULL; - } - } - return has_config_script(); -} - -static int -getchar(void) -{ - int c = -1; -#ifdef INTRO_USERCONFIG - static int intro = 0; -#endif - - if (has_config_script()) - { - /* Consume character from loaded userconfig script, display */ - userconfig_boot_parsing = 1; - c = *config_script; - config_script++; - config_script_size--; - - } else { - -#ifdef INTRO_USERCONFIG - if (userconfig_boot_parsing) { - if (!(boothowto & RB_CONFIG)) { - /* userconfig_script, !RB_CONFIG -> quit */ - if (intro == 0) { - c = 'q'; - config_script = "uit\n"; - config_script_size = strlen(config_script); - /* userconfig_script will be 1 on the next pass */ - } - } else { - /* userconfig_script, RB_CONFIG -> cngetc() */ - } - } else { - if (!(boothowto & RB_CONFIG)) { - /* no userconfig_script, !RB_CONFIG -> show intro */ - if (intro == 0) { - intro = 1; - c = 'i'; - config_script = "ntro\n"; - config_script_size = strlen(config_script); - /* userconfig_script will be 1 on the next pass */ - } - } else { - /* no userconfig_script, RB_CONFIG -> cngetc() */ - } - } -#else /* !INTRO_USERCONFIG */ - /* assert(boothowto & RB_CONFIG) */ -#endif /* INTRO_USERCONFIG */ - userconfig_boot_parsing = 0; - if (c <= 0) - c = cngetc(); - } - return(c); -} - -#ifndef FALSE -#define FALSE (0) -#define TRUE (!FALSE) -#endif - -#ifdef VISUAL_USERCONFIG - -typedef struct -{ - char dev[16]; /* device basename */ - char name[60]; /* long name */ - int attrib; /* things to do with the device */ - int class; /* device classification */ -} DEV_INFO; - -#define FLG_INVISIBLE (1<<0) /* device should not be shown */ -#define FLG_MANDATORY (1<<1) /* device can be edited but not disabled */ -#define FLG_FIXIRQ (1<<2) /* device IRQ cannot be changed */ -#define FLG_FIXIOBASE (1<<3) /* device iobase cannot be changed */ -#define FLG_FIXMADDR (1<<4) /* device maddr cannot be changed */ -#define FLG_FIXMSIZE (1<<5) /* device msize cannot be changed */ -#define FLG_FIXDRQ (1<<6) /* device DRQ cannot be changed */ -#define FLG_FIXED (FLG_FIXIRQ|FLG_FIXIOBASE|FLG_FIXMADDR|FLG_FIXMSIZE|FLG_FIXDRQ) -#define FLG_IMMUTABLE (FLG_FIXED|FLG_MANDATORY) - -#define CLS_STORAGE 1 /* storage devices */ -#define CLS_NETWORK 2 /* network interfaces */ -#define CLS_COMMS 3 /* serial, parallel ports */ -#define CLS_INPUT 4 /* user input : mice, keyboards, joysticks etc */ -#define CLS_MMEDIA 5 /* "multimedia" devices (sound, video, etc) */ -#define CLS_MISC 255 /* none of the above */ - - -typedef struct -{ - char name[60]; - int number; -} DEVCLASS_INFO; - -static DEVCLASS_INFO devclass_names[] = { -{ "Storage : ", CLS_STORAGE}, -{ "Network : ", CLS_NETWORK}, -{ "Communications : ", CLS_COMMS}, -{ "Input : ", CLS_INPUT}, -{ "Multimedia : ", CLS_MMEDIA}, -{ "Miscellaneous : ", CLS_MISC}, -{ "",0}}; - - -/********************* EDIT THIS LIST **********************/ - -/** Notes : - ** - ** - Devices that shouldn't be seen or removed should be marked FLG_INVISIBLE. - ** - XXX The list below should be reviewed by the driver authors to verify - ** that the correct flags have been set for each driver, and that the - ** descriptions are accurate. - **/ - -static DEV_INFO device_info[] = { -/*---Name----- ---Description---------------------------------------------- */ -{"adv", "AdvanSys SCSI narrow controller", 0, CLS_STORAGE}, -{"bt", "Buslogic SCSI controller", 0, CLS_STORAGE}, -{"aha", "Adaptec 154x SCSI controller", 0, CLS_STORAGE}, -{"aic", "Adaptec 152x SCSI and compatible SCSI cards", 0, CLS_STORAGE}, -{"nca", "ProAudio Spectrum SCSI and compatibles", 0, CLS_STORAGE}, -{"sea", "Seagate ST01/ST02 SCSI and compatibles", 0, CLS_STORAGE}, -{"stg", "TMC 18C30/18C50 based SCSI cards", 0, CLS_STORAGE}, -{"ata", "ATA/ATAPI compatible disk controller", 0, CLS_STORAGE}, -{"fdc", "Floppy disk controller", FLG_FIXED, CLS_STORAGE}, -{"mcd", "Mitsumi CD-ROM", 0, CLS_STORAGE}, -{"scd", "Sony CD-ROM", 0, CLS_STORAGE}, -{"matcd", "Matsushita/Panasonic/Creative CDROM", 0, CLS_STORAGE}, -{"wt", "Wangtek/Archive QIC-02 Tape drive", 0, CLS_STORAGE}, -{"ad", "ATA/ATAPI compatible storage device", FLG_INVISIBLE, CLS_STORAGE}, -{"fd", "Floppy disk device", FLG_INVISIBLE, CLS_STORAGE}, - -{"cs", "IBM EtherJet, CS89x0-based Ethernet adapters",0, CLS_NETWORK}, -{"ed", "NE1000,NE2000,3C503,WD/SMC80xx Ethernet adapters",0, CLS_NETWORK}, -{"el", "3C501 Ethernet adapter", 0, CLS_NETWORK}, -{"ep", "3C509 Ethernet adapter", 0, CLS_NETWORK}, -{"ex", "Intel EtherExpress Pro/10 Ethernet adapter", 0, CLS_NETWORK}, -{"fe", "Fujitsu MB86960A/MB86965A Ethernet adapters", 0, CLS_NETWORK}, -{"ie", "AT&T Starlan 10 and EN100, 3C507, NI5210 Ethernet adapters",0,CLS_NETWORK}, -{"le", "DEC Etherworks 2 and 3 Ethernet adapters", 0, CLS_NETWORK}, -{"lnc", "Isolan, Novell NE2100/NE32-VL Ethernet adapters", 0,CLS_NETWORK}, -{"sn", "SMC/Megahertz Ethernet adapters", 0,CLS_NETWORK}, -{"xe", "Xircom PC Card Ethernet adapter", 0, CLS_NETWORK}, -{"rdp", "RealTek RTL8002 Pocket Ethernet", 0, CLS_NETWORK}, - -{"sio", "8250/16450/16550 Serial port", 0, CLS_COMMS}, -{"cx", "Cronyx/Sigma multiport sync/async adapter",0, CLS_COMMS}, -{"rc", "RISCom/8 multiport async adapter", 0, CLS_COMMS}, -{"cy", "Cyclades multiport async adapter", 0, CLS_COMMS}, -{"dgb", "Digiboard PC/Xe, PC/Xi async adapter", 0, CLS_COMMS}, -{"si", "Specialix SI/XIO/SX async adapter", 0, CLS_COMMS}, -{"stl", "Stallion EasyIO/Easy Connection 8/32 async adapter",0, CLS_COMMS}, -{"stli", "Stallion intelligent async adapter" ,0, CLS_COMMS}, -{"ppc", "Parallel Port chipset", 0, CLS_COMMS}, -{"gp", "National Instruments AT-GPIB/TNT driver", 0, CLS_COMMS}, - -{"atkbdc", "Keyboard controller", FLG_INVISIBLE, CLS_INPUT}, -{"atkbd", "Keyboard", FLG_FIXED, CLS_INPUT}, -{"mse", "Microsoft Bus Mouse", 0, CLS_INPUT}, -{"psm", "PS/2 Mouse", FLG_FIXED, CLS_INPUT}, -{"joy", "Joystick", FLG_FIXED, CLS_INPUT}, -{"vt", "PCVT console driver", FLG_IMMUTABLE, CLS_INPUT}, -{"sc", "Syscons console driver", FLG_IMMUTABLE, CLS_INPUT}, - -{"sbc", "PCM Creative SoundBlaster/ESS/Avance sounce cards", 0,CLS_MMEDIA}, -{"gusc", "PCM Gravis UltraSound sound cards", 0, CLS_MMEDIA}, -{"pcm", "PCM Generic soundcard support", 0, CLS_MMEDIA}, -{"sb", "VOXWARE Soundblaster PCM (SB/Pro/16, ProAudio Spectrum)",0,CLS_MMEDIA}, -{"sbxvi", "VOXWARE Soundblaster 16", 0, CLS_MMEDIA}, -{"sbmidi", "VOXWARE Soundblaster MIDI interface", 0, CLS_MMEDIA}, -{"awe", "VOXWARE AWE32 MIDI", 0, CLS_MMEDIA}, -{"pas", "VOXWARE ProAudio Spectrum PCM and MIDI", 0, CLS_MMEDIA}, -{"gus", "VOXWARE Gravis Ultrasound, Ultrasound 16 and Ultrasound MAX",0,CLS_MMEDIA}, -{"gusxvi", "VOXWARE Gravis Ultrasound 16-bit PCM", 0, CLS_MMEDIA}, -{"gusmax", "VOXWARE Gravis Ultrasound MAX", 0, CLS_MMEDIA}, -{"mss", "VOXWARE Microsoft Sound System", 0, CLS_MMEDIA}, -{"opl", "VOXWARE OPL-2/3 FM, SB/Pro/16, ProAudio Spectrum",0,CLS_MMEDIA}, -{"mpu", "VOXWARE Roland MPU401 MIDI", 0, CLS_MMEDIA}, -{"sscape", "VOXWARE Ensoniq Soundscape MIDI interface", 0, CLS_MMEDIA}, -{"sscape_mss", "VOXWARE Ensoniq Soundscape PCM", 0, CLS_MMEDIA}, -{"uart", "VOXWARE 6850 MIDI UART", 0, CLS_MMEDIA}, -{"pca", "PC speaker PCM audio driver", FLG_FIXED, CLS_MMEDIA}, -{"ctx", "Coretex-I frame grabber", 0, CLS_MMEDIA}, -{"spigot", "Creative Labs Video Spigot video capture", 0, CLS_MMEDIA}, -{"scc", "IBM Smart Capture Card", 0, CLS_MMEDIA}, -{"gsc", "Genius GS-4500 hand scanner", 0, CLS_MMEDIA}, -{"asc", "AmiScan scanner", 0, CLS_MMEDIA}, - -{"apm", "Advanced Power Management", FLG_FIXED, CLS_MISC}, -{"pcic", "PC-card controller", 0, CLS_MISC}, -{"npx", "Math coprocessor", FLG_IMMUTABLE, CLS_MISC}, -{"vga", "Catchall PCI VGA driver", FLG_INVISIBLE, CLS_MISC}, -{"","",0,0}}; - - -typedef struct _devlist_struct -{ - char name[80]; - int attrib; /* flag values as per the FLG_* defines above */ - int class; /* disk, etc as per the CLS_* defines above */ - char dev[16]; - int iobase,irq,drq,maddr,msize,unit,flags,id; - int comment; /* 0 = device, 1 = comment, 2 = collapsed comment */ - int conflicts; /* set/reset by findconflict, count of conflicts */ - int changed; /* nonzero if the device has been edited */ - struct uc_device *device; - struct _devlist_struct *prev,*next; -} DEV_LIST; - - -#define DEV_DEVICE 0 -#define DEV_COMMENT 1 -#define DEV_ZOOMED 2 - -#define LIST_CURRENT (1<<0) -#define LIST_SELECTED (1<<1) - -#define KEY_EXIT 0 /* return codes from dolist() and friends */ -#define KEY_DO 1 -#define KEY_DEL 2 -#define KEY_TAB 3 -#define KEY_REDRAW 4 - -#define KEY_UP 5 /* these only returned from editval() */ -#define KEY_DOWN 6 -#define KEY_LEFT 7 -#define KEY_RIGHT 8 -#define KEY_NULL 9 /* this allows us to spin & redraw */ - -#define KEY_ZOOM 10 /* these for zoom all/collapse all */ -#define KEY_UNZOOM 11 - -#define KEY_HELP 12 /* duh? */ - -static void redraw(void); -static void insdev(DEV_LIST *dev, DEV_LIST *list); -static int devinfo(DEV_LIST *dev); -static int visuserconfig(void); - -static DEV_LIST *active = NULL,*inactive = NULL; /* driver lists */ -static DEV_LIST *alist,*ilist; /* visible heads of the driver lists */ -static DEV_LIST scratch; /* scratch record */ -static int conflicts; /* total conflict count */ - - -static char lines[] = "--------------------------------------------------------------------------------"; -static char spaces[] = " "; - - -/** - ** Device manipulation stuff : find, describe, configure. - **/ - -/** - ** setdev - ** - ** Sets the device referenced by (*dev) to the parameters in the struct, - ** and the enable flag according to (enabled) - **/ -static void -setdev(DEV_LIST *dev, int enabled) -{ - dev->device->id_iobase = dev->iobase; /* copy happy */ - dev->device->id_irq = (u_short)(dev->irq < 16 ? 1<<dev->irq : 0); /* IRQ is bitfield */ - dev->device->id_drq = (short)dev->drq; - dev->device->id_maddr = (caddr_t)dev->maddr; - dev->device->id_msize = dev->msize; - dev->device->id_flags = dev->flags; - dev->device->id_enabled = enabled; -} - - -/** - ** getdevs - ** - ** Walk the kernel device tables and build the active and inactive lists - **/ -static void -getdevs(void) -{ - int i; - struct uc_device *ap; - - ap = uc_devtab; /* pointer to array of devices */ - for (i = 0; ap[i].id_id; i++) /* for each device in this table */ - { - scratch.unit = ap[i].id_unit; /* device parameters */ - strcpy(scratch.dev,ap[i].id_name); - scratch.iobase = ap[i].id_iobase; - scratch.irq = ffs(ap[i].id_irq)-1; - scratch.drq = ap[i].id_drq; - scratch.maddr = (int)ap[i].id_maddr; - scratch.msize = ap[i].id_msize; - scratch.flags = ap[i].id_flags; - - scratch.comment = DEV_DEVICE; /* admin stuff */ - scratch.conflicts = 0; - scratch.device = &ap[i]; /* save pointer for later reference */ - scratch.changed = 0; - if (!devinfo(&scratch)) /* get more info on the device */ - insdev(&scratch,ap[i].id_enabled?active:inactive); - } -} - - -/** - ** Devinfo - ** - ** Fill in (dev->name), (dev->attrib) and (dev->type) from the device_info array. - ** If the device is unknown, put it in the CLS_MISC class, with no flags. - ** - ** If the device is marked "invisible", return nonzero; the caller should - ** not insert any such device into either list. - ** - **/ -static int -devinfo(DEV_LIST *dev) -{ - int i; - - for (i = 0; device_info[i].class; i++) - { - if (!strcmp(dev->dev,device_info[i].dev)) - { - if (device_info[i].attrib & FLG_INVISIBLE) /* forget we ever saw this one */ - return(1); - strcpy(dev->name,device_info[i].name); /* get the name */ - dev->attrib = device_info[i].attrib; - dev->class = device_info[i].class; - return(0); - } - } - strcpy(dev->name,"Unknown device"); - dev->attrib = 0; - dev->class = CLS_MISC; - return(0); -} - - -/** - ** List manipulation stuff : add, move, initialise, free, traverse - ** - ** Note that there are assumptions throughout this code that - ** the first entry in a list will never move. (assumed to be - ** a comment). - **/ - - -/** - ** Adddev - ** - ** appends a copy of (dev) to the end of (*list) - **/ -static void -addev(DEV_LIST *dev, DEV_LIST **list) -{ - - DEV_LIST *lp,*ap; - - lp = (DEV_LIST *)malloc(sizeof(DEV_LIST),M_DEVL,M_WAITOK); - bcopy(dev,lp,sizeof(DEV_LIST)); /* create copied record */ - - if (*list) /* list exists */ - { - ap = *list; - while(ap->next) - ap = ap->next; /* scoot to end of list */ - lp->prev = ap; - lp->next = NULL; - ap->next = lp; - }else{ /* list does not yet exist */ - *list = lp; - lp->prev = lp->next = NULL; /* list now exists */ - } -} - - -/** - ** Findspot - ** - ** Finds the 'appropriate' place for (dev) in (list) - ** - ** 'Appropriate' means in numeric order with other devices of the same type, - ** or in alphabetic order following a comment of the appropriate type. - ** or at the end of the list if an appropriate comment is not found. (this should - ** never happen) - ** (Note that the appropriate point is never the top, but may be the bottom) - **/ -static DEV_LIST * -findspot(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *ap = NULL; - - /* search for a previous instance of the same device */ - for (ap = list; ap; ap = ap->next) - { - if (ap->comment != DEV_DEVICE) /* ignore comments */ - continue; - if (!strcmp(dev->dev,ap->dev)) /* same base device */ - { - if ((dev->unit <= ap->unit) /* belongs before (equal is bad) */ - || !ap->next) /* or end of list */ - { - ap = ap->prev; /* back up one */ - break; /* done here */ - } - if (ap->next) /* if the next item exists */ - { - if (ap->next->comment != DEV_DEVICE) /* next is a comment */ - break; - if (strcmp(dev->dev,ap->next->dev)) /* next is a different device */ - break; - } - } - } - - if (!ap) /* not sure yet */ - { - /* search for a class that the device might belong to */ - for (ap = list; ap; ap = ap->next) - { - if (ap->comment != DEV_DEVICE) /* look for simlar devices */ - continue; - if (dev->class != ap->class) /* of same class too 8) */ - continue; - if (strcmp(dev->dev,ap->dev) < 0) /* belongs before the current entry */ - { - ap = ap->prev; /* back up one */ - break; /* done here */ - } - if (ap->next) /* if the next item exists */ - if (ap->next->comment != DEV_DEVICE) /* next is a comment, go here */ - break; - } - } - - if (!ap) /* didn't find a match */ - { - for (ap = list; ap->next; ap = ap->next) /* try for a matching comment */ - if ((ap->comment != DEV_DEVICE) - && (ap->class == dev->class)) /* appropriate place? */ - break; - } /* or just put up with last */ - - return(ap); -} - - -/** - ** Insdev - ** - ** Inserts a copy of (dev) at the appropriate point in (list) - **/ -static void -insdev(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *lp,*ap; - - lp = (DEV_LIST *)malloc(sizeof(DEV_LIST),M_DEVL,M_WAITOK); - bcopy(dev,lp,sizeof(DEV_LIST)); /* create copied record */ - - ap = findspot(lp,list); /* find appropriate spot */ - lp->next = ap->next; /* point to next */ - if (ap->next) - ap->next->prev = lp; /* point next to new */ - lp->prev = ap; /* point new to current */ - ap->next = lp; /* and current to new */ -} - - -/** - ** Movedev - ** - ** Moves (dev) from its current list to an appropriate place in (list) - ** (dev) may not come from the top of a list, but it may from the bottom. - **/ -static void -movedev(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *ap; - - ap = findspot(dev,list); - dev->prev->next = dev->next; /* remove from old list */ - if (dev->next) - dev->next->prev = dev->prev; - - dev->next = ap->next; /* insert in new list */ - if (ap->next) - ap->next->prev = dev; /* point next to new */ - dev->prev = ap; /* point new to current */ - ap->next = dev; /* and current to new */ -} - - -/** - ** Initlist - ** - ** Initialises (*list) with the basic headings - **/ -static void -initlist(DEV_LIST **list) -{ - int i; - - for(i = 0; devclass_names[i].name[0]; i++) /* for each devtype name */ - { - strcpy(scratch.name,devclass_names[i].name); - scratch.comment = DEV_ZOOMED; - scratch.class = devclass_names[i].number; - scratch.attrib = FLG_MANDATORY; /* can't be moved */ - addev(&scratch,list); /* add to the list */ - } -} - - -/** - ** savelist - ** - ** Walks (list) and saves the settings of any entry marked as changed. - ** - ** The device's active field is set according to (active). - ** - ** Builds the uc_devlist used by kget to extract the changed device information. - ** The code for this was taken almost verbatim from the original module. - **/ -static void -savelist(DEV_LIST *list, int active) -{ - struct uc_device *id_p,*id_pn; - char *name; - - while (list) - { - if ((list->comment == DEV_DEVICE) && /* is a device */ - (list->changed) && /* has been changed */ - (list->device != NULL)) { /* has an uc_device structure */ - - setdev(list,active); /* set the device itself */ - - id_pn = NULL; - for (id_p=uc_devlist; id_p; id_p=id_p->id_next) - { /* look on the list for it */ - if (id_p->id_id == list->device->id_id) - { - name = list->device->id_name; - id_pn = id_p->id_next; - if (id_p->id_name) - free(id_p->id_name, M_DEVL); - bcopy(list->device,id_p,sizeof(struct uc_device)); - save_resource(list->device); - id_p->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_p->id_name, name); - id_pn->id_next = uc_devlist; - id_p->id_next = id_pn; - break; - } - } - if (!id_pn) /* not already on the list */ - { - name = list->device->id_name; - id_pn = malloc(sizeof(struct uc_device),M_DEVL,M_WAITOK); - bcopy(list->device,id_pn,sizeof(struct uc_device)); - save_resource(list->device); - id_pn->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_pn->id_name, name); - id_pn->id_next = uc_devlist; - uc_devlist = id_pn; /* park at top of list */ - } - } - list = list->next; - } -} - - -/** - ** nukelist - ** - ** Frees all storage in use by a (list). - **/ -static void -nukelist(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return; - while(list->prev) /* walk to head of list */ - list = list->prev; - - while(list) - { - dp = list; - list = list->next; - free(dp,M_DEVL); - } -} - - -/** - ** prevent - ** - ** Returns the previous entry in (list), skipping zoomed regions. Returns NULL - ** if there is no previous entry. (Only possible if list->prev == NULL given the - ** premise that there is always a comment at the head of the list) - **/ -static DEV_LIST * -prevent(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return(NULL); - dp = list->prev; /* start back one */ - while(dp) - { - if (dp->comment == DEV_ZOOMED) /* previous section is zoomed */ - return(dp); /* so skip to comment */ - if (dp->comment == DEV_COMMENT) /* not zoomed */ - return(list->prev); /* one back as normal */ - dp = dp->prev; /* backpedal */ - } - return(dp); /* NULL, we can assume */ -} - - -/** - ** nextent - ** - ** Returns the next entry in (list), skipping zoomed regions. Returns NULL - ** if there is no next entry. (Possible if the current entry is last, or - ** if the current entry is the last heading and it's collapsed) - **/ -static DEV_LIST * -nextent(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return(NULL); - if (list->comment != DEV_ZOOMED) /* no reason to skip */ - return(list->next); - dp = list->next; - while(dp) - { - if (dp->comment != DEV_DEVICE) /* found another heading */ - break; - dp = dp->next; - } - return(dp); /* back we go */ -} - - -/** - ** ofsent - ** - ** Returns the (ofs)th entry down from (list), or NULL if it doesn't exist - **/ -static DEV_LIST * -ofsent(int ofs, DEV_LIST *list) -{ - while (ofs-- && list) - list = nextent(list); - return(list); -} - - -/** - ** findconflict - ** - ** Scans every element of (list) and sets the conflict tags appropriately - ** Returns the number of conflicts found. - **/ -static int -findconflict(DEV_LIST *list) -{ - int count = 0; /* number of conflicts found */ - DEV_LIST *dp,*sp; - - for (dp = list; dp; dp = dp->next) /* over the whole list */ - { - if (dp->comment != DEV_DEVICE) /* comments don't usually conflict */ - continue; - - dp->conflicts = 0; /* assume the best */ - for (sp = list; sp; sp = sp->next) /* scan the entire list for conflicts */ - { - if (sp->comment != DEV_DEVICE) /* likewise */ - continue; - - if (sp == dp) /* always conflict with itself */ - continue; - - if ((dp->iobase > 0) && /* iobase conflict? */ - (dp->iobase == sp->iobase)) - dp->conflicts = 1; - if ((dp->irq > 0) && /* irq conflict? */ - (dp->irq == sp->irq)) - dp->conflicts = 1; - if ((dp->drq > 0) && /* drq conflict? */ - (dp->drq == sp->drq)) - dp->conflicts = 1; - if ((sp->maddr > 0) && /* maddr/msize conflict? */ - (dp->maddr > 0) && - (sp->maddr + ((sp->msize == 0) ? 1 : sp->msize) > dp->maddr) && - (dp->maddr + ((dp->msize == 0) ? 1 : dp->msize) > sp->maddr)) - dp->conflicts = 1; - } - count += dp->conflicts; /* count conflicts */ - } - return(count); -} - - -/** - ** expandlist - ** - ** Unzooms all headings in (list) - **/ -static void -expandlist(DEV_LIST *list) -{ - while(list) - { - if (list->comment == DEV_COMMENT) - list->comment = DEV_ZOOMED; - list = list->next; - } -} - - -/** - ** collapselist - ** - ** Zooms all headings in (list) - **/ -static void -collapselist(DEV_LIST *list) -{ - while(list) - { - if (list->comment == DEV_ZOOMED) - list->comment = DEV_COMMENT; - list = list->next; - } -} - - -/** - ** Screen-manipulation stuff - ** - ** This is the basic screen layout : - ** - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 0 -|---Active Drivers----------------------------xx Conflicts------Dev---IRQ--Port--| - ** 1 -| ........................ ....... .. 0x....| - ** 2 -| ........................ ....... .. 0x....| - ** 3 -| ........................ ....... .. 0x....| - ** 4 -| ........................ ....... .. 0x....| - ** 5 -| ........................ ....... .. 0x....| - ** 6 -| ........................ ....... .. 0x....| - ** 7 -| ........................ ....... .. 0x....| - ** 8 -| ........................ ....... .. 0x....| - ** 9 -|---Inactive Drivers--------------------------------------------Dev--------------| - ** 10-| ........................ ....... | - ** 11-| ........................ ....... | - ** 12-| ........................ ....... | - ** 13-| ........................ ....... | - ** 14-| ........................ ....... | - ** 15-| ........................ ....... | - ** 16-| ........................ ....... | - ** 17-|------------------------------------------------------UP-DOWN-------------------| - ** 18-| Relevant parameters for the current device | - ** 19-| | - ** 20-| | - ** 21-|--------------------------------------------------------------------------------| - ** 22-| Help texts go here | - ** 23-| | - ** +--------------------------------------------------------------------------------+ - ** - ** Help texts - ** - ** On a collapsed comment : - ** - ** [Enter] Expand device list [z] Expand all lists - ** [TAB] Change fields [Q] Save and Exit - ** - ** On an expanded comment : - ** - ** [Enter] Collapse device list [Z] Collapse all lists - ** [TAB] Change fields [Q] Save and Exit - ** - ** On a comment with no followers - ** - ** - ** [TAB] Change fields [Q] Save and Exit - ** - ** On a device in the active list - ** - ** [Enter] Edit device parameters [DEL] Disable device - ** [TAB] Change fields [Q] Save and Exit [?] Help - ** - ** On a device in the inactive list - ** - ** [Enter] Enable device - ** [TAB] Change fields [Q] Save and Exit [?] Help - ** - ** While editing parameters - ** - ** <parameter-specific help here> - ** [TAB] Change fields [Q] Save device parameters - **/ - - - -/** - ** - ** The base-level screen primitives : - ** - ** bold() - enter bold mode \E[1m (md) - ** inverse() - enter inverse mode \E[7m (so) - ** normal() - clear bold/inverse mode \E[m (se) - ** clear() - clear the screen \E[H\E[J (ce) - ** move(x,y) - move the cursor to x,y \E[y;xH: (cm) - **/ - -static void -bold(void) -{ - printf("\033[1m"); -} - -static void -inverse(void) -{ - printf("\033[7m"); -} - -static void -normal(void) -{ - printf("\033[m"); -} - -static void -clear(void) -{ - normal(); - printf("\033[H\033[J"); -} - -static void -move(int x, int y) -{ - printf("\033[%d;%dH",y+1,x+1); -} - - -/** - ** - ** High-level screen primitives : - ** - ** putxyl(x,y,str,len) - put (len) bytes of (str) at (x,y), supports embedded formatting - ** putxy(x,y,str) - put (str) at (x,y), supports embedded formatting - ** erase(x,y,w,h) - clear the box (x,y,w,h) - ** txtbox(x,y,w,y,str) - put (str) in a region at (x,y,w,h) - ** putmsg(str) - put (str) in the message area - ** puthelp(str) - put (str) in the upper helpline - ** pad(str,len) - pad (str) to (len) with spaces - ** drawline(row,detail,list,inverse,*dhelp) - ** - draws a line for (*list) at (row) onscreen. If (detail) is - ** nonzero, include port, IRQ and maddr, if (inverse) is nonzero, - ** draw the line in inverse video, and display (*dhelp) on the - ** helpline. - ** drawlist(row,num,detail,list) - ** - draw (num) entries from (list) at (row) onscreen, passile (detail) - ** through to drawline(). - ** showparams(dev) - displays the relevant parameters for (dev) below the lists onscreen. - ** yesno(str) - displays (str) in the message area, and returns nonzero on 'y' or 'Y' - ** redraw(); - Redraws the entire screen layout, including the - ** - two list panels. - **/ - -/** - ** putxy - ** writes (str) at x,y onscreen - ** putxyl - ** writes up to (len) of (str) at x,y onscreen. - ** - ** Supports embedded formatting : - ** !i - inverse mode. - ** !b - bold mode. - ** !n - normal mode. - **/ -static void -putxyl(int x, int y, char *str, int len) -{ - move(x,y); - normal(); - - while((*str) && (len--)) - { - if (*str == '!') /* format escape? */ - { - switch(*(str+1)) /* depending on the next character */ - { - case 'i': - inverse(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - case 'b': - bold(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - case 'n': - normal(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - default: - putchar(*str++); /* not an escape */ - } - }else{ - putchar(*str++); /* emit the character */ - } - } -} - -#define putxy(x,y,str) putxyl(x,y,str,-1) - - -/** - ** erase - ** - ** Erases the region (x,y,w,h) - **/ -static void -erase(int x, int y, int w, int h) -{ - int i; - - normal(); - for (i = 0; i < h; i++) - putxyl(x,y++,spaces,w); -} - - -/** - ** txtbox - ** - ** Writes (str) into the region (x,y,w,h), supports embedded formatting using - ** putxy. Lines are not wrapped, newlines must be forced with \n. - **/ -static void -txtbox(int x, int y, int w, int h, char *str) -{ - int i = 0; - - h--; - while((str[i]) && h) - { - if (str[i] == '\n') /* newline */ - { - putxyl(x,y,str,(i<w)?i:w); /* write lesser of i or w */ - y++; /* move down */ - h--; /* room for one less */ - str += (i+1); /* skip first newline */ - i = 0; /* zero offset */ - }else{ - i++; /* next character */ - } - } - if (h) /* end of string, not region */ - putxyl(x,y,str,w); -} - - -/** - ** putmsg - ** - ** writes (msg) in the helptext area - **/ -static void -putmsg(char *msg) -{ - erase(0,18,80,3); /* clear area */ - txtbox(0,18,80,3,msg); -} - - -/** - ** puthelp - ** - ** Writes (msg) in the helpline area - **/ -static void -puthelp(char *msg) -{ - erase(0,22,80,1); - putxy(0,22,msg); -} - - -/** - ** masterhelp - ** - ** Draws the help message at the bottom of the screen - **/ -static void -masterhelp(char *msg) -{ - erase(0,23,80,1); - putxy(0,23,msg); -} - - -/** - ** pad - ** - ** space-pads a (str) to (len) characters - **/ -static void -pad(char *str, int len) -{ - int i; - - for (i = 0; str[i]; i++) /* find the end of the string */ - ; - if (i >= len) /* no padding needed */ - return; - while(i < len) /* pad */ - str[i++] = ' '; - str[i] = '\0'; -} - - -/** - ** drawline - ** - ** Displays entry (ofs) of (list) in region at (row) onscreen, optionally displaying - ** the port and IRQ fields if (detail) is nonzero. If (inverse), in inverse video. - ** - ** The text (dhelp) is displayed if the item is a normal device, otherwise - ** help is shown for normal or zoomed comments - **/ -static void -drawline(int row, int detail, DEV_LIST *list, int inverse, char *dhelp) -{ - char lbuf[90],nb[70],db[20],ib[16],pb[16]; - - if (list->comment == DEV_DEVICE) - { - nb[0] = ' '; - strncpy(nb+1,list->name,57); - }else{ - strncpy(nb,list->name,58); - if ((list->comment == DEV_ZOOMED) && (list->next)) - if (list->next->comment == DEV_DEVICE) /* only mention if there's something hidden */ - strcat(nb," (Collapsed)"); - } - nb[58] = '\0'; - pad(nb,60); - if (list->conflicts) /* device in conflict? */ - { - if (inverse) - { - strcpy(nb+54," !nCONF!i "); /* tag conflict, careful of length */ - }else{ - strcpy(nb+54," !iCONF!n "); /* tag conflict, careful of length */ - } - } - if (list->comment == DEV_DEVICE) - { - sprintf(db,"%s%d",list->dev,list->unit); - pad(db,8); - }else{ - strcpy(db," "); - } - if ((list->irq > 0) && detail && (list->comment == DEV_DEVICE)) - { - sprintf(ib," %d",list->irq); - pad(ib,4); - }else{ - strcpy(ib," "); - } - if ((list->iobase > 0) && detail && (list->comment == DEV_DEVICE)) - { - sprintf(pb,"0x%x",list->iobase); - pad(pb,7); - }else{ - strcpy(pb," "); - } - - sprintf(lbuf," %s%s%s%s%s",inverse?"!i":"",nb,db,ib,pb); - - putxyl(0,row,lbuf,80); - if (dhelp) - { - switch(list->comment) - { - case DEV_DEVICE: /* ordinary device */ - puthelp(dhelp); - break; - case DEV_COMMENT: - puthelp(""); - if (list->next) - if (list->next->comment == DEV_DEVICE) - puthelp(" [!bEnter!n] Collapse device list [!bC!n] Collapse all lists"); - break; - case DEV_ZOOMED: - puthelp(""); - if (list->next) - if (list->next->comment == DEV_DEVICE) - puthelp(" [!bEnter!n] Expand device list [!bX!n] Expand all lists"); - break; - default: - puthelp(" WARNING: This list entry corrupted!"); - break; - } - } - move(0,row); /* put the cursor somewhere relevant */ -} - - -/** - ** drawlist - ** - ** Displays (num) lines of the contents of (list) at (row), optionally displaying the - ** port and IRQ fields as well if (detail) is nonzero - ** - ** printf in the kernel is essentially useless, so we do most of the hard work ourselves here. - **/ -static void -drawlist(int row, int num, int detail, DEV_LIST *list) -{ - int ofs; - - for(ofs = 0; ofs < num; ofs++) - { - if (list) - { - drawline(row+ofs,detail,list,0,NULL); /* NULL -> don't draw empty help string */ - list = nextent(list); /* move down visible list */ - }else{ - erase(0,row+ofs,80,1); - } - } -} - - -/** - ** redrawactive - ** - ** Redraws the active list - **/ -static void -redrawactive(void) -{ - char cbuf[16]; - - if (conflicts) - { - sprintf(cbuf,"!i%d conflict%s-",conflicts,(conflicts>1)?"s":""); - putxy(45,0,cbuf); - }else{ - putxyl(45,0,lines,16); - } - drawlist(1,8,1,alist); /* draw device lists */ -} - -/** - ** redrawinactive - ** - ** Redraws the inactive list - **/ -static void -redrawinactive(void) -{ - drawlist(10,7,0,ilist); /* draw device lists */ -} - - -/** - ** redraw - ** - ** Clear the screen and redraw the entire layout - **/ -static void -redraw(void) -{ - clear(); - putxy(0,0,lines); - putxy(3,0,"!bActive!n-!bDrivers"); - putxy(63,0,"!bDev!n---!bIRQ!n--!bPort"); - putxy(0,9,lines); - putxy(3,9,"!bInactive!n-!bDrivers"); - putxy(63,9,"!bDev"); - putxy(0,17,lines); - putxy(0,21,lines); - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save and Exit [!b?!n] Help"); - - redrawactive(); - redrawinactive(); -} - - -/** - ** yesnocancel - ** - ** Put (str) in the message area, and return 1 if the user hits 'y' or 'Y', - ** 2 if they hit 'c' or 'C', or 0 for 'n' or 'N'. - **/ -static int -yesnocancel(char *str) -{ - - putmsg(str); - for(;;) - switch(getchar()) - { - case -1: - case 'n': - case 'N': - return(0); - - case 'y': - case 'Y': - return(1); - - case 'c': - case 'C': - return(2); - } -} - - -/** - ** showparams - ** - ** Show device parameters in the region below the lists - ** - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 17-|--------------------------------------------------------------------------------| - ** 18-| Port address : 0x0000 Memory address : 0x00000 Conflict allowed | - ** 19-| IRQ number : 00 Memory size : 0x0000 | - ** 20-| Flags : 0x0000 DRQ number : 00 | - ** 21-|--------------------------------------------------------------------------------| - **/ -static void -showparams(DEV_LIST *dev) -{ - char buf[80]; - - erase(0,18,80,3); /* clear area */ - if (!dev) - return; - if (dev->comment != DEV_DEVICE) - return; - - - if (dev->iobase > 0) - { - sprintf(buf,"Port address : 0x%x",dev->iobase); - putxy(1,18,buf); - } - - if (dev->irq > 0) - { - sprintf(buf,"IRQ number : %d",dev->irq); - putxy(1,19,buf); - } - sprintf(buf,"Flags : 0x%x",dev->flags); - putxy(1,20,buf); - if (dev->maddr > 0) - { - sprintf(buf,"Memory address : 0x%x",dev->maddr); - putxy(26,18,buf); - } - if (dev->msize > 0) - { - sprintf(buf,"Memory size : 0x%x",dev->msize); - putxy(26,19,buf); - } - - if (dev->drq > 0) - { - sprintf(buf,"DRQ number : %d",dev->drq); - putxy(26,20,buf); - } -} - - -/** - ** Editing functions for device parameters - ** - ** editval(x,y,width,hex,min,max,val) - Edit (*val) in a field (width) wide at (x,y) - ** onscreen. Refuse values outsise (min) and (max). - ** editparams(dev) - Edit the parameters for (dev) - **/ - - -#define VetRet(code) \ -{ \ - if ((i >= min) && (i <= max)) /* legit? */ \ - { \ - *val = i; \ - sprintf(buf,hex?"0x%x":"%d",i); \ - putxy(hex?x-2:x,y,buf); \ - return(code); /* all done and exit */ \ - } \ - i = *val; /* restore original value */ \ - delta = 1; /* restore other stuff */ \ -} - - -/** - ** editval - ** - ** Edit (*val) at (x,y) in (hex)?hex:decimal mode, allowing values between (min) and (max) - ** in a field (width) wide. (Allow one space) - ** If (ro) is set, we're in "readonly" mode, so disallow edits. - ** - ** Return KEY_TAB on \t, KEY_EXIT on 'q' - **/ -static int -editval(int x, int y, int width, int hex, int min, int max, int *val, int ro) -{ - int i = *val; /* work with copy of the value */ - char buf[2+11+1],tc[11+1]; /* display buffer, text copy */ - int xp = 0; /* cursor offset into text copy */ - int delta = 1; /* force redraw first time in */ - int c; - int extended = 0; /* stage counter for extended key sequences */ - - if (hex) /* we presume there's a leading 0x onscreen */ - putxy(x-2,y,"!i0x"); /* coz there sure is now */ - - for (;;) - { - if (delta) /* only update if necessary */ - { - sprintf(tc,hex?"%x":"%d",i); /* make a text copy of the value */ - sprintf(buf,"!i%s",tc); /* format for printing */ - erase(x,y,width,1); /* clear the area */ - putxy(x,y,buf); /* write */ - xp = strlen(tc); /* cursor always at end */ - move(x+xp,y); /* position the cursor */ - } - - c = getchar(); - - switch(extended) /* escape handling */ - { - case 0: - if (c == 0x1b) /* esc? */ - { - extended = 1; /* flag and spin */ - continue; - } - extended = 0; - break; /* nope, drop through */ - - case 1: /* there was an escape prefix */ - if (c == '[' || c == 'O') /* second character in sequence */ - { - extended = 2; - continue; - } - if (c == 0x1b) - return(KEY_EXIT); /* double esc exits */ - extended = 0; - break; /* nup, not a sequence. */ - - case 2: - extended = 0; - switch(c) /* looks like the real McCoy */ - { - case 'A': - VetRet(KEY_UP); /* leave if OK */ - continue; - case 'B': - VetRet(KEY_DOWN); /* leave if OK */ - continue; - case 'C': - VetRet(KEY_RIGHT); /* leave if OK */ - continue; - case 'D': - VetRet(KEY_LEFT); /* leave if OK */ - continue; - - default: - continue; - } - } - - switch(c) - { - case '\t': /* trying to tab off */ - VetRet(KEY_TAB); /* verify and maybe return */ - break; - - case -1: - case 'q': - case 'Q': - VetRet(KEY_EXIT); - break; - - case '\b': - case '\177': /* BS or DEL */ - if (ro) /* readonly? */ - { - puthelp(" !iThis value cannot be edited (Press ESC)"); - while(getchar() != 0x1b); /* wait for key */ - return(KEY_NULL); /* spin */ - } - if (xp) /* still something left to delete */ - { - i = (hex ? i/0x10u : i/10); /* strip last digit */ - delta = 1; /* force update */ - } - break; - - case 588: - VetRet(KEY_UP); - break; - - case '\r': - case '\n': - case 596: - VetRet(KEY_DOWN); - break; - - case 591: - VetRet(KEY_LEFT); - break; - - case 593: - VetRet(KEY_RIGHT); - break; - - default: - if (ro) /* readonly? */ - { - puthelp(" !iThis value cannot be edited (Press ESC)"); - while(getchar() != 0x1b); /* wait for key */ - return(KEY_NULL); /* spin */ - } - if (xp >= width) /* no room for more characters anyway */ - break; - if (hex) - { - if ((c >= '0') && (c <= '9')) - { - i = i*0x10 + (c-'0'); /* update value */ - delta = 1; - break; - } - if ((c >= 'a') && (c <= 'f')) - { - i = i*0x10 + (c-'a'+0xa); - delta = 1; - break; - } - if ((c >= 'A') && (c <= 'F')) - { - i = i*0x10 + (c-'A'+0xa); - delta = 1; - break; - } - }else{ - if ((c >= '0') && (c <= '9')) - { - i = i*10 + (c-'0'); /* update value */ - delta = 1; /* force redraw */ - break; - } - } - break; - } - } -} - - -/** - ** editparams - ** - ** Edit the parameters for (dev) - ** - ** Note that it's _always_ possible to edit the flags, otherwise it might be - ** possible for this to spin in an endless loop... - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 17-|--------------------------------------------------------------------------------| - ** 18-| Port address : 0x0000 Memory address : 0x00000 Conflict allowed | - ** 19-| IRQ number : 00 Memory size : 0x0000 | - ** 20-| Flags : 0x0000 DRQ number : 00 | - ** 21-|--------------------------------------------------------------------------------| - ** - ** The "intelligence" in this function that hops around based on the directional - ** returns from editval isn't very smart, and depends on the layout above. - **/ -static void -editparams(DEV_LIST *dev) -{ - int ret; - char buf[16]; /* needs to fit the device name */ - - putxy(2,17,"!bParameters!n-!bfor!n-!bdevice!n-"); - sprintf(buf,"!b%s",dev->dev); - putxy(24,17,buf); - - erase(1,22,80,1); - for (;;) - { - ep_iobase: - if (dev->iobase > 0) - { - puthelp(" IO Port address (Hexadecimal, 0x1-0xffff)"); - ret = editval(18,18,5,1,0x1,0xffff,&(dev->iobase),(dev->attrib & FLG_FIXIOBASE)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->maddr > 0) - goto ep_maddr; - break; - - case KEY_TAB: - case KEY_DOWN: - goto ep_irq; - } - goto ep_iobase; - } - ep_irq: - if (dev->irq > 0) - { - puthelp(" Interrupt number (Decimal, 1-15)"); - ret = editval(16,19,3,0,1,15,&(dev->irq),(dev->attrib & FLG_FIXIRQ)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->msize > 0) - goto ep_msize; - break; - - case KEY_UP: - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_TAB: - case KEY_DOWN: - goto ep_flags; - } - goto ep_irq; - } - ep_flags: - puthelp(" Device-specific flag values."); - ret = editval(18,20,8,1,INT_MIN,INT_MAX,&(dev->flags),0); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_UP: - if (dev->irq > 0) - goto ep_irq; - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_DOWN: - if (dev->maddr > 0) - goto ep_maddr; - if (dev->msize > 0) - goto ep_msize; - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_maddr; - } - goto ep_flags; - ep_maddr: - if (dev->maddr > 0) - { - puthelp(" Device memory start address (Hexadecimal, 0x1-0xfffff)"); - ret = editval(45,18,6,1,0x1,0xfffff,&(dev->maddr),(dev->attrib & FLG_FIXMADDR)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_UP: - goto ep_flags; - - case KEY_DOWN: - if (dev->msize > 0) - goto ep_msize; - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_msize; - } - goto ep_maddr; - } - ep_msize: - if (dev->msize > 0) - { - puthelp(" Device memory size (Hexadecimal, 0x1-0x10000)"); - ret = editval(45,19,5,1,0x1,0x10000,&(dev->msize),(dev->attrib & FLG_FIXMSIZE)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - if (dev->irq > 0) - goto ep_irq; - break; - - case KEY_UP: - if (dev->maddr > 0) - goto ep_maddr; - goto ep_flags; - - case KEY_DOWN: - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_drq; - } - goto ep_msize; - } - ep_drq: - if (dev->drq > 0) - { - puthelp(" Device DMA request number (Decimal, 1-7)"); - ret = editval(43,20,2,0,1,7,&(dev->drq),(dev->attrib & FLG_FIXDRQ)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - goto ep_flags; - - case KEY_UP: - if (dev->msize > 0) - goto ep_msize; - if (dev->maddr > 0) - goto ep_maddr; - goto ep_flags; - - case KEY_TAB: - goto ep_iobase; - } - goto ep_drq; - } - } - ep_exit: - dev->changed = 1; /* mark as changed */ -} - -static char *helptext[] = -{ - " Using the UserConfig kernel settings editor", - " -------------------------------------------", - "", - "VISUAL MODE:", - "", - "- - Layout -", - "", - "The screen displays a list of available drivers, divided into two", - "scrolling lists: Active Drivers, and Inactive Drivers. Each list is", - "by default collapsed and can be expanded to show all the drivers", - "available in each category. The parameters for the currently selected", - "driver are shown at the bottom of the screen.", - "", - "- - Moving around -", - "", - "To move in the current list, use the UP and DOWN cursor keys to select", - "an item (the selected item will be highlighted). If the item is a", - "category name, you may alternatively expand or collapse the list of", - "drivers for that category by pressing [!bENTER!n]. Once the category is", - "expanded, you can select each driver in the same manner and either:", - "", - " - change its parameters using [!bENTER!n]", - " - move it to the Inactive list using [!bDEL!n]", - "", - "Use the [!bTAB!n] key to toggle between the Active and Inactive list; if", - "you need to move a driver from the Inactive list back to the Active", - "one, select it in the Inactive list, using [!bTAB!n] to change lists if", - "necessary, and press [!bENTER!n] -- the device will be moved back to", - "its place in the Active list.", - "", - "- - Altering the list/parameters -", - "", - "Any drivers for devices not installed in your system should be moved", - "to the Inactive list, until there are no remaining parameter conflicts", - "between the drivers, as indicated at the top.", - "", - "Once the list of Active drivers only contains entries for the devices", - "present in your system, you can set their parameters (Interrupt, DMA", - "channel, I/O addresses). To do this, select the driver and press", - "[!bENTER!n]: it is now possible to edit the settings at the", - "bottom of the screen. Use [!bTAB!n] to change fields, and when you are", - "finished, use [!bQ!n] to return to the list.", - "", - "- - Saving changes -", - "", - "When all settings seem correct, and you wish to proceed with the", - "kernel device probing and boot, press [!bQ!n] -- you will be asked to", - "confirm your choice.", - "", - NULL -}; - - -/** - ** helpscreen - ** - ** Displays help text onscreen for people that are confused, using a simple - ** pager. - **/ -static void -helpscreen(void) -{ - int topline = 0; /* where we are in the text */ - int line = 0; /* last line we displayed */ - int c, delta = 1; - char prompt[80]; - - for (;;) /* loop until user quits */ - { - /* display help text */ - if (delta) - { - clear(); /* remove everything else */ - for (line = topline; - (line < (topline + 24)) && (helptext[line]); - line++) - putxy(0,line-topline,helptext[line]); - delta = 0; - } - - /* prompt */ - sprintf(prompt,"!i --%s-- [U]p [D]own [Q]uit !n",helptext[line] ? "MORE" : "END"); - putxy(0,24,prompt); - - c = getchar(); /* so what do they say? */ - - switch (c) - { - case 'u': - case 'U': - case 'b': - case 'B': /* wired into 'more' users' fingers */ - if (topline > 0) /* room to go up? */ - { - topline -= 24; - if (topline < 0) /* don't go too far */ - topline = 0; - delta = 1; - } - break; - - case 'd': - case 'D': - case ' ': /* expected by most people */ - if (helptext[line]) /* maybe more below? */ - { - topline += 24; - delta = 1; - } - break; - - case 'q': - case 'Q': - redraw(); /* restore the screen */ - return; - } - } -} - - -/** - ** High-level control functions - **/ - - -/** - ** dolist - ** - ** Handle user movement within (*list) in the region starting at (row) onscreen with - ** (num) lines, starting at (*ofs) offset from row onscreen. - ** Pass (detail) on to drawing routines. - ** - ** If the user hits a key other than a cursor key, maybe return a code. - ** - ** (*list) points to the device at the top line in the region, (*ofs) is the - ** position of the highlight within the region. All routines below - ** this take only a device and an absolute row : use ofsent() to find the - ** device, and add (*ofs) to (row) to find the absolute row. - **/ -static int -dolist(int row, int num, int detail, int *ofs, DEV_LIST **list, char *dhelp) -{ - int extended = 0; - int c; - DEV_LIST *lp; - int delta = 1; - - for(;;) - { - if (delta) - { - showparams(ofsent(*ofs,*list)); /* show device parameters */ - drawline(row+*ofs,detail,ofsent(*ofs,*list),1,dhelp); /* highlight current line */ - delta = 0; - } - - c = getchar(); /* get a character */ - if ((extended == 2) || (c==588) || (c==596)) /* console gives "alternative" codes */ - { - extended = 0; /* no longer */ - switch(c) - { - case 588: /* syscons' idea of 'up' */ - case 'A': /* up */ - if (*ofs) /* just a move onscreen */ - { - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp);/* unhighlight current line */ - (*ofs)--; /* move up */ - }else{ - lp = prevent(*list); /* can we go up? */ - if (!lp) /* no */ - break; - *list = lp; /* yes, move up list */ - drawlist(row,num,detail,*list); - } - delta = 1; - break; - - case 596: /* dooby-do */ - case 'B': /* down */ - lp = ofsent(*ofs,*list); /* get current item */ - if (!nextent(lp)) - break; /* nothing more to move to */ - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp); /* unhighlight current line */ - if (*ofs < (num-1)) /* room to move onscreen? */ - { - (*ofs)++; - }else{ - *list = nextent(*list); /* scroll region down */ - drawlist(row,num,detail,*list); - } - delta = 1; - break; - } - }else{ - switch(c) - { - case '\033': - extended=1; - break; - - case '[': /* cheat : always precedes cursor move */ - case 'O': /* ANSI application key mode */ - if (extended==1) - extended=2; - else - extended=0; - break; - - case 'Q': - case 'q': - return(KEY_EXIT); /* user requests exit */ - - case '\r': - case '\n': - return(KEY_DO); /* "do" something */ - - case '\b': - case '\177': - case 599: - return(KEY_DEL); /* "delete" response */ - - case 'X': - case 'x': - return(KEY_UNZOOM); /* expand everything */ - - case 'C': - case 'c': - return(KEY_ZOOM); /* collapse everything */ - - case '\t': - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp); /* unhighlight current line */ - return(KEY_TAB); /* "move" response */ - - case '\014': /* ^L, redraw */ - return(KEY_REDRAW); - - case '?': /* helptext */ - return(KEY_HELP); - - } - } - } -} - - -/** - ** visuserconfig - ** - ** Do the fullscreen config thang - **/ -static int -visuserconfig(void) -{ - int actofs = 0, inactofs = 0, mode = 0, ret = -1, i; - DEV_LIST *dp; - - initlist(&active); - initlist(&inactive); - alist = active; - ilist = inactive; - - getdevs(); - - conflicts = findconflict(active); /* find conflicts in the active list only */ - - redraw(); - - for(;;) - { - switch(mode) - { - case 0: /* active devices */ - ret = dolist(1,8,1,&actofs,&alist, - " [!bEnter!n] Edit device parameters [!bDEL!n] Disable device"); - switch(ret) - { - case KEY_TAB: - mode = 1; /* swap lists */ - break; - - case KEY_REDRAW: - redraw(); - break; - - case KEY_ZOOM: - alist = active; - actofs = 0; - expandlist(active); - redrawactive(); - break; - - case KEY_UNZOOM: - alist = active; - actofs = 0; - collapselist(active); - redrawactive(); - break; - - case KEY_DEL: - dp = ofsent(actofs,alist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->attrib & FLG_MANDATORY) /* can't be deleted */ - break; - if (dp == alist) /* moving top item on list? */ - { - if (dp->next) - { - alist = dp->next; /* point list to non-moving item */ - }else{ - alist = dp->prev; /* end of list, go back instead */ - } - }else{ - if (!dp->next) /* moving last item on list? */ - actofs--; - } - dp->conflicts = 0; /* no conflicts on the inactive list */ - movedev(dp,inactive); /* shift to inactive list */ - conflicts = findconflict(active); /* update conflict tags */ - dp->changed = 1; - redrawactive(); /* redraw */ - redrawinactive(); - } - break; - - case KEY_DO: /* edit device parameters */ - dp = ofsent(actofs,alist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->comment == DEV_DEVICE) /* can't edit comments, zoom? */ - { - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save device parameters"); - editparams(dp); - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save and Exit [!b?!n] Help"); - putxy(0,17,lines); - conflicts = findconflict(active); /* update conflict tags */ - }else{ /* DO on comment = zoom */ - switch(dp->comment) /* Depends on current state */ - { - case DEV_COMMENT: /* not currently zoomed */ - dp->comment = DEV_ZOOMED; - break; - - case DEV_ZOOMED: - dp->comment = DEV_COMMENT; - break; - } - } - redrawactive(); - } - break; - } - break; - - case 1: /* inactive devices */ - ret = dolist(10,7,0,&inactofs,&ilist, - " [!bEnter!n] Enable device "); - switch(ret) - { - case KEY_TAB: - mode = 0; - break; - - case KEY_REDRAW: - redraw(); - break; - - case KEY_ZOOM: - ilist = inactive; - inactofs = 0; - expandlist(inactive); - redrawinactive(); - break; - - case KEY_UNZOOM: - ilist = inactive; - inactofs = 0; - collapselist(inactive); - redrawinactive(); - break; - - case KEY_DO: - dp = ofsent(inactofs,ilist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->comment == DEV_DEVICE) /* can't move comments, zoom? */ - { - if (dp == ilist) /* moving top of list? */ - { - if (dp->next) - { - ilist = dp->next; /* point list to non-moving item */ - }else{ - ilist = dp->prev; /* can't go down, go up instead */ - } - }else{ - if (!dp->next) /* last entry on list? */ - inactofs--; /* shift cursor up one */ - } - - movedev(dp,active); /* shift to active list */ - conflicts = findconflict(active); /* update conflict tags */ - dp->changed = 1; - alist = dp; /* put at top and current */ - actofs = 0; - while(dp->comment == DEV_DEVICE) - dp = dp->prev; /* forcibly unzoom section */ - dp ->comment = DEV_COMMENT; - mode = 0; /* and swap modes to follow it */ - - }else{ /* DO on comment = zoom */ - switch(dp->comment) /* Depends on current state */ - { - case DEV_COMMENT: /* not currently zoomed */ - dp->comment = DEV_ZOOMED; - break; - - case DEV_ZOOMED: - dp->comment = DEV_COMMENT; - break; - } - } - redrawactive(); /* redraw */ - redrawinactive(); - } - break; - - default: /* nothing else relevant here */ - break; - } - break; - default: - mode = 0; /* shouldn't happen... */ - } - - /* handle returns that are the same for both modes */ - switch (ret) { - case KEY_HELP: - helpscreen(); - break; - - case KEY_EXIT: - i = yesnocancel(" Save these parameters before exiting? ([!bY!n]es/[!bN!n]o/[!bC!n]ancel) "); - switch(i) - { - case 2: /* cancel */ - redraw(); - break; - - case 1: /* save and exit */ - savelist(active,1); - savelist(inactive,0); - - case 0: /* exit */ - nukelist(active); /* clean up after ourselves */ - nukelist(inactive); - normal(); - clear(); - return(1); - } - break; - } - } -} -#endif /* VISUAL_USERCONFIG */ - -/* - * Copyright (c) 1991 Regents of the University of California. - * All rights reserved. - * Copyright (c) 1994 Jordan K. Hubbard - * All rights reserved. - * Copyright (c) 1994 David Greenman - * All rights reserved. - * - * Many additional changes by Bruce Evans - * - * This code is derived from software contributed by the - * University of California Berkeley, Jordan K. Hubbard, - * David Greenman and Bruce Evans. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: - * 1. Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * 3. All advertising materials mentioning features or use of this software - * must display the following acknowledgement: - * This product includes software developed by the University of - * California, Berkeley and its contributors. - * 4. Neither the name of the University nor the names of its contributors - * may be used to endorse or promote products derived from this software - * without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND - * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS - * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT - * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY - * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF - * SUCH DAMAGE. - * - * $FreeBSD$ - */ - -#define PARM_DEVSPEC 0x1 -#define PARM_INT 0x2 -#define PARM_ADDR 0x3 -#define PARM_STRING 0x4 - -typedef struct _cmdparm { - int type; - union { - struct uc_device *dparm; - int iparm; - union { - void *aparm; - const char *sparm; - } u; - } parm; -} CmdParm; - -typedef int (*CmdFunc)(CmdParm *); - -typedef struct _cmd { - char *name; - CmdFunc handler; - CmdParm *parms; -} Cmd; - - -static int lsdevtab(struct uc_device *); -static struct uc_device *find_device(char *, int); -static struct uc_device *search_devtable(struct uc_device *, char *, int); -static void cngets(char *, int); -static Cmd *parse_cmd(char *); -static int parse_args(const char *, CmdParm *); -static int save_dev(struct uc_device *); - -static int list_devices(CmdParm *); -static int set_device_ioaddr(CmdParm *); -static int set_device_irq(CmdParm *); -static int set_device_drq(CmdParm *); -static int set_device_iosize(CmdParm *); -static int set_device_mem(CmdParm *); -static int set_device_flags(CmdParm *); -static int set_device_enable(CmdParm *); -static int set_device_disable(CmdParm *); -static int quitfunc(CmdParm *); -static int helpfunc(CmdParm *); -#if defined(INTRO_USERCONFIG) -static int introfunc(CmdParm *); -#endif - -static int lineno; - -#ifdef DEV_EISA - -#include <dev/eisa/eisaconf.h> - -static int set_num_eisa_slots(CmdParm *); - -#endif - -static CmdParm addr_parms[] = { - { PARM_DEVSPEC, {} }, - { PARM_ADDR, {} }, - { -1, {} }, -}; - -static CmdParm int_parms[] = { - { PARM_DEVSPEC, {} }, - { PARM_INT, {} }, - { -1, {} }, -}; - -static CmdParm dev_parms[] = { - { PARM_DEVSPEC, {} }, - { -1, {} }, -}; - -#ifdef DEV_EISA -static CmdParm int_arg[] = { - { PARM_INT, {} }, - { -1, {} }, -}; -#endif - -static Cmd CmdList[] = { - { "?", helpfunc, NULL }, /* ? (help) */ - { "di", set_device_disable, dev_parms }, /* disable dev */ - { "dr", set_device_drq, int_parms }, /* drq dev # */ -#ifdef DEV_EISA - { "ei", set_num_eisa_slots, int_arg }, /* # EISA slots */ -#endif - { "en", set_device_enable, dev_parms }, /* enable dev */ - { "ex", quitfunc, NULL }, /* exit (quit) */ - { "f", set_device_flags, int_parms }, /* flags dev mask */ - { "h", helpfunc, NULL }, /* help */ -#if defined(INTRO_USERCONFIG) - { "intro", introfunc, NULL }, /* intro screen */ -#endif - { "iom", set_device_mem, addr_parms }, /* iomem dev addr */ - { "ios", set_device_iosize, int_parms }, /* iosize dev size */ - { "ir", set_device_irq, int_parms }, /* irq dev # */ - { "l", list_devices, NULL }, /* ls, list */ - { "po", set_device_ioaddr, int_parms }, /* port dev addr */ - { "res", (CmdFunc)cpu_reset, NULL }, /* reset CPU */ - { "q", quitfunc, NULL }, /* quit */ -#ifdef VISUAL_USERCONFIG - { "v", (CmdFunc)visuserconfig, NULL }, /* visual mode */ -#endif - { NULL, NULL, NULL }, -}; - -void -userconfig(void) -{ - static char banner = 1; - char input[80]; - int rval; - Cmd *cmd; - - load_devtab(); - init_config_script(); - while (1) { - - /* Only display signon banner if we are about to go interactive */ - if (!has_config_script()) { - if (!(boothowto & RB_CONFIG)) -#ifdef INTRO_USERCONFIG - banner = 0; -#else - return; -#endif - if (banner) { - banner = 0; - printf("FreeBSD Kernel Configuration Utility - Version 1.2\n" - " Type \"help\" for help" -#ifdef VISUAL_USERCONFIG - " or \"visual\" to go to the visual\n" - " configuration interface (requires MGA/VGA display or\n" - " serial terminal capable of displaying ANSI graphics)" -#endif - ".\n"); - } - } - - printf("config> "); - cngets(input, 80); - if (input[0] == '\0') - continue; - cmd = parse_cmd(input); - if (!cmd) { - printf("Invalid command or syntax. Type `?' for help.\n"); - continue; - } - rval = (*cmd->handler)(cmd->parms); - if (rval) { - free_devtab(); - return; - } - } -} - -static Cmd * -parse_cmd(char *cmd) -{ - Cmd *cp; - - for (cp = CmdList; cp->name; cp++) { - int len = strlen(cp->name); - - if (!strncmp(cp->name, cmd, len)) { - while (*cmd && *cmd != ' ' && *cmd != '\t') - ++cmd; - if (parse_args(cmd, cp->parms)) - return NULL; - else - return cp; - } - } - return NULL; -} - -static int -parse_args(const char *cmd, CmdParm *parms) -{ - while (1) { - char *ptr; - - if (*cmd == ' ' || *cmd == '\t') { - ++cmd; - continue; - } - if (parms == NULL || parms->type == -1) { - if (*cmd == '\0') - return 0; - printf("Extra arg(s): %s\n", cmd); - return 1; - } - if (parms->type == PARM_DEVSPEC) { - int i = 0; - char devname[64]; - int unit = 0; - - while (*cmd && !(*cmd == ' ' || *cmd == '\t' || - (*cmd >= '0' && *cmd <= '9'))) - devname[i++] = *(cmd++); - devname[i] = '\0'; - if (*cmd >= '0' && *cmd <= '9') { - unit = strtoul(cmd, &ptr, 10); - if (cmd == ptr) { - printf("Invalid device number\n"); - /* XXX should print invalid token here and elsewhere. */ - return 1; - } - /* XXX else should require end of token. */ - cmd = ptr; - } - if ((parms->parm.dparm = find_device(devname, unit)) == NULL) { - printf("No such device: %s%d\n", devname, unit); - return 1; - } - ++parms; - continue; - } - if (parms->type == PARM_INT) { - parms->parm.iparm = strtoul(cmd, &ptr, 0); - if (cmd == ptr) { - printf("Invalid numeric argument\n"); - return 1; - } - cmd = ptr; - ++parms; - continue; - } - if (parms->type == PARM_ADDR) { - parms->parm.u.aparm = (void *)(uintptr_t)strtoul(cmd, &ptr, 0); - if (cmd == ptr) { - printf("Invalid address argument\n"); - return 1; - } - cmd = ptr; - ++parms; - continue; - } - if (parms->type == PARM_STRING) { - parms->parm.u.sparm = cmd; - return 0; - } - } - return 0; -} - -static int -list_devices(CmdParm *parms) -{ - lineno = 0; - if (lsdevtab(uc_devtab)) return 0; -#ifdef DEV_EISA - printf("\nNumber of EISA slots to probe: %d\n", num_eisa_slots); -#endif - return 0; -} - -static int -set_device_ioaddr(CmdParm *parms) -{ - parms[0].parm.dparm->id_iobase = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_irq(CmdParm *parms) -{ - unsigned irq; - - irq = parms[1].parm.iparm; - if (irq == 2) { - printf("Warning: Remapping IRQ 2 to IRQ 9\n"); - irq = 9; - } - else if (irq != -1 && irq > 15) { - printf("An IRQ > 15 would be invalid.\n"); - return 0; - } - parms[0].parm.dparm->id_irq = (irq < 16 ? 1 << irq : 0); - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_drq(CmdParm *parms) -{ - unsigned drq; - - /* - * The bounds checking is just to ensure that the value can be printed - * in 5 characters. 32768 gets converted to -32768 and doesn't fit. - */ - drq = parms[1].parm.iparm; - parms[0].parm.dparm->id_drq = (drq < 32768 ? drq : -1); - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_iosize(CmdParm *parms) -{ - parms[0].parm.dparm->id_msize = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_mem(CmdParm *parms) -{ - parms[0].parm.dparm->id_maddr = parms[1].parm.u.aparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_flags(CmdParm *parms) -{ - parms[0].parm.dparm->id_flags = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_enable(CmdParm *parms) -{ - parms[0].parm.dparm->id_enabled = TRUE; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_disable(CmdParm *parms) -{ - parms[0].parm.dparm->id_enabled = FALSE; - save_dev(parms[0].parm.dparm); - return 0; -} - -#ifdef DEV_EISA -static int -set_num_eisa_slots(CmdParm *parms) -{ - int num_slots; - - num_slots = parms[0].parm.iparm; - num_eisa_slots = (num_slots <= 16 ? num_slots : 10); - return 0; -} -#endif - -static int -quitfunc(CmdParm *parms) -{ - /* - * If kernel config supplied, and we are parsing it, and -c also supplied, - * ignore a quit command, This provides a safety mechanism to allow - * recovery from a damaged/buggy kernel config. - */ - if ((boothowto & RB_CONFIG) && userconfig_boot_parsing) - return 0; - return 1; -} - -static int -helpfunc(CmdParm *parms) -{ - printf( - "Command\t\t\tDescription\n" - "-------\t\t\t-----------\n" - "ls\t\t\tList currently configured devices\n" - "port <devname> <addr>\tSet device port (i/o address)\n" - "irq <devname> <number>\tSet device irq\n" - "drq <devname> <number>\tSet device drq\n" - "iomem <devname> <addr>\tSet device maddr (memory address)\n" - "iosize <devname> <size>\tSet device memory size\n" - "flags <devname> <mask>\tSet device flags\n" - "enable <devname>\tEnable device\n" - "disable <devname>\tDisable device (will not be probed)\n"); -#ifdef DEV_EISA - printf("eisa <number>\t\tSet the number of EISA slots to probe\n"); -#endif - printf( - "quit\t\t\tExit this configuration utility\n" - "reset\t\t\tReset CPU\n"); -#ifdef VISUAL_USERCONFIG - printf("visual\t\t\tGo to fullscreen mode.\n"); -#endif - printf( - "help\t\t\tThis message\n\n" - "Commands may be abbreviated to a unique prefix\n"); - return 0; -} - -#if defined(INTRO_USERCONFIG) - -#if defined (VISUAL_USERCONFIG) -static void -center(int y, char *str) -{ - putxy((80 - strlen(str)) / 2, y, str); -} -#endif - -static int -introfunc(CmdParm *parms) -{ -#if defined (VISUAL_USERCONFIG) - int curr_item, first_time, extended = 0; - static char *choices[] = { - " Skip kernel configuration and continue with installation ", - " Start kernel configuration in full-screen visual mode ", - " Start kernel configuration in CLI mode ", - }; - - clear(); - center(2, "!bKernel Configuration Menu!n"); - - curr_item = 0; - first_time = 1; - while (1) { - char tmp[80]; - int c, i; - - if (!extended) { - for (i = 0; i < 3; i++) { - tmp[0] = '\0'; - if (curr_item == i) - strcpy(tmp, "!i"); - strcat(tmp, choices[i]); - if (curr_item == i) - strcat(tmp, "!n"); - putxy(10, 5 + i, tmp); - } - - if (first_time) { - putxy(2, 10, "Here you have the chance to go into kernel configuration mode, making"); - putxy(2, 11, "any changes which may be necessary to properly adjust the kernel to"); - putxy(2, 12, "match your hardware configuration."); - putxy(2, 14, "If you are installing FreeBSD for the first time, select Visual Mode"); - putxy(2, 15, "(press Down-Arrow then ENTER)."); - putxy(2, 17, "If you need to do more specialized kernel configuration and are an"); - putxy(2, 18, "experienced FreeBSD user, select CLI mode."); - putxy(2, 20, "If you are !icertain!n that you do not need to configure your kernel"); - putxy(2, 21, "then simply press ENTER or Q now."); - first_time = 0; - } - - move(0, 0); /* move the cursor out of the way */ - } - c = getchar(); - if ((extended == 2) || (c == 588) || (c == 596)) { /* console gives "alternative" codes */ - extended = 0; /* no longer */ - switch (c) { - case 588: - case 'A': /* up */ - if (curr_item > 0) - --curr_item; - break; - - case 596: - case 'B': /* down */ - if (curr_item < 2) - ++curr_item; - break; - } - } - else { - switch(c) { - case '\033': - extended = 1; - break; - - case '[': /* cheat : always precedes cursor move */ - case 'O': /* ANSI application key mode */ - if (extended == 1) - extended = 2; - else - extended = 0; - break; - - case -1: - case 'Q': - case 'q': - clear(); - return 1; /* user requests exit */ - - case '1': /* select an item */ - case 'S': - case 's': - curr_item = 0; - break; - case '2': - case 'V': - case 'v': - curr_item = 1; - break; - case '3': - case 'C': - case 'c': - curr_item = 2; - break; - - case 'U': /* up */ - case 'u': - case 'P': - case 'p': - if (curr_item > 0) - --curr_item; - break; - - case 'D': /* down */ - case 'd': - case 'N': - case 'n': - if (curr_item < 2) - ++curr_item; - break; - - case '\r': - case '\n': - clear(); - if (!curr_item) - return 1; - else if (curr_item == 1) - return visuserconfig(); - else { - putxy(0, 1, "Type \"help\" for help or \"quit\" to exit."); - /* enable quitfunc */ - userconfig_boot_parsing=0; - move (0, 3); - boothowto |= RB_CONFIG; /* force -c */ - return 0; - } - break; - } - } - } -#endif -} -#endif - -static int -lsdevtab(struct uc_device *dt) -{ - for (; dt->id_id != 0; dt++) { - char dname[80]; - - if (lineno >= 23) { - printf("<More> "); - if (!userconfig_boot_parsing) { - if (getchar() == 'q') { - printf("quit\n"); - return (1); - } - printf("\n"); - } - lineno = 0; - } - if (lineno == 0) { - printf( -"Device port irq drq iomem iosize unit flags enab\n" - ); - ++lineno; - } - sprintf(dname, "%s%d", dt->id_name, dt->id_unit); - printf("%-9.9s%-#11x%-6d%-6d%-8p%-9d%-6d%-#11x%-5s\n", - dname, /* dt->id_id, dt->id_driver(by name), */ dt->id_iobase, - ffs(dt->id_irq) - 1, dt->id_drq, dt->id_maddr, dt->id_msize, - /* dt->id_intr(by name), */ dt->id_unit, dt->id_flags, - dt->id_enabled ? "Yes" : "No"); - ++lineno; - } - return(0); -} - -static void -load_devtab(void) -{ - int i, val; - int count = resource_count(); - int id = 1; - int dt; - char *name; - int unit; - - uc_devtab = malloc(sizeof(struct uc_device) * (count + 1), M_DEVL, - M_WAITOK | M_ZERO); - dt = 0; - for (i = 0; i < count; i++) { - name = resource_query_name(i); - unit = resource_query_unit(i); - if (unit < 0) - continue; /* skip wildcards */ - uc_devtab[dt].id_id = id++; - resource_int_value(name, unit, "port", &uc_devtab[dt].id_iobase); - val = 0; - resource_int_value(name, unit, "irq", &val); - uc_devtab[dt].id_irq = (1 << val); - resource_int_value(name, unit, "drq", &uc_devtab[dt].id_drq); - resource_int_value(name, unit, "maddr",(int *)&uc_devtab[dt].id_maddr); - resource_int_value(name, unit, "msize", &uc_devtab[dt].id_msize); - uc_devtab[dt].id_unit = unit; - resource_int_value(name, unit, "flags", &uc_devtab[dt].id_flags); - val = 0; - resource_int_value(name, unit, "disabled", &val); - uc_devtab[dt].id_enabled = !val; - uc_devtab[dt].id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(uc_devtab[dt].id_name, name); - dt++; - } -} - -static void -free_devtab(void) -{ - int i; - int count = resource_count(); - - for (i = 0; i < count; i++) - if (uc_devtab[i].id_name) - free(uc_devtab[i].id_name, M_DEVL); - free(uc_devtab, M_DEVL); -} - -static struct uc_device * -find_device(char *devname, int unit) -{ - struct uc_device *ret; - - if ((ret = search_devtable(uc_devtab, devname, unit)) != NULL) - return ret; - return NULL; -} - -static struct uc_device * -search_devtable(struct uc_device *dt, char *devname, int unit) -{ - int i; - - for (i = 0; dt->id_id != 0; dt++) - if (!strcmp(dt->id_name, devname) && dt->id_unit == unit) - return dt; - return NULL; -} - -static void -cngets(char *input, int maxin) -{ - int c, nchars = 0; - - while (1) { - c = getchar(); - /* Treat ^H or ^? as backspace */ - if ((c == '\010' || c == '\177')) { - if (nchars) { - printf("\010 \010"); - *--input = '\0', --nchars; - } - continue; - } - /* Treat ^U or ^X as kill line */ - else if ((c == '\025' || c == '\030')) { - while (nchars) { - printf("\010 \010"); - *--input = '\0', --nchars; - } - continue; - } - printf("%c", c); - if ((++nchars == maxin) || (c == '\n') || (c == '\r') || ( c == -1)) { - *input = '\0'; - break; - } - *input++ = (u_char)c; - } -} - -static void -save_resource(struct uc_device *idev) -{ - char *name; - int unit; - - name = idev->id_name; - unit = idev->id_unit; - resource_set_int(name, unit, "port", idev->id_iobase); - resource_set_int(name, unit, "irq", ffs(idev->id_irq) - 1); - resource_set_int(name, unit, "drq", idev->id_drq); - resource_set_int(name, unit, "maddr", (int)idev->id_maddr); - resource_set_int(name, unit, "msize", idev->id_msize); - resource_set_int(name, unit, "flags", idev->id_flags); - resource_set_int(name, unit, "disabled", !idev->id_enabled); -} - -static int -save_dev(idev) -struct uc_device *idev; -{ - struct uc_device *id_p,*id_pn; - char *name = idev->id_name; - - for (id_p = uc_devlist; id_p; id_p = id_p->id_next) { - if (id_p->id_id == idev->id_id) { - id_pn = id_p->id_next; - if (id_p->id_name) - free(id_p->id_name, M_DEVL); - bcopy(idev,id_p,sizeof(struct uc_device)); - save_resource(idev); - id_p->id_name = malloc(strlen(name)+1, M_DEVL,M_WAITOK); - strcpy(id_p->id_name, name); - id_p->id_next = id_pn; - return 1; - } - } - id_pn = malloc(sizeof(struct uc_device),M_DEVL,M_WAITOK); - bcopy(idev,id_pn,sizeof(struct uc_device)); - save_resource(idev); - id_pn->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_pn->id_name, name); - id_pn->id_next = uc_devlist; - uc_devlist = id_pn; - return 0; -} - - diff --git a/sys/pc98/i386/userconfig.c b/sys/pc98/i386/userconfig.c deleted file mode 100644 index 1a8821f..0000000 --- a/sys/pc98/i386/userconfig.c +++ /dev/null @@ -1,3178 +0,0 @@ -/** - ** Copyright (c) 1995 - ** Michael Smith, msmith@freebsd.org. All rights reserved. - ** - ** This code contains a module marked : - - * Copyright (c) 1991 Regents of the University of California. - * All rights reserved. - * Copyright (c) 1994 Jordan K. Hubbard - * All rights reserved. - * Copyright (c) 1994 David Greenman - * All rights reserved. - * - * Many additional changes by Bruce Evans - * - * This code is derived from software contributed by the - * University of California Berkeley, Jordan K. Hubbard, - * David Greenman and Bruce Evans. - - ** As such, it contains code subject to the above copyrights. - ** The module and its copyright can be found below. - ** - ** Redistribution and use in source and binary forms, with or without - ** modification, are permitted provided that the following conditions - ** are met: - ** 1. Redistributions of source code must retain the above copyright - ** notice, this list of conditions and the following disclaimer as - ** the first lines of this file unmodified. - ** 2. Redistributions in binary form must reproduce the above copyright - ** notice, this list of conditions and the following disclaimer in the - ** documentation and/or other materials provided with the distribution. - ** 3. All advertising materials mentioning features or use of this software - ** must display the following acknowledgment: - ** This product includes software developed by Michael Smith. - ** 4. The name of the author may not be used to endorse or promote products - ** derived from this software without specific prior written permission. - ** - ** THIS SOFTWARE IS PROVIDED BY MICHAEL SMITH ``AS IS'' AND ANY EXPRESS OR - ** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES - ** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. - ** IN NO EVENT SHALL MICHAEL SMITH BE LIABLE FOR ANY DIRECT, INDIRECT, - ** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT - ** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, - ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY - ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF - ** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - ** - ** $FreeBSD$ - **/ - -/** - ** USERCONFIG - ** - ** Kernel boot-time configuration manipulation tool for FreeBSD. - ** - ** Two modes of operation are supported : the default is the line-editor mode, - ** the command "visual" invokes the fullscreen mode. - ** - ** The line-editor mode is the old favorite from FreeBSD 2.0/20.05 &c., the - ** fullscreen mode requires syscons or a minimal-ansi serial console. - **/ - -/** - ** USERCONFIG, visual mode. - ** - ** msmith@freebsd.org - ** - ** Look for "EDIT THIS LIST" to add to the list of known devices - ** - ** - ** There are a number of assumptions made in this code. - ** - ** - That the console supports a minimal set of ANSI escape sequences - ** (See the screen manipulation section for a summary) - ** and has at least 24 rows. - ** - That values less than or equal to zero for any of the device - ** parameters indicate that the driver does not use the parameter. - ** - That flags are _always_ editable. - ** - ** Devices marked as disabled are imported as such. - ** - ** For this tool to be useful, the list of devices below _MUST_ be updated - ** when a new driver is brought into the kernel. It is not possible to - ** extract this information from the drivers in the kernel. - ** - ** XXX - TODO: - ** - ** - Display _what_ a device conflicts with. - ** - Implement page up/down (as what?) - ** - Wizard mode (no restrictions) - ** - Find out how to put syscons back into low-intensity mode so that the - ** !b escape is useful on the console. (It seems to be that it actually - ** gets low/high intensity backwards. That looks OK.) - ** - ** - Only display headings with devices under them. (difficult) - **/ - -/* - * PC-9801 port by KATO Takenori <kato@eclogite.eps.nagoya-u.ac.jp> - */ - -#include "opt_userconfig.h" -#define COMPAT_OLDISA /* get the definitions */ - -#include <sys/param.h> -#include <sys/systm.h> -#include <sys/kernel.h> -#include <sys/malloc.h> -#include <sys/reboot.h> -#include <sys/linker.h> -#include <sys/sysctl.h> -#include <sys/bus.h> -#include <sys/cons.h> - -#include <machine/md_var.h> -#include <machine/limits.h> - -#define _I386_ISA_ISA_DEVICE_H_ - -static MALLOC_DEFINE(M_DEVL, "uc_devlist", "uc_device lists in userconfig()"); - -#include <machine/uc_device.h> -static struct uc_device *uc_devlist; /* list read by kget to extract changes */ -static struct uc_device *uc_devtab; /* fake uc_device table */ - -static int userconfig_boot_parsing; /* set if we are reading from the boot instructions */ - -#define putchar(x) cnputc(x) - -static void load_devtab(void); -static void free_devtab(void); -static void save_resource(struct uc_device *); - -static int -sysctl_machdep_uc_devlist(SYSCTL_HANDLER_ARGS) -{ - struct uc_device *id; - int error=0; - char name[8]; - - if(!req->oldptr) { - /* Only sizing */ - id=uc_devlist; - while(id) { - error+=sizeof(struct uc_device)+8; - id=id->id_next; - } - return(SYSCTL_OUT(req,0,error)); - } else { - /* Output the data. The buffer is filled with consecutive - * struct uc_device and char buf[8], containing the name - * (not guaranteed to end with '\0'). - */ - id=uc_devlist; - while(id) { - error=sysctl_handle_opaque(oidp,id, - sizeof(struct uc_device),req); - if(error) return(error); - strncpy(name,id->id_name,8); - error=sysctl_handle_opaque(oidp,name, - 8,req); - if(error) return(error); - id=id->id_next; - } - return(0); - } -} - -SYSCTL_PROC( _machdep, OID_AUTO, uc_devlist, CTLFLAG_RD, - 0, 0, sysctl_machdep_uc_devlist, "A", - "List of ISA devices changed in UserConfig"); - -/* -** Obtain command input. -** -** Initially, input is read from a possibly-loaded script. -** At the end of the script, or if no script is supplied, -** behaviour is determined by the RB_CONFIG (-c) flag. If -** the flag is set, user input is read from the console; if -** unset, the 'quit' command is invoked and userconfig -** will exit. -** -** Note that quit commands encountered in the script will be -** ignored if the RB_CONFIG flag is supplied. -*/ -static const char *config_script; -static int config_script_size; /* use of int for -ve magic value */ - -#define has_config_script() (config_script_size > 0) - -static int -init_config_script(void) -{ - caddr_t autoentry, autoattr; - - /* Look for loaded userconfig script */ - autoentry = preload_search_by_type("userconfig_script"); - if (autoentry != NULL) { - /* We have one, get size and data */ - config_script_size = 0; - if ((autoattr = preload_search_info(autoentry, MODINFO_SIZE)) != NULL) - config_script_size = (size_t)*(u_int32_t *)autoattr; - config_script = NULL; - if ((autoattr = preload_search_info(autoentry, MODINFO_ADDR)) != NULL) - config_script = *(const char **)autoattr; - /* sanity check */ - if ((config_script_size == 0) || (config_script == NULL)) { - config_script_size = 0; - config_script = NULL; - } - } - return has_config_script(); -} - -static int -getchar(void) -{ - int c = -1; -#ifdef INTRO_USERCONFIG - static int intro = 0; -#endif - - if (has_config_script()) - { - /* Consume character from loaded userconfig script, display */ - userconfig_boot_parsing = 1; - c = *config_script; - config_script++; - config_script_size--; - - } else { - -#ifdef INTRO_USERCONFIG - if (userconfig_boot_parsing) { - if (!(boothowto & RB_CONFIG)) { - /* userconfig_script, !RB_CONFIG -> quit */ - if (intro == 0) { - c = 'q'; - config_script = "uit\n"; - config_script_size = strlen(config_script); - /* userconfig_script will be 1 on the next pass */ - } - } else { - /* userconfig_script, RB_CONFIG -> cngetc() */ - } - } else { - if (!(boothowto & RB_CONFIG)) { - /* no userconfig_script, !RB_CONFIG -> show intro */ - if (intro == 0) { - intro = 1; - c = 'i'; - config_script = "ntro\n"; - config_script_size = strlen(config_script); - /* userconfig_script will be 1 on the next pass */ - } - } else { - /* no userconfig_script, RB_CONFIG -> cngetc() */ - } - } -#else /* !INTRO_USERCONFIG */ - /* assert(boothowto & RB_CONFIG) */ -#endif /* INTRO_USERCONFIG */ - userconfig_boot_parsing = 0; - if (c <= 0) - c = cngetc(); - } - return(c); -} - -#ifndef FALSE -#define FALSE (0) -#define TRUE (!FALSE) -#endif - -#ifdef VISUAL_USERCONFIG - -typedef struct -{ - char dev[16]; /* device basename */ - char name[60]; /* long name */ - int attrib; /* things to do with the device */ - int class; /* device classification */ -} DEV_INFO; - -#define FLG_INVISIBLE (1<<0) /* device should not be shown */ -#define FLG_MANDATORY (1<<1) /* device can be edited but not disabled */ -#define FLG_FIXIRQ (1<<2) /* device IRQ cannot be changed */ -#define FLG_FIXIOBASE (1<<3) /* device iobase cannot be changed */ -#define FLG_FIXMADDR (1<<4) /* device maddr cannot be changed */ -#define FLG_FIXMSIZE (1<<5) /* device msize cannot be changed */ -#define FLG_FIXDRQ (1<<6) /* device DRQ cannot be changed */ -#define FLG_FIXED (FLG_FIXIRQ|FLG_FIXIOBASE|FLG_FIXMADDR|FLG_FIXMSIZE|FLG_FIXDRQ) -#define FLG_IMMUTABLE (FLG_FIXED|FLG_MANDATORY) - -#define CLS_STORAGE 1 /* storage devices */ -#define CLS_NETWORK 2 /* network interfaces */ -#define CLS_COMMS 3 /* serial, parallel ports */ -#define CLS_INPUT 4 /* user input : mice, keyboards, joysticks etc */ -#define CLS_MMEDIA 5 /* "multimedia" devices (sound, video, etc) */ -#define CLS_MISC 255 /* none of the above */ - - -typedef struct -{ - char name[60]; - int number; -} DEVCLASS_INFO; - -static DEVCLASS_INFO devclass_names[] = { -{ "Storage : ", CLS_STORAGE}, -{ "Network : ", CLS_NETWORK}, -{ "Communications : ", CLS_COMMS}, -{ "Input : ", CLS_INPUT}, -{ "Multimedia : ", CLS_MMEDIA}, -{ "Miscellaneous : ", CLS_MISC}, -{ "",0}}; - - -/********************* EDIT THIS LIST **********************/ - -/** Notes : - ** - ** - Devices that shouldn't be seen or removed should be marked FLG_INVISIBLE. - ** - XXX The list below should be reviewed by the driver authors to verify - ** that the correct flags have been set for each driver, and that the - ** descriptions are accurate. - **/ - -static DEV_INFO device_info[] = { -/*---Name----- ---Description---------------------------------------------- */ -#ifdef PC98 -{"bs", "PC-9801-55 SCSI Interface", 0, CLS_STORAGE}, -{"wdc", "IDE/ESDI/MFM disk controller", 0, CLS_STORAGE}, -#endif -{"adv", "AdvanSys SCSI narrow controller", 0, CLS_STORAGE}, -{"bt", "Buslogic SCSI controller", 0, CLS_STORAGE}, -{"aha", "Adaptec 154x SCSI controller", 0, CLS_STORAGE}, -{"aic", "Adaptec 152x SCSI and compatible SCSI cards", 0, CLS_STORAGE}, -{"nca", "ProAudio Spectrum SCSI and compatibles", 0, CLS_STORAGE}, -{"sea", "Seagate ST01/ST02 SCSI and compatibles", 0, CLS_STORAGE}, -{"stg", "TMC 18C30/18C50 based SCSI cards", 0, CLS_STORAGE}, -{"ata", "ATA/ATAPI compatible disk controller", 0, CLS_STORAGE}, -{"fdc", "Floppy disk controller", FLG_FIXED, CLS_STORAGE}, -{"mcd", "Mitsumi CD-ROM", 0, CLS_STORAGE}, -{"scd", "Sony CD-ROM", 0, CLS_STORAGE}, -{"matcd", "Matsushita/Panasonic/Creative CDROM", 0, CLS_STORAGE}, -{"wt", "Wangtek/Archive QIC-02 Tape drive", 0, CLS_STORAGE}, -{"wd", "IDE or ST506 compatible storage device", FLG_INVISIBLE, CLS_STORAGE}, -{"ad", "ATA/ATAPI compatible storage device", FLG_INVISIBLE, CLS_STORAGE}, -{"fd", "Floppy disk device", FLG_INVISIBLE, CLS_STORAGE}, - -{"cs", "IBM EtherJet, CS89x0-based Ethernet adapters",0, CLS_NETWORK}, -#ifdef PC98 -{"ed", "NS8390 Ethernet adapters", 0, CLS_NETWORK}, -#else -{"ed", "NE1000,NE2000,3C503,WD/SMC80xx Ethernet adapters",0, CLS_NETWORK}, -#endif -{"el", "3C501 Ethernet adapter", 0, CLS_NETWORK}, -{"ep", "3C509 Ethernet adapter", 0, CLS_NETWORK}, -{"ex", "Intel EtherExpress Pro/10 Ethernet adapter", 0, CLS_NETWORK}, -{"fe", "Fujitsu MB86960A/MB86965A Ethernet adapters", 0, CLS_NETWORK}, -{"ie", "AT&T Starlan 10 and EN100, 3C507, NI5210 Ethernet adapters",0,CLS_NETWORK}, -{"le", "DEC Etherworks 2 and 3 Ethernet adapters", 0, CLS_NETWORK}, -{"lnc", "Isolan, Novell NE2100/NE32-VL Ethernet adapters", 0,CLS_NETWORK}, -{"sn", "SMC/Megahertz Ethernet adapters", 0,CLS_NETWORK}, -{"snc", "SONIC Ethernet adapters", 0,CLS_NETWORK}, -{"xe", "Xircom PC Card Ethernet adapter", 0, CLS_NETWORK}, -{"rdp", "RealTek RTL8002 Pocket Ethernet", 0, CLS_NETWORK}, - -{"sio", "8250/16450/16550 Serial port", 0, CLS_COMMS}, -{"cx", "Cronyx/Sigma multiport sync/async adapter",0, CLS_COMMS}, -{"rc", "RISCom/8 multiport async adapter", 0, CLS_COMMS}, -{"cy", "Cyclades multiport async adapter", 0, CLS_COMMS}, -{"dgb", "Digiboard PC/Xe, PC/Xi async adapter", 0, CLS_COMMS}, -{"si", "Specialix SI/XIO/SX async adapter", 0, CLS_COMMS}, -{"stl", "Stallion EasyIO/Easy Connection 8/32 async adapter",0, CLS_COMMS}, -{"stli", "Stallion intelligent async adapter" ,0, CLS_COMMS}, -#ifdef PC98 -{"olpt", "Parallel printer port", 0, CLS_COMMS}, -#endif -{"ppc", "Parallel Port chipset", 0, CLS_COMMS}, -{"gp", "National Instruments AT-GPIB/TNT driver", 0, CLS_COMMS}, - -{"atkbdc", "Keyboard controller", FLG_INVISIBLE, CLS_INPUT}, -{"atkbd", "Keyboard", FLG_FIXED, CLS_INPUT}, -#ifdef PC98 -{"pckbd", "Keyboard", FLG_FIXED, CLS_INPUT}, -#endif -{"mse", "Microsoft Bus Mouse", 0, CLS_INPUT}, -{"psm", "PS/2 Mouse", FLG_FIXED, CLS_INPUT}, -{"joy", "Joystick", FLG_FIXED, CLS_INPUT}, -{"vt", "PCVT console driver", FLG_IMMUTABLE, CLS_INPUT}, -{"sc", "Syscons console driver", FLG_IMMUTABLE, CLS_INPUT}, - -#ifdef PC98 -{"nss", "PC-9801-86 Sound Board", 0, CLS_MMEDIA}, -#endif -{"sbc", "PCM Creative SoundBlaster/ESS/Avance sounce cards", 0,CLS_MMEDIA}, -{"gusc", "PCM Gravis UltraSound sound cards", 0, CLS_MMEDIA}, -{"pcm", "PCM Generic soundcard support", 0, CLS_MMEDIA}, -{"sb", "VOXWARE Soundblaster PCM (SB/Pro/16, ProAudio Spectrum)",0,CLS_MMEDIA}, -{"sbxvi", "VOXWARE Soundblaster 16", 0, CLS_MMEDIA}, -{"sbmidi", "VOXWARE Soundblaster MIDI interface", 0, CLS_MMEDIA}, -{"awe", "VOXWARE AWE32 MIDI", 0, CLS_MMEDIA}, -{"pas", "VOXWARE ProAudio Spectrum PCM and MIDI", 0, CLS_MMEDIA}, -{"gus", "VOXWARE Gravis Ultrasound, Ultrasound 16 and Ultrasound MAX",0,CLS_MMEDIA}, -{"gusxvi", "VOXWARE Gravis Ultrasound 16-bit PCM", 0, CLS_MMEDIA}, -{"gusmax", "VOXWARE Gravis Ultrasound MAX", 0, CLS_MMEDIA}, -{"mss", "VOXWARE Microsoft Sound System", 0, CLS_MMEDIA}, -{"opl", "VOXWARE OPL-2/3 FM, SB/Pro/16, ProAudio Spectrum",0,CLS_MMEDIA}, -{"mpu", "VOXWARE Roland MPU401 MIDI", 0, CLS_MMEDIA}, -{"sscape", "VOXWARE Ensoniq Soundscape MIDI interface", 0, CLS_MMEDIA}, -{"sscape_mss", "VOXWARE Ensoniq Soundscape PCM", 0, CLS_MMEDIA}, -{"uart", "VOXWARE 6850 MIDI UART", 0, CLS_MMEDIA}, -{"pca", "PC speaker PCM audio driver", FLG_FIXED, CLS_MMEDIA}, -{"ctx", "Coretex-I frame grabber", 0, CLS_MMEDIA}, -{"spigot", "Creative Labs Video Spigot video capture", 0, CLS_MMEDIA}, -{"scc", "IBM Smart Capture Card", 0, CLS_MMEDIA}, -{"gsc", "Genius GS-4500 hand scanner", 0, CLS_MMEDIA}, -{"asc", "AmiScan scanner", 0, CLS_MMEDIA}, - -{"apm", "Advanced Power Management", FLG_FIXED, CLS_MISC}, -{"pcic", "PC-card controller", 0, CLS_MISC}, -{"npx", "Math coprocessor", FLG_IMMUTABLE, CLS_MISC}, -#ifdef PC98 -{"gdc", "Graphic Display Controller", FLG_INVISIBLE, CLS_MISC}, -#endif -{"vga", "Catchall PCI VGA driver", FLG_INVISIBLE, CLS_MISC}, -{"","",0,0}}; - - -typedef struct _devlist_struct -{ - char name[80]; - int attrib; /* flag values as per the FLG_* defines above */ - int class; /* disk, etc as per the CLS_* defines above */ - char dev[16]; - int iobase,irq,drq,maddr,msize,unit,flags,id; - int comment; /* 0 = device, 1 = comment, 2 = collapsed comment */ - int conflicts; /* set/reset by findconflict, count of conflicts */ - int changed; /* nonzero if the device has been edited */ - struct uc_device *device; - struct _devlist_struct *prev,*next; -} DEV_LIST; - - -#define DEV_DEVICE 0 -#define DEV_COMMENT 1 -#define DEV_ZOOMED 2 - -#define LIST_CURRENT (1<<0) -#define LIST_SELECTED (1<<1) - -#define KEY_EXIT 0 /* return codes from dolist() and friends */ -#define KEY_DO 1 -#define KEY_DEL 2 -#define KEY_TAB 3 -#define KEY_REDRAW 4 - -#define KEY_UP 5 /* these only returned from editval() */ -#define KEY_DOWN 6 -#define KEY_LEFT 7 -#define KEY_RIGHT 8 -#define KEY_NULL 9 /* this allows us to spin & redraw */ - -#define KEY_ZOOM 10 /* these for zoom all/collapse all */ -#define KEY_UNZOOM 11 - -#define KEY_HELP 12 /* duh? */ - -static void redraw(void); -static void insdev(DEV_LIST *dev, DEV_LIST *list); -static int devinfo(DEV_LIST *dev); -static int visuserconfig(void); - -static DEV_LIST *active = NULL,*inactive = NULL; /* driver lists */ -static DEV_LIST *alist,*ilist; /* visible heads of the driver lists */ -static DEV_LIST scratch; /* scratch record */ -static int conflicts; /* total conflict count */ - - -static char lines[] = "--------------------------------------------------------------------------------"; -static char spaces[] = " "; - - -/** - ** Device manipulation stuff : find, describe, configure. - **/ - -/** - ** setdev - ** - ** Sets the device referenced by (*dev) to the parameters in the struct, - ** and the enable flag according to (enabled) - **/ -static void -setdev(DEV_LIST *dev, int enabled) -{ - dev->device->id_iobase = dev->iobase; /* copy happy */ - dev->device->id_irq = (u_short)(dev->irq < 16 ? 1<<dev->irq : 0); /* IRQ is bitfield */ - dev->device->id_drq = (short)dev->drq; - dev->device->id_maddr = (caddr_t)dev->maddr; - dev->device->id_msize = dev->msize; - dev->device->id_flags = dev->flags; - dev->device->id_enabled = enabled; -} - - -/** - ** getdevs - ** - ** Walk the kernel device tables and build the active and inactive lists - **/ -static void -getdevs(void) -{ - int i; - struct uc_device *ap; - - ap = uc_devtab; /* pointer to array of devices */ - for (i = 0; ap[i].id_id; i++) /* for each device in this table */ - { - scratch.unit = ap[i].id_unit; /* device parameters */ - strcpy(scratch.dev,ap[i].id_name); - scratch.iobase = ap[i].id_iobase; - scratch.irq = ffs(ap[i].id_irq)-1; - scratch.drq = ap[i].id_drq; - scratch.maddr = (int)ap[i].id_maddr; - scratch.msize = ap[i].id_msize; - scratch.flags = ap[i].id_flags; - - scratch.comment = DEV_DEVICE; /* admin stuff */ - scratch.conflicts = 0; - scratch.device = &ap[i]; /* save pointer for later reference */ - scratch.changed = 0; - if (!devinfo(&scratch)) /* get more info on the device */ - insdev(&scratch,ap[i].id_enabled?active:inactive); - } -} - - -/** - ** Devinfo - ** - ** Fill in (dev->name), (dev->attrib) and (dev->type) from the device_info array. - ** If the device is unknown, put it in the CLS_MISC class, with no flags. - ** - ** If the device is marked "invisible", return nonzero; the caller should - ** not insert any such device into either list. - ** - **/ -static int -devinfo(DEV_LIST *dev) -{ - int i; - - for (i = 0; device_info[i].class; i++) - { - if (!strcmp(dev->dev,device_info[i].dev)) - { - if (device_info[i].attrib & FLG_INVISIBLE) /* forget we ever saw this one */ - return(1); - strcpy(dev->name,device_info[i].name); /* get the name */ - dev->attrib = device_info[i].attrib; - dev->class = device_info[i].class; - return(0); - } - } - strcpy(dev->name,"Unknown device"); - dev->attrib = 0; - dev->class = CLS_MISC; - return(0); -} - - -/** - ** List manipulation stuff : add, move, initialise, free, traverse - ** - ** Note that there are assumptions throughout this code that - ** the first entry in a list will never move. (assumed to be - ** a comment). - **/ - - -/** - ** Adddev - ** - ** appends a copy of (dev) to the end of (*list) - **/ -static void -addev(DEV_LIST *dev, DEV_LIST **list) -{ - - DEV_LIST *lp,*ap; - - lp = (DEV_LIST *)malloc(sizeof(DEV_LIST),M_DEVL,M_WAITOK); - bcopy(dev,lp,sizeof(DEV_LIST)); /* create copied record */ - - if (*list) /* list exists */ - { - ap = *list; - while(ap->next) - ap = ap->next; /* scoot to end of list */ - lp->prev = ap; - lp->next = NULL; - ap->next = lp; - }else{ /* list does not yet exist */ - *list = lp; - lp->prev = lp->next = NULL; /* list now exists */ - } -} - - -/** - ** Findspot - ** - ** Finds the 'appropriate' place for (dev) in (list) - ** - ** 'Appropriate' means in numeric order with other devices of the same type, - ** or in alphabetic order following a comment of the appropriate type. - ** or at the end of the list if an appropriate comment is not found. (this should - ** never happen) - ** (Note that the appropriate point is never the top, but may be the bottom) - **/ -static DEV_LIST * -findspot(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *ap = NULL; - - /* search for a previous instance of the same device */ - for (ap = list; ap; ap = ap->next) - { - if (ap->comment != DEV_DEVICE) /* ignore comments */ - continue; - if (!strcmp(dev->dev,ap->dev)) /* same base device */ - { - if ((dev->unit <= ap->unit) /* belongs before (equal is bad) */ - || !ap->next) /* or end of list */ - { - ap = ap->prev; /* back up one */ - break; /* done here */ - } - if (ap->next) /* if the next item exists */ - { - if (ap->next->comment != DEV_DEVICE) /* next is a comment */ - break; - if (strcmp(dev->dev,ap->next->dev)) /* next is a different device */ - break; - } - } - } - - if (!ap) /* not sure yet */ - { - /* search for a class that the device might belong to */ - for (ap = list; ap; ap = ap->next) - { - if (ap->comment != DEV_DEVICE) /* look for simlar devices */ - continue; - if (dev->class != ap->class) /* of same class too 8) */ - continue; - if (strcmp(dev->dev,ap->dev) < 0) /* belongs before the current entry */ - { - ap = ap->prev; /* back up one */ - break; /* done here */ - } - if (ap->next) /* if the next item exists */ - if (ap->next->comment != DEV_DEVICE) /* next is a comment, go here */ - break; - } - } - - if (!ap) /* didn't find a match */ - { - for (ap = list; ap->next; ap = ap->next) /* try for a matching comment */ - if ((ap->comment != DEV_DEVICE) - && (ap->class == dev->class)) /* appropriate place? */ - break; - } /* or just put up with last */ - - return(ap); -} - - -/** - ** Insdev - ** - ** Inserts a copy of (dev) at the appropriate point in (list) - **/ -static void -insdev(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *lp,*ap; - - lp = (DEV_LIST *)malloc(sizeof(DEV_LIST),M_DEVL,M_WAITOK); - bcopy(dev,lp,sizeof(DEV_LIST)); /* create copied record */ - - ap = findspot(lp,list); /* find appropriate spot */ - lp->next = ap->next; /* point to next */ - if (ap->next) - ap->next->prev = lp; /* point next to new */ - lp->prev = ap; /* point new to current */ - ap->next = lp; /* and current to new */ -} - - -/** - ** Movedev - ** - ** Moves (dev) from its current list to an appropriate place in (list) - ** (dev) may not come from the top of a list, but it may from the bottom. - **/ -static void -movedev(DEV_LIST *dev, DEV_LIST *list) -{ - DEV_LIST *ap; - - ap = findspot(dev,list); - dev->prev->next = dev->next; /* remove from old list */ - if (dev->next) - dev->next->prev = dev->prev; - - dev->next = ap->next; /* insert in new list */ - if (ap->next) - ap->next->prev = dev; /* point next to new */ - dev->prev = ap; /* point new to current */ - ap->next = dev; /* and current to new */ -} - - -/** - ** Initlist - ** - ** Initialises (*list) with the basic headings - **/ -static void -initlist(DEV_LIST **list) -{ - int i; - - for(i = 0; devclass_names[i].name[0]; i++) /* for each devtype name */ - { - strcpy(scratch.name,devclass_names[i].name); - scratch.comment = DEV_ZOOMED; - scratch.class = devclass_names[i].number; - scratch.attrib = FLG_MANDATORY; /* can't be moved */ - addev(&scratch,list); /* add to the list */ - } -} - - -/** - ** savelist - ** - ** Walks (list) and saves the settings of any entry marked as changed. - ** - ** The device's active field is set according to (active). - ** - ** Builds the uc_devlist used by kget to extract the changed device information. - ** The code for this was taken almost verbatim from the original module. - **/ -static void -savelist(DEV_LIST *list, int active) -{ - struct uc_device *id_p,*id_pn; - char *name; - - while (list) - { - if ((list->comment == DEV_DEVICE) && /* is a device */ - (list->changed) && /* has been changed */ - (list->device != NULL)) { /* has an uc_device structure */ - - setdev(list,active); /* set the device itself */ - - id_pn = NULL; - for (id_p=uc_devlist; id_p; id_p=id_p->id_next) - { /* look on the list for it */ - if (id_p->id_id == list->device->id_id) - { - name = list->device->id_name; - id_pn = id_p->id_next; - if (id_p->id_name) - free(id_p->id_name, M_DEVL); - bcopy(list->device,id_p,sizeof(struct uc_device)); - save_resource(list->device); - id_p->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_p->id_name, name); - id_pn->id_next = uc_devlist; - id_p->id_next = id_pn; - break; - } - } - if (!id_pn) /* not already on the list */ - { - name = list->device->id_name; - id_pn = malloc(sizeof(struct uc_device),M_DEVL,M_WAITOK); - bcopy(list->device,id_pn,sizeof(struct uc_device)); - save_resource(list->device); - id_pn->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_pn->id_name, name); - id_pn->id_next = uc_devlist; - uc_devlist = id_pn; /* park at top of list */ - } - } - list = list->next; - } -} - - -/** - ** nukelist - ** - ** Frees all storage in use by a (list). - **/ -static void -nukelist(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return; - while(list->prev) /* walk to head of list */ - list = list->prev; - - while(list) - { - dp = list; - list = list->next; - free(dp,M_DEVL); - } -} - - -/** - ** prevent - ** - ** Returns the previous entry in (list), skipping zoomed regions. Returns NULL - ** if there is no previous entry. (Only possible if list->prev == NULL given the - ** premise that there is always a comment at the head of the list) - **/ -static DEV_LIST * -prevent(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return(NULL); - dp = list->prev; /* start back one */ - while(dp) - { - if (dp->comment == DEV_ZOOMED) /* previous section is zoomed */ - return(dp); /* so skip to comment */ - if (dp->comment == DEV_COMMENT) /* not zoomed */ - return(list->prev); /* one back as normal */ - dp = dp->prev; /* backpedal */ - } - return(dp); /* NULL, we can assume */ -} - - -/** - ** nextent - ** - ** Returns the next entry in (list), skipping zoomed regions. Returns NULL - ** if there is no next entry. (Possible if the current entry is last, or - ** if the current entry is the last heading and it's collapsed) - **/ -static DEV_LIST * -nextent(DEV_LIST *list) -{ - DEV_LIST *dp; - - if (!list) - return(NULL); - if (list->comment != DEV_ZOOMED) /* no reason to skip */ - return(list->next); - dp = list->next; - while(dp) - { - if (dp->comment != DEV_DEVICE) /* found another heading */ - break; - dp = dp->next; - } - return(dp); /* back we go */ -} - - -/** - ** ofsent - ** - ** Returns the (ofs)th entry down from (list), or NULL if it doesn't exist - **/ -static DEV_LIST * -ofsent(int ofs, DEV_LIST *list) -{ - while (ofs-- && list) - list = nextent(list); - return(list); -} - - -/** - ** findconflict - ** - ** Scans every element of (list) and sets the conflict tags appropriately - ** Returns the number of conflicts found. - **/ -static int -findconflict(DEV_LIST *list) -{ - int count = 0; /* number of conflicts found */ - DEV_LIST *dp,*sp; - - for (dp = list; dp; dp = dp->next) /* over the whole list */ - { - if (dp->comment != DEV_DEVICE) /* comments don't usually conflict */ - continue; - - dp->conflicts = 0; /* assume the best */ - for (sp = list; sp; sp = sp->next) /* scan the entire list for conflicts */ - { - if (sp->comment != DEV_DEVICE) /* likewise */ - continue; - - if (sp == dp) /* always conflict with itself */ - continue; - - if ((dp->iobase > 0) && /* iobase conflict? */ - (dp->iobase == sp->iobase)) - dp->conflicts = 1; - if ((dp->irq > 0) && /* irq conflict? */ - (dp->irq == sp->irq)) - dp->conflicts = 1; - if ((dp->drq > 0) && /* drq conflict? */ - (dp->drq == sp->drq)) - dp->conflicts = 1; - if ((sp->maddr > 0) && /* maddr/msize conflict? */ - (dp->maddr > 0) && - (sp->maddr + ((sp->msize == 0) ? 1 : sp->msize) > dp->maddr) && - (dp->maddr + ((dp->msize == 0) ? 1 : dp->msize) > sp->maddr)) - dp->conflicts = 1; - } - count += dp->conflicts; /* count conflicts */ - } - return(count); -} - - -/** - ** expandlist - ** - ** Unzooms all headings in (list) - **/ -static void -expandlist(DEV_LIST *list) -{ - while(list) - { - if (list->comment == DEV_COMMENT) - list->comment = DEV_ZOOMED; - list = list->next; - } -} - - -/** - ** collapselist - ** - ** Zooms all headings in (list) - **/ -static void -collapselist(DEV_LIST *list) -{ - while(list) - { - if (list->comment == DEV_ZOOMED) - list->comment = DEV_COMMENT; - list = list->next; - } -} - - -/** - ** Screen-manipulation stuff - ** - ** This is the basic screen layout : - ** - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 0 -|---Active Drivers----------------------------xx Conflicts------Dev---IRQ--Port--| - ** 1 -| ........................ ....... .. 0x....| - ** 2 -| ........................ ....... .. 0x....| - ** 3 -| ........................ ....... .. 0x....| - ** 4 -| ........................ ....... .. 0x....| - ** 5 -| ........................ ....... .. 0x....| - ** 6 -| ........................ ....... .. 0x....| - ** 7 -| ........................ ....... .. 0x....| - ** 8 -| ........................ ....... .. 0x....| - ** 9 -|---Inactive Drivers--------------------------------------------Dev--------------| - ** 10-| ........................ ....... | - ** 11-| ........................ ....... | - ** 12-| ........................ ....... | - ** 13-| ........................ ....... | - ** 14-| ........................ ....... | - ** 15-| ........................ ....... | - ** 16-| ........................ ....... | - ** 17-|------------------------------------------------------UP-DOWN-------------------| - ** 18-| Relevant parameters for the current device | - ** 19-| | - ** 20-| | - ** 21-|--------------------------------------------------------------------------------| - ** 22-| Help texts go here | - ** 23-| | - ** +--------------------------------------------------------------------------------+ - ** - ** Help texts - ** - ** On a collapsed comment : - ** - ** [Enter] Expand device list [z] Expand all lists - ** [TAB] Change fields [Q] Save and Exit - ** - ** On an expanded comment : - ** - ** [Enter] Collapse device list [Z] Collapse all lists - ** [TAB] Change fields [Q] Save and Exit - ** - ** On a comment with no followers - ** - ** - ** [TAB] Change fields [Q] Save and Exit - ** - ** On a device in the active list - ** - ** [Enter] Edit device parameters [DEL] Disable device - ** [TAB] Change fields [Q] Save and Exit [?] Help - ** - ** On a device in the inactive list - ** - ** [Enter] Enable device - ** [TAB] Change fields [Q] Save and Exit [?] Help - ** - ** While editing parameters - ** - ** <parameter-specific help here> - ** [TAB] Change fields [Q] Save device parameters - **/ - - - -/** - ** - ** The base-level screen primitives : - ** - ** bold() - enter bold mode \E[1m (md) - ** inverse() - enter inverse mode \E[7m (so) - ** normal() - clear bold/inverse mode \E[m (se) - ** clear() - clear the screen \E[H\E[J (ce) - ** move(x,y) - move the cursor to x,y \E[y;xH: (cm) - **/ - -static void -bold(void) -{ - printf("\033[1m"); -} - -static void -inverse(void) -{ - printf("\033[7m"); -} - -static void -normal(void) -{ - printf("\033[m"); -} - -static void -clear(void) -{ - normal(); - printf("\033[H\033[J"); -} - -static void -move(int x, int y) -{ - printf("\033[%d;%dH",y+1,x+1); -} - - -/** - ** - ** High-level screen primitives : - ** - ** putxyl(x,y,str,len) - put (len) bytes of (str) at (x,y), supports embedded formatting - ** putxy(x,y,str) - put (str) at (x,y), supports embedded formatting - ** erase(x,y,w,h) - clear the box (x,y,w,h) - ** txtbox(x,y,w,y,str) - put (str) in a region at (x,y,w,h) - ** putmsg(str) - put (str) in the message area - ** puthelp(str) - put (str) in the upper helpline - ** pad(str,len) - pad (str) to (len) with spaces - ** drawline(row,detail,list,inverse,*dhelp) - ** - draws a line for (*list) at (row) onscreen. If (detail) is - ** nonzero, include port, IRQ and maddr, if (inverse) is nonzero, - ** draw the line in inverse video, and display (*dhelp) on the - ** helpline. - ** drawlist(row,num,detail,list) - ** - draw (num) entries from (list) at (row) onscreen, passile (detail) - ** through to drawline(). - ** showparams(dev) - displays the relevant parameters for (dev) below the lists onscreen. - ** yesno(str) - displays (str) in the message area, and returns nonzero on 'y' or 'Y' - ** redraw(); - Redraws the entire screen layout, including the - ** - two list panels. - **/ - -/** - ** putxy - ** writes (str) at x,y onscreen - ** putxyl - ** writes up to (len) of (str) at x,y onscreen. - ** - ** Supports embedded formatting : - ** !i - inverse mode. - ** !b - bold mode. - ** !n - normal mode. - **/ -static void -putxyl(int x, int y, char *str, int len) -{ - move(x,y); - normal(); - - while((*str) && (len--)) - { - if (*str == '!') /* format escape? */ - { - switch(*(str+1)) /* depending on the next character */ - { - case 'i': - inverse(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - case 'b': - bold(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - case 'n': - normal(); - str +=2; /* skip formatting */ - len++; /* doesn't count for length */ - break; - - default: - putchar(*str++); /* not an escape */ - } - }else{ - putchar(*str++); /* emit the character */ - } - } -} - -#define putxy(x,y,str) putxyl(x,y,str,-1) - - -/** - ** erase - ** - ** Erases the region (x,y,w,h) - **/ -static void -erase(int x, int y, int w, int h) -{ - int i; - - normal(); - for (i = 0; i < h; i++) - putxyl(x,y++,spaces,w); -} - - -/** - ** txtbox - ** - ** Writes (str) into the region (x,y,w,h), supports embedded formatting using - ** putxy. Lines are not wrapped, newlines must be forced with \n. - **/ -static void -txtbox(int x, int y, int w, int h, char *str) -{ - int i = 0; - - h--; - while((str[i]) && h) - { - if (str[i] == '\n') /* newline */ - { - putxyl(x,y,str,(i<w)?i:w); /* write lesser of i or w */ - y++; /* move down */ - h--; /* room for one less */ - str += (i+1); /* skip first newline */ - i = 0; /* zero offset */ - }else{ - i++; /* next character */ - } - } - if (h) /* end of string, not region */ - putxyl(x,y,str,w); -} - - -/** - ** putmsg - ** - ** writes (msg) in the helptext area - **/ -static void -putmsg(char *msg) -{ - erase(0,18,80,3); /* clear area */ - txtbox(0,18,80,3,msg); -} - - -/** - ** puthelp - ** - ** Writes (msg) in the helpline area - **/ -static void -puthelp(char *msg) -{ - erase(0,22,80,1); - putxy(0,22,msg); -} - - -/** - ** masterhelp - ** - ** Draws the help message at the bottom of the screen - **/ -static void -masterhelp(char *msg) -{ - erase(0,23,80,1); - putxy(0,23,msg); -} - - -/** - ** pad - ** - ** space-pads a (str) to (len) characters - **/ -static void -pad(char *str, int len) -{ - int i; - - for (i = 0; str[i]; i++) /* find the end of the string */ - ; - if (i >= len) /* no padding needed */ - return; - while(i < len) /* pad */ - str[i++] = ' '; - str[i] = '\0'; -} - - -/** - ** drawline - ** - ** Displays entry (ofs) of (list) in region at (row) onscreen, optionally displaying - ** the port and IRQ fields if (detail) is nonzero. If (inverse), in inverse video. - ** - ** The text (dhelp) is displayed if the item is a normal device, otherwise - ** help is shown for normal or zoomed comments - **/ -static void -drawline(int row, int detail, DEV_LIST *list, int inverse, char *dhelp) -{ - char lbuf[90],nb[70],db[20],ib[16],pb[16]; - - if (list->comment == DEV_DEVICE) - { - nb[0] = ' '; - strncpy(nb+1,list->name,57); - }else{ - strncpy(nb,list->name,58); - if ((list->comment == DEV_ZOOMED) && (list->next)) - if (list->next->comment == DEV_DEVICE) /* only mention if there's something hidden */ - strcat(nb," (Collapsed)"); - } - nb[58] = '\0'; - pad(nb,60); - if (list->conflicts) /* device in conflict? */ - { - if (inverse) - { - strcpy(nb+54," !nCONF!i "); /* tag conflict, careful of length */ - }else{ - strcpy(nb+54," !iCONF!n "); /* tag conflict, careful of length */ - } - } - if (list->comment == DEV_DEVICE) - { - sprintf(db,"%s%d",list->dev,list->unit); - pad(db,8); - }else{ - strcpy(db," "); - } - if ((list->irq > 0) && detail && (list->comment == DEV_DEVICE)) - { - sprintf(ib," %d",list->irq); - pad(ib,4); - }else{ - strcpy(ib," "); - } - if ((list->iobase > 0) && detail && (list->comment == DEV_DEVICE)) - { - sprintf(pb,"0x%x",list->iobase); - pad(pb,7); - }else{ - strcpy(pb," "); - } - - sprintf(lbuf," %s%s%s%s%s",inverse?"!i":"",nb,db,ib,pb); - - putxyl(0,row,lbuf,80); - if (dhelp) - { - switch(list->comment) - { - case DEV_DEVICE: /* ordinary device */ - puthelp(dhelp); - break; - case DEV_COMMENT: - puthelp(""); - if (list->next) - if (list->next->comment == DEV_DEVICE) - puthelp(" [!bEnter!n] Collapse device list [!bC!n] Collapse all lists"); - break; - case DEV_ZOOMED: - puthelp(""); - if (list->next) - if (list->next->comment == DEV_DEVICE) - puthelp(" [!bEnter!n] Expand device list [!bX!n] Expand all lists"); - break; - default: - puthelp(" WARNING: This list entry corrupted!"); - break; - } - } - move(0,row); /* put the cursor somewhere relevant */ -} - - -/** - ** drawlist - ** - ** Displays (num) lines of the contents of (list) at (row), optionally displaying the - ** port and IRQ fields as well if (detail) is nonzero - ** - ** printf in the kernel is essentially useless, so we do most of the hard work ourselves here. - **/ -static void -drawlist(int row, int num, int detail, DEV_LIST *list) -{ - int ofs; - - for(ofs = 0; ofs < num; ofs++) - { - if (list) - { - drawline(row+ofs,detail,list,0,NULL); /* NULL -> don't draw empty help string */ - list = nextent(list); /* move down visible list */ - }else{ - erase(0,row+ofs,80,1); - } - } -} - - -/** - ** redrawactive - ** - ** Redraws the active list - **/ -static void -redrawactive(void) -{ - char cbuf[16]; - - if (conflicts) - { - sprintf(cbuf,"!i%d conflict%s-",conflicts,(conflicts>1)?"s":""); - putxy(45,0,cbuf); - }else{ - putxyl(45,0,lines,16); - } - drawlist(1,8,1,alist); /* draw device lists */ -} - -/** - ** redrawinactive - ** - ** Redraws the inactive list - **/ -static void -redrawinactive(void) -{ - drawlist(10,7,0,ilist); /* draw device lists */ -} - - -/** - ** redraw - ** - ** Clear the screen and redraw the entire layout - **/ -static void -redraw(void) -{ - clear(); - putxy(0,0,lines); - putxy(3,0,"!bActive!n-!bDrivers"); - putxy(63,0,"!bDev!n---!bIRQ!n--!bPort"); - putxy(0,9,lines); - putxy(3,9,"!bInactive!n-!bDrivers"); - putxy(63,9,"!bDev"); - putxy(0,17,lines); - putxy(0,21,lines); - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save and Exit [!b?!n] Help"); - - redrawactive(); - redrawinactive(); -} - - -/** - ** yesnocancel - ** - ** Put (str) in the message area, and return 1 if the user hits 'y' or 'Y', - ** 2 if they hit 'c' or 'C', or 0 for 'n' or 'N'. - **/ -static int -yesnocancel(char *str) -{ - - putmsg(str); - for(;;) - switch(getchar()) - { - case -1: - case 'n': - case 'N': - return(0); - - case 'y': - case 'Y': - return(1); - - case 'c': - case 'C': - return(2); - } -} - - -/** - ** showparams - ** - ** Show device parameters in the region below the lists - ** - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 17-|--------------------------------------------------------------------------------| - ** 18-| Port address : 0x0000 Memory address : 0x00000 Conflict allowed | - ** 19-| IRQ number : 00 Memory size : 0x0000 | - ** 20-| Flags : 0x0000 DRQ number : 00 | - ** 21-|--------------------------------------------------------------------------------| - **/ -static void -showparams(DEV_LIST *dev) -{ - char buf[80]; - - erase(0,18,80,3); /* clear area */ - if (!dev) - return; - if (dev->comment != DEV_DEVICE) - return; - - - if (dev->iobase > 0) - { - sprintf(buf,"Port address : 0x%x",dev->iobase); - putxy(1,18,buf); - } - - if (dev->irq > 0) - { - sprintf(buf,"IRQ number : %d",dev->irq); - putxy(1,19,buf); - } - sprintf(buf,"Flags : 0x%x",dev->flags); - putxy(1,20,buf); - if (dev->maddr > 0) - { - sprintf(buf,"Memory address : 0x%x",dev->maddr); - putxy(26,18,buf); - } - if (dev->msize > 0) - { - sprintf(buf,"Memory size : 0x%x",dev->msize); - putxy(26,19,buf); - } - - if (dev->drq > 0) - { - sprintf(buf,"DRQ number : %d",dev->drq); - putxy(26,20,buf); - } -} - - -/** - ** Editing functions for device parameters - ** - ** editval(x,y,width,hex,min,max,val) - Edit (*val) in a field (width) wide at (x,y) - ** onscreen. Refuse values outsise (min) and (max). - ** editparams(dev) - Edit the parameters for (dev) - **/ - - -#define VetRet(code) \ -{ \ - if ((i >= min) && (i <= max)) /* legit? */ \ - { \ - *val = i; \ - sprintf(buf,hex?"0x%x":"%d",i); \ - putxy(hex?x-2:x,y,buf); \ - return(code); /* all done and exit */ \ - } \ - i = *val; /* restore original value */ \ - delta = 1; /* restore other stuff */ \ -} - - -/** - ** editval - ** - ** Edit (*val) at (x,y) in (hex)?hex:decimal mode, allowing values between (min) and (max) - ** in a field (width) wide. (Allow one space) - ** If (ro) is set, we're in "readonly" mode, so disallow edits. - ** - ** Return KEY_TAB on \t, KEY_EXIT on 'q' - **/ -static int -editval(int x, int y, int width, int hex, int min, int max, int *val, int ro) -{ - int i = *val; /* work with copy of the value */ - char buf[2+11+1],tc[11+1]; /* display buffer, text copy */ - int xp = 0; /* cursor offset into text copy */ - int delta = 1; /* force redraw first time in */ - int c; - int extended = 0; /* stage counter for extended key sequences */ - - if (hex) /* we presume there's a leading 0x onscreen */ - putxy(x-2,y,"!i0x"); /* coz there sure is now */ - - for (;;) - { - if (delta) /* only update if necessary */ - { - sprintf(tc,hex?"%x":"%d",i); /* make a text copy of the value */ - sprintf(buf,"!i%s",tc); /* format for printing */ - erase(x,y,width,1); /* clear the area */ - putxy(x,y,buf); /* write */ - xp = strlen(tc); /* cursor always at end */ - move(x+xp,y); /* position the cursor */ - } - - c = getchar(); - - switch(extended) /* escape handling */ - { - case 0: - if (c == 0x1b) /* esc? */ - { - extended = 1; /* flag and spin */ - continue; - } - extended = 0; - break; /* nope, drop through */ - - case 1: /* there was an escape prefix */ - if (c == '[' || c == 'O') /* second character in sequence */ - { - extended = 2; - continue; - } - if (c == 0x1b) - return(KEY_EXIT); /* double esc exits */ - extended = 0; - break; /* nup, not a sequence. */ - - case 2: - extended = 0; - switch(c) /* looks like the real McCoy */ - { - case 'A': - VetRet(KEY_UP); /* leave if OK */ - continue; - case 'B': - VetRet(KEY_DOWN); /* leave if OK */ - continue; - case 'C': - VetRet(KEY_RIGHT); /* leave if OK */ - continue; - case 'D': - VetRet(KEY_LEFT); /* leave if OK */ - continue; - - default: - continue; - } - } - - switch(c) - { - case '\t': /* trying to tab off */ - VetRet(KEY_TAB); /* verify and maybe return */ - break; - - case -1: - case 'q': - case 'Q': - VetRet(KEY_EXIT); - break; - - case '\b': - case '\177': /* BS or DEL */ - if (ro) /* readonly? */ - { - puthelp(" !iThis value cannot be edited (Press ESC)"); - while(getchar() != 0x1b); /* wait for key */ - return(KEY_NULL); /* spin */ - } - if (xp) /* still something left to delete */ - { - i = (hex ? i/0x10u : i/10); /* strip last digit */ - delta = 1; /* force update */ - } - break; - - case 588: - VetRet(KEY_UP); - break; - - case '\r': - case '\n': - case 596: - VetRet(KEY_DOWN); - break; - - case 591: - VetRet(KEY_LEFT); - break; - - case 593: - VetRet(KEY_RIGHT); - break; - - default: - if (ro) /* readonly? */ - { - puthelp(" !iThis value cannot be edited (Press ESC)"); - while(getchar() != 0x1b); /* wait for key */ - return(KEY_NULL); /* spin */ - } - if (xp >= width) /* no room for more characters anyway */ - break; - if (hex) - { - if ((c >= '0') && (c <= '9')) - { - i = i*0x10 + (c-'0'); /* update value */ - delta = 1; - break; - } - if ((c >= 'a') && (c <= 'f')) - { - i = i*0x10 + (c-'a'+0xa); - delta = 1; - break; - } - if ((c >= 'A') && (c <= 'F')) - { - i = i*0x10 + (c-'A'+0xa); - delta = 1; - break; - } - }else{ - if ((c >= '0') && (c <= '9')) - { - i = i*10 + (c-'0'); /* update value */ - delta = 1; /* force redraw */ - break; - } - } - break; - } - } -} - - -/** - ** editparams - ** - ** Edit the parameters for (dev) - ** - ** Note that it's _always_ possible to edit the flags, otherwise it might be - ** possible for this to spin in an endless loop... - ** 0 5 10 15 20 25 30 35 40 45 50 55 60 67 70 75 - ** |....|....|....|....|....|....|....|....|....|....|....|....|....|....|....|.... - ** +--------------------------------------------------------------------------------+ - ** 17-|--------------------------------------------------------------------------------| - ** 18-| Port address : 0x0000 Memory address : 0x00000 Conflict allowed | - ** 19-| IRQ number : 00 Memory size : 0x0000 | - ** 20-| Flags : 0x0000 DRQ number : 00 | - ** 21-|--------------------------------------------------------------------------------| - ** - ** The "intelligence" in this function that hops around based on the directional - ** returns from editval isn't very smart, and depends on the layout above. - **/ -static void -editparams(DEV_LIST *dev) -{ - int ret; - char buf[16]; /* needs to fit the device name */ - - putxy(2,17,"!bParameters!n-!bfor!n-!bdevice!n-"); - sprintf(buf,"!b%s",dev->dev); - putxy(24,17,buf); - - erase(1,22,80,1); - for (;;) - { - ep_iobase: - if (dev->iobase > 0) - { - puthelp(" IO Port address (Hexadecimal, 0x1-0xffff)"); - ret = editval(18,18,5,1,0x1,0xffff,&(dev->iobase),(dev->attrib & FLG_FIXIOBASE)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->maddr > 0) - goto ep_maddr; - break; - - case KEY_TAB: - case KEY_DOWN: - goto ep_irq; - } - goto ep_iobase; - } - ep_irq: - if (dev->irq > 0) - { - puthelp(" Interrupt number (Decimal, 1-15)"); - ret = editval(16,19,3,0,1,15,&(dev->irq),(dev->attrib & FLG_FIXIRQ)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->msize > 0) - goto ep_msize; - break; - - case KEY_UP: - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_TAB: - case KEY_DOWN: - goto ep_flags; - } - goto ep_irq; - } - ep_flags: - puthelp(" Device-specific flag values."); - ret = editval(18,20,8,1,INT_MIN,INT_MAX,&(dev->flags),0); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_RIGHT: - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_UP: - if (dev->irq > 0) - goto ep_irq; - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_DOWN: - if (dev->maddr > 0) - goto ep_maddr; - if (dev->msize > 0) - goto ep_msize; - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_maddr; - } - goto ep_flags; - ep_maddr: - if (dev->maddr > 0) - { - puthelp(" Device memory start address (Hexadecimal, 0x1-0xfffff)"); - ret = editval(45,18,6,1,0x1,0xfffff,&(dev->maddr),(dev->attrib & FLG_FIXMADDR)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - if (dev->iobase > 0) - goto ep_iobase; - break; - - case KEY_UP: - goto ep_flags; - - case KEY_DOWN: - if (dev->msize > 0) - goto ep_msize; - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_msize; - } - goto ep_maddr; - } - ep_msize: - if (dev->msize > 0) - { - puthelp(" Device memory size (Hexadecimal, 0x1-0x10000)"); - ret = editval(45,19,5,1,0x1,0x10000,&(dev->msize),(dev->attrib & FLG_FIXMSIZE)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - if (dev->irq > 0) - goto ep_irq; - break; - - case KEY_UP: - if (dev->maddr > 0) - goto ep_maddr; - goto ep_flags; - - case KEY_DOWN: - if (dev->drq > 0) - goto ep_drq; - break; - - case KEY_TAB: - goto ep_drq; - } - goto ep_msize; - } - ep_drq: - if (dev->drq > 0) - { - puthelp(" Device DMA request number (Decimal, 1-7)"); - ret = editval(43,20,2,0,1,7,&(dev->drq),(dev->attrib & FLG_FIXDRQ)); - switch(ret) - { - case KEY_EXIT: - goto ep_exit; - - case KEY_LEFT: - goto ep_flags; - - case KEY_UP: - if (dev->msize > 0) - goto ep_msize; - if (dev->maddr > 0) - goto ep_maddr; - goto ep_flags; - - case KEY_TAB: - goto ep_iobase; - } - goto ep_drq; - } - } - ep_exit: - dev->changed = 1; /* mark as changed */ -} - -static char *helptext[] = -{ - " Using the UserConfig kernel settings editor", - " -------------------------------------------", - "", - "VISUAL MODE:", - "", - "- - Layout -", - "", - "The screen displays a list of available drivers, divided into two", - "scrolling lists: Active Drivers, and Inactive Drivers. Each list is", - "by default collapsed and can be expanded to show all the drivers", - "available in each category. The parameters for the currently selected", - "driver are shown at the bottom of the screen.", - "", - "- - Moving around -", - "", - "To move in the current list, use the UP and DOWN cursor keys to select", - "an item (the selected item will be highlighted). If the item is a", - "category name, you may alternatively expand or collapse the list of", - "drivers for that category by pressing [!bENTER!n]. Once the category is", - "expanded, you can select each driver in the same manner and either:", - "", - " - change its parameters using [!bENTER!n]", - " - move it to the Inactive list using [!bDEL!n]", - "", - "Use the [!bTAB!n] key to toggle between the Active and Inactive list; if", - "you need to move a driver from the Inactive list back to the Active", - "one, select it in the Inactive list, using [!bTAB!n] to change lists if", - "necessary, and press [!bENTER!n] -- the device will be moved back to", - "its place in the Active list.", - "", - "- - Altering the list/parameters -", - "", - "Any drivers for devices not installed in your system should be moved", - "to the Inactive list, until there are no remaining parameter conflicts", - "between the drivers, as indicated at the top.", - "", - "Once the list of Active drivers only contains entries for the devices", - "present in your system, you can set their parameters (Interrupt, DMA", - "channel, I/O addresses). To do this, select the driver and press", - "[!bENTER!n]: it is now possible to edit the settings at the", - "bottom of the screen. Use [!bTAB!n] to change fields, and when you are", - "finished, use [!bQ!n] to return to the list.", - "", - "- - Saving changes -", - "", - "When all settings seem correct, and you wish to proceed with the", - "kernel device probing and boot, press [!bQ!n] -- you will be asked to", - "confirm your choice.", - "", - NULL -}; - - -/** - ** helpscreen - ** - ** Displays help text onscreen for people that are confused, using a simple - ** pager. - **/ -static void -helpscreen(void) -{ - int topline = 0; /* where we are in the text */ - int line = 0; /* last line we displayed */ - int c, delta = 1; - char prompt[80]; - - for (;;) /* loop until user quits */ - { - /* display help text */ - if (delta) - { - clear(); /* remove everything else */ - for (line = topline; - (line < (topline + 24)) && (helptext[line]); - line++) - putxy(0,line-topline,helptext[line]); - delta = 0; - } - - /* prompt */ - sprintf(prompt,"!i --%s-- [U]p [D]own [Q]uit !n",helptext[line] ? "MORE" : "END"); - putxy(0,24,prompt); - - c = getchar(); /* so what do they say? */ - - switch (c) - { - case 'u': - case 'U': - case 'b': - case 'B': /* wired into 'more' users' fingers */ - if (topline > 0) /* room to go up? */ - { - topline -= 24; - if (topline < 0) /* don't go too far */ - topline = 0; - delta = 1; - } - break; - - case 'd': - case 'D': - case ' ': /* expected by most people */ - if (helptext[line]) /* maybe more below? */ - { - topline += 24; - delta = 1; - } - break; - - case 'q': - case 'Q': - redraw(); /* restore the screen */ - return; - } - } -} - - -/** - ** High-level control functions - **/ - - -/** - ** dolist - ** - ** Handle user movement within (*list) in the region starting at (row) onscreen with - ** (num) lines, starting at (*ofs) offset from row onscreen. - ** Pass (detail) on to drawing routines. - ** - ** If the user hits a key other than a cursor key, maybe return a code. - ** - ** (*list) points to the device at the top line in the region, (*ofs) is the - ** position of the highlight within the region. All routines below - ** this take only a device and an absolute row : use ofsent() to find the - ** device, and add (*ofs) to (row) to find the absolute row. - **/ -static int -dolist(int row, int num, int detail, int *ofs, DEV_LIST **list, char *dhelp) -{ - int extended = 0; - int c; - DEV_LIST *lp; - int delta = 1; - - for(;;) - { - if (delta) - { - showparams(ofsent(*ofs,*list)); /* show device parameters */ - drawline(row+*ofs,detail,ofsent(*ofs,*list),1,dhelp); /* highlight current line */ - delta = 0; - } - - c = getchar(); /* get a character */ - if ((extended == 2) || (c==588) || (c==596)) /* console gives "alternative" codes */ - { - extended = 0; /* no longer */ - switch(c) - { - case 588: /* syscons' idea of 'up' */ - case 'A': /* up */ - if (*ofs) /* just a move onscreen */ - { - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp);/* unhighlight current line */ - (*ofs)--; /* move up */ - }else{ - lp = prevent(*list); /* can we go up? */ - if (!lp) /* no */ - break; - *list = lp; /* yes, move up list */ - drawlist(row,num,detail,*list); - } - delta = 1; - break; - - case 596: /* dooby-do */ - case 'B': /* down */ - lp = ofsent(*ofs,*list); /* get current item */ - if (!nextent(lp)) - break; /* nothing more to move to */ - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp); /* unhighlight current line */ - if (*ofs < (num-1)) /* room to move onscreen? */ - { - (*ofs)++; - }else{ - *list = nextent(*list); /* scroll region down */ - drawlist(row,num,detail,*list); - } - delta = 1; - break; - } - }else{ - switch(c) - { - case '\033': - extended=1; - break; - - case '[': /* cheat : always precedes cursor move */ - case 'O': /* ANSI application key mode */ - if (extended==1) - extended=2; - else - extended=0; - break; - - case 'Q': - case 'q': - return(KEY_EXIT); /* user requests exit */ - - case '\r': - case '\n': - return(KEY_DO); /* "do" something */ - - case '\b': - case '\177': - case 599: - return(KEY_DEL); /* "delete" response */ - - case 'X': - case 'x': - return(KEY_UNZOOM); /* expand everything */ - - case 'C': - case 'c': - return(KEY_ZOOM); /* collapse everything */ - - case '\t': - drawline(row+*ofs,detail,ofsent(*ofs,*list),0,dhelp); /* unhighlight current line */ - return(KEY_TAB); /* "move" response */ - - case '\014': /* ^L, redraw */ - return(KEY_REDRAW); - - case '?': /* helptext */ - return(KEY_HELP); - - } - } - } -} - - -/** - ** visuserconfig - ** - ** Do the fullscreen config thang - **/ -static int -visuserconfig(void) -{ - int actofs = 0, inactofs = 0, mode = 0, ret = -1, i; - DEV_LIST *dp; - - initlist(&active); - initlist(&inactive); - alist = active; - ilist = inactive; - - getdevs(); - - conflicts = findconflict(active); /* find conflicts in the active list only */ - - redraw(); - - for(;;) - { - switch(mode) - { - case 0: /* active devices */ - ret = dolist(1,8,1,&actofs,&alist, - " [!bEnter!n] Edit device parameters [!bDEL!n] Disable device"); - switch(ret) - { - case KEY_TAB: - mode = 1; /* swap lists */ - break; - - case KEY_REDRAW: - redraw(); - break; - - case KEY_ZOOM: - alist = active; - actofs = 0; - expandlist(active); - redrawactive(); - break; - - case KEY_UNZOOM: - alist = active; - actofs = 0; - collapselist(active); - redrawactive(); - break; - - case KEY_DEL: - dp = ofsent(actofs,alist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->attrib & FLG_MANDATORY) /* can't be deleted */ - break; - if (dp == alist) /* moving top item on list? */ - { - if (dp->next) - { - alist = dp->next; /* point list to non-moving item */ - }else{ - alist = dp->prev; /* end of list, go back instead */ - } - }else{ - if (!dp->next) /* moving last item on list? */ - actofs--; - } - dp->conflicts = 0; /* no conflicts on the inactive list */ - movedev(dp,inactive); /* shift to inactive list */ - conflicts = findconflict(active); /* update conflict tags */ - dp->changed = 1; - redrawactive(); /* redraw */ - redrawinactive(); - } - break; - - case KEY_DO: /* edit device parameters */ - dp = ofsent(actofs,alist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->comment == DEV_DEVICE) /* can't edit comments, zoom? */ - { - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save device parameters"); - editparams(dp); - masterhelp(" [!bTAB!n] Change fields [!bQ!n] Save and Exit [!b?!n] Help"); - putxy(0,17,lines); - conflicts = findconflict(active); /* update conflict tags */ - }else{ /* DO on comment = zoom */ - switch(dp->comment) /* Depends on current state */ - { - case DEV_COMMENT: /* not currently zoomed */ - dp->comment = DEV_ZOOMED; - break; - - case DEV_ZOOMED: - dp->comment = DEV_COMMENT; - break; - } - } - redrawactive(); - } - break; - } - break; - - case 1: /* inactive devices */ - ret = dolist(10,7,0,&inactofs,&ilist, - " [!bEnter!n] Enable device "); - switch(ret) - { - case KEY_TAB: - mode = 0; - break; - - case KEY_REDRAW: - redraw(); - break; - - case KEY_ZOOM: - ilist = inactive; - inactofs = 0; - expandlist(inactive); - redrawinactive(); - break; - - case KEY_UNZOOM: - ilist = inactive; - inactofs = 0; - collapselist(inactive); - redrawinactive(); - break; - - case KEY_DO: - dp = ofsent(inactofs,ilist); /* get current device */ - if (dp) /* paranoia... */ - { - if (dp->comment == DEV_DEVICE) /* can't move comments, zoom? */ - { - if (dp == ilist) /* moving top of list? */ - { - if (dp->next) - { - ilist = dp->next; /* point list to non-moving item */ - }else{ - ilist = dp->prev; /* can't go down, go up instead */ - } - }else{ - if (!dp->next) /* last entry on list? */ - inactofs--; /* shift cursor up one */ - } - - movedev(dp,active); /* shift to active list */ - conflicts = findconflict(active); /* update conflict tags */ - dp->changed = 1; - alist = dp; /* put at top and current */ - actofs = 0; - while(dp->comment == DEV_DEVICE) - dp = dp->prev; /* forcibly unzoom section */ - dp ->comment = DEV_COMMENT; - mode = 0; /* and swap modes to follow it */ - - }else{ /* DO on comment = zoom */ - switch(dp->comment) /* Depends on current state */ - { - case DEV_COMMENT: /* not currently zoomed */ - dp->comment = DEV_ZOOMED; - break; - - case DEV_ZOOMED: - dp->comment = DEV_COMMENT; - break; - } - } - redrawactive(); /* redraw */ - redrawinactive(); - } - break; - - default: /* nothing else relevant here */ - break; - } - break; - default: - mode = 0; /* shouldn't happen... */ - } - - /* handle returns that are the same for both modes */ - switch (ret) { - case KEY_HELP: - helpscreen(); - break; - - case KEY_EXIT: - i = yesnocancel(" Save these parameters before exiting? ([!bY!n]es/[!bN!n]o/[!bC!n]ancel) "); - switch(i) - { - case 2: /* cancel */ - redraw(); - break; - - case 1: /* save and exit */ - savelist(active,1); - savelist(inactive,0); - - case 0: /* exit */ - nukelist(active); /* clean up after ourselves */ - nukelist(inactive); - normal(); - clear(); - return(1); - } - break; - } - } -} -#endif /* VISUAL_USERCONFIG */ - -/* - * Copyright (c) 1991 Regents of the University of California. - * All rights reserved. - * Copyright (c) 1994 Jordan K. Hubbard - * All rights reserved. - * Copyright (c) 1994 David Greenman - * All rights reserved. - * - * Many additional changes by Bruce Evans - * - * This code is derived from software contributed by the - * University of California Berkeley, Jordan K. Hubbard, - * David Greenman and Bruce Evans. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: - * 1. Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * 3. All advertising materials mentioning features or use of this software - * must display the following acknowledgement: - * This product includes software developed by the University of - * California, Berkeley and its contributors. - * 4. Neither the name of the University nor the names of its contributors - * may be used to endorse or promote products derived from this software - * without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND - * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS - * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT - * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY - * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF - * SUCH DAMAGE. - * - * $FreeBSD$ - */ - -#define PARM_DEVSPEC 0x1 -#define PARM_INT 0x2 -#define PARM_ADDR 0x3 -#define PARM_STRING 0x4 - -typedef struct _cmdparm { - int type; - union { - struct uc_device *dparm; - int iparm; - union { - void *aparm; - const char *sparm; - } u; - } parm; -} CmdParm; - -typedef int (*CmdFunc)(CmdParm *); - -typedef struct _cmd { - char *name; - CmdFunc handler; - CmdParm *parms; -} Cmd; - - -static int lsdevtab(struct uc_device *); -static struct uc_device *find_device(char *, int); -static struct uc_device *search_devtable(struct uc_device *, char *, int); -static void cngets(char *, int); -static Cmd *parse_cmd(char *); -static int parse_args(const char *, CmdParm *); -static int save_dev(struct uc_device *); - -static int list_devices(CmdParm *); -static int set_device_ioaddr(CmdParm *); -static int set_device_irq(CmdParm *); -static int set_device_drq(CmdParm *); -static int set_device_iosize(CmdParm *); -static int set_device_mem(CmdParm *); -static int set_device_flags(CmdParm *); -static int set_device_enable(CmdParm *); -static int set_device_disable(CmdParm *); -static int quitfunc(CmdParm *); -static int helpfunc(CmdParm *); -#if defined(INTRO_USERCONFIG) -static int introfunc(CmdParm *); -#endif - -static int lineno; - -#ifdef DEV_EISA - -#include <dev/eisa/eisaconf.h> - -static int set_num_eisa_slots(CmdParm *); - -#endif - -static CmdParm addr_parms[] = { - { PARM_DEVSPEC, {} }, - { PARM_ADDR, {} }, - { -1, {} }, -}; - -static CmdParm int_parms[] = { - { PARM_DEVSPEC, {} }, - { PARM_INT, {} }, - { -1, {} }, -}; - -static CmdParm dev_parms[] = { - { PARM_DEVSPEC, {} }, - { -1, {} }, -}; - -#ifdef DEV_EISA -static CmdParm int_arg[] = { - { PARM_INT, {} }, - { -1, {} }, -}; -#endif - -static Cmd CmdList[] = { - { "?", helpfunc, NULL }, /* ? (help) */ - { "di", set_device_disable, dev_parms }, /* disable dev */ - { "dr", set_device_drq, int_parms }, /* drq dev # */ -#ifdef DEV_EISA - { "ei", set_num_eisa_slots, int_arg }, /* # EISA slots */ -#endif - { "en", set_device_enable, dev_parms }, /* enable dev */ - { "ex", quitfunc, NULL }, /* exit (quit) */ - { "f", set_device_flags, int_parms }, /* flags dev mask */ - { "h", helpfunc, NULL }, /* help */ -#if defined(INTRO_USERCONFIG) - { "intro", introfunc, NULL }, /* intro screen */ -#endif - { "iom", set_device_mem, addr_parms }, /* iomem dev addr */ - { "ios", set_device_iosize, int_parms }, /* iosize dev size */ - { "ir", set_device_irq, int_parms }, /* irq dev # */ - { "l", list_devices, NULL }, /* ls, list */ - { "po", set_device_ioaddr, int_parms }, /* port dev addr */ - { "res", (CmdFunc)cpu_reset, NULL }, /* reset CPU */ - { "q", quitfunc, NULL }, /* quit */ -#ifdef VISUAL_USERCONFIG - { "v", (CmdFunc)visuserconfig, NULL }, /* visual mode */ -#endif - { NULL, NULL, NULL }, -}; - -void -userconfig(void) -{ - static char banner = 1; - char input[80]; - int rval; - Cmd *cmd; - - load_devtab(); - init_config_script(); - while (1) { - - /* Only display signon banner if we are about to go interactive */ - if (!has_config_script()) { - if (!(boothowto & RB_CONFIG)) -#ifdef INTRO_USERCONFIG - banner = 0; -#else - return; -#endif - if (banner) { - banner = 0; - printf("FreeBSD Kernel Configuration Utility - Version 1.2\n" - " Type \"help\" for help" -#ifdef VISUAL_USERCONFIG - " or \"visual\" to go to the visual\n" - " configuration interface (requires MGA/VGA display or\n" - " serial terminal capable of displaying ANSI graphics)" -#endif - ".\n"); - } - } - - printf("config> "); - cngets(input, 80); - if (input[0] == '\0') - continue; - cmd = parse_cmd(input); - if (!cmd) { - printf("Invalid command or syntax. Type `?' for help.\n"); - continue; - } - rval = (*cmd->handler)(cmd->parms); - if (rval) { - free_devtab(); - return; - } - } -} - -static Cmd * -parse_cmd(char *cmd) -{ - Cmd *cp; - - for (cp = CmdList; cp->name; cp++) { - int len = strlen(cp->name); - - if (!strncmp(cp->name, cmd, len)) { - while (*cmd && *cmd != ' ' && *cmd != '\t') - ++cmd; - if (parse_args(cmd, cp->parms)) - return NULL; - else - return cp; - } - } - return NULL; -} - -static int -parse_args(const char *cmd, CmdParm *parms) -{ - while (1) { - char *ptr; - - if (*cmd == ' ' || *cmd == '\t') { - ++cmd; - continue; - } - if (parms == NULL || parms->type == -1) { - if (*cmd == '\0') - return 0; - printf("Extra arg(s): %s\n", cmd); - return 1; - } - if (parms->type == PARM_DEVSPEC) { - int i = 0; - char devname[64]; - int unit = 0; - - while (*cmd && !(*cmd == ' ' || *cmd == '\t' || - (*cmd >= '0' && *cmd <= '9'))) - devname[i++] = *(cmd++); - devname[i] = '\0'; - if (*cmd >= '0' && *cmd <= '9') { - unit = strtoul(cmd, &ptr, 10); - if (cmd == ptr) { - printf("Invalid device number\n"); - /* XXX should print invalid token here and elsewhere. */ - return 1; - } - /* XXX else should require end of token. */ - cmd = ptr; - } - if ((parms->parm.dparm = find_device(devname, unit)) == NULL) { - printf("No such device: %s%d\n", devname, unit); - return 1; - } - ++parms; - continue; - } - if (parms->type == PARM_INT) { - parms->parm.iparm = strtoul(cmd, &ptr, 0); - if (cmd == ptr) { - printf("Invalid numeric argument\n"); - return 1; - } - cmd = ptr; - ++parms; - continue; - } - if (parms->type == PARM_ADDR) { - parms->parm.u.aparm = (void *)(uintptr_t)strtoul(cmd, &ptr, 0); - if (cmd == ptr) { - printf("Invalid address argument\n"); - return 1; - } - cmd = ptr; - ++parms; - continue; - } - if (parms->type == PARM_STRING) { - parms->parm.u.sparm = cmd; - return 0; - } - } - return 0; -} - -static int -list_devices(CmdParm *parms) -{ - lineno = 0; - if (lsdevtab(uc_devtab)) return 0; -#ifdef DEV_EISA - printf("\nNumber of EISA slots to probe: %d\n", num_eisa_slots); -#endif - return 0; -} - -static int -set_device_ioaddr(CmdParm *parms) -{ - parms[0].parm.dparm->id_iobase = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_irq(CmdParm *parms) -{ - unsigned irq; - - irq = parms[1].parm.iparm; -#ifndef PC98 - if (irq == 2) { - printf("Warning: Remapping IRQ 2 to IRQ 9\n"); - irq = 9; - } - else if (irq != -1 && irq > 15) { -#else - if (irq != -1 && irq > 15) { -#endif - printf("An IRQ > 15 would be invalid.\n"); - return 0; - } - parms[0].parm.dparm->id_irq = (irq < 16 ? 1 << irq : 0); - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_drq(CmdParm *parms) -{ - unsigned drq; - - /* - * The bounds checking is just to ensure that the value can be printed - * in 5 characters. 32768 gets converted to -32768 and doesn't fit. - */ - drq = parms[1].parm.iparm; - parms[0].parm.dparm->id_drq = (drq < 32768 ? drq : -1); - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_iosize(CmdParm *parms) -{ - parms[0].parm.dparm->id_msize = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_mem(CmdParm *parms) -{ - parms[0].parm.dparm->id_maddr = parms[1].parm.u.aparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_flags(CmdParm *parms) -{ - parms[0].parm.dparm->id_flags = parms[1].parm.iparm; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_enable(CmdParm *parms) -{ - parms[0].parm.dparm->id_enabled = TRUE; - save_dev(parms[0].parm.dparm); - return 0; -} - -static int -set_device_disable(CmdParm *parms) -{ - parms[0].parm.dparm->id_enabled = FALSE; - save_dev(parms[0].parm.dparm); - return 0; -} - -#ifdef DEV_EISA -static int -set_num_eisa_slots(CmdParm *parms) -{ - int num_slots; - - num_slots = parms[0].parm.iparm; - num_eisa_slots = (num_slots <= 16 ? num_slots : 10); - return 0; -} -#endif - -static int -quitfunc(CmdParm *parms) -{ - /* - * If kernel config supplied, and we are parsing it, and -c also supplied, - * ignore a quit command, This provides a safety mechanism to allow - * recovery from a damaged/buggy kernel config. - */ - if ((boothowto & RB_CONFIG) && userconfig_boot_parsing) - return 0; - return 1; -} - -static int -helpfunc(CmdParm *parms) -{ - printf( - "Command\t\t\tDescription\n" - "-------\t\t\t-----------\n" - "ls\t\t\tList currently configured devices\n" - "port <devname> <addr>\tSet device port (i/o address)\n" - "irq <devname> <number>\tSet device irq\n" - "drq <devname> <number>\tSet device drq\n" - "iomem <devname> <addr>\tSet device maddr (memory address)\n" - "iosize <devname> <size>\tSet device memory size\n" - "flags <devname> <mask>\tSet device flags\n" - "enable <devname>\tEnable device\n" - "disable <devname>\tDisable device (will not be probed)\n"); -#ifdef DEV_EISA - printf("eisa <number>\t\tSet the number of EISA slots to probe\n"); -#endif - printf( - "quit\t\t\tExit this configuration utility\n" - "reset\t\t\tReset CPU\n"); -#ifdef VISUAL_USERCONFIG - printf("visual\t\t\tGo to fullscreen mode.\n"); -#endif - printf( - "help\t\t\tThis message\n\n" - "Commands may be abbreviated to a unique prefix\n"); - return 0; -} - -#if defined(INTRO_USERCONFIG) - -#if defined (VISUAL_USERCONFIG) -static void -center(int y, char *str) -{ - putxy((80 - strlen(str)) / 2, y, str); -} -#endif - -static int -introfunc(CmdParm *parms) -{ -#if defined (VISUAL_USERCONFIG) - int curr_item, first_time, extended = 0; - static char *choices[] = { - " Skip kernel configuration and continue with installation ", - " Start kernel configuration in full-screen visual mode ", - " Start kernel configuration in CLI mode ", - }; - - clear(); - center(2, "!bKernel Configuration Menu!n"); - - curr_item = 0; - first_time = 1; - while (1) { - char tmp[80]; - int c, i; - - if (!extended) { - for (i = 0; i < 3; i++) { - tmp[0] = '\0'; - if (curr_item == i) - strcpy(tmp, "!i"); - strcat(tmp, choices[i]); - if (curr_item == i) - strcat(tmp, "!n"); - putxy(10, 5 + i, tmp); - } - - if (first_time) { - putxy(2, 10, "Here you have the chance to go into kernel configuration mode, making"); - putxy(2, 11, "any changes which may be necessary to properly adjust the kernel to"); - putxy(2, 12, "match your hardware configuration."); - putxy(2, 14, "If you are installing FreeBSD for the first time, select Visual Mode"); - putxy(2, 15, "(press Down-Arrow then ENTER)."); - putxy(2, 17, "If you need to do more specialized kernel configuration and are an"); - putxy(2, 18, "experienced FreeBSD user, select CLI mode."); - putxy(2, 20, "If you are !icertain!n that you do not need to configure your kernel"); - putxy(2, 21, "then simply press ENTER or Q now."); - first_time = 0; - } - - move(0, 0); /* move the cursor out of the way */ - } - c = getchar(); - if ((extended == 2) || (c == 588) || (c == 596)) { /* console gives "alternative" codes */ - extended = 0; /* no longer */ - switch (c) { - case 588: - case 'A': /* up */ - if (curr_item > 0) - --curr_item; - break; - - case 596: - case 'B': /* down */ - if (curr_item < 2) - ++curr_item; - break; - } - } - else { - switch(c) { - case '\033': - extended = 1; - break; - - case '[': /* cheat : always precedes cursor move */ - case 'O': /* ANSI application key mode */ - if (extended == 1) - extended = 2; - else - extended = 0; - break; - - case -1: - case 'Q': - case 'q': - clear(); - return 1; /* user requests exit */ - - case '1': /* select an item */ - case 'S': - case 's': - curr_item = 0; - break; - case '2': - case 'V': - case 'v': - curr_item = 1; - break; - case '3': - case 'C': - case 'c': - curr_item = 2; - break; - - case 'U': /* up */ - case 'u': - case 'P': - case 'p': - if (curr_item > 0) - --curr_item; - break; - - case 'D': /* down */ - case 'd': - case 'N': - case 'n': - if (curr_item < 2) - ++curr_item; - break; - - case '\r': - case '\n': - clear(); - if (!curr_item) - return 1; - else if (curr_item == 1) - return visuserconfig(); - else { - putxy(0, 1, "Type \"help\" for help or \"quit\" to exit."); - /* enable quitfunc */ - userconfig_boot_parsing=0; - move (0, 3); - boothowto |= RB_CONFIG; /* force -c */ - return 0; - } - break; - } - } - } -#endif -} -#endif - -static int -lsdevtab(struct uc_device *dt) -{ - for (; dt->id_id != 0; dt++) { - char dname[80]; - - if (lineno >= 23) { - printf("<More> "); - if (!userconfig_boot_parsing) { - if (getchar() == 'q') { - printf("quit\n"); - return (1); - } - printf("\n"); - } - lineno = 0; - } - if (lineno == 0) { - printf( -"Device port irq drq iomem iosize unit flags enab\n" - ); - ++lineno; - } - sprintf(dname, "%s%d", dt->id_name, dt->id_unit); - printf("%-9.9s%-#11x%-6d%-6d%-8p%-9d%-6d%-#11x%-5s\n", - dname, /* dt->id_id, dt->id_driver(by name), */ dt->id_iobase, - ffs(dt->id_irq) - 1, dt->id_drq, dt->id_maddr, dt->id_msize, - /* dt->id_intr(by name), */ dt->id_unit, dt->id_flags, - dt->id_enabled ? "Yes" : "No"); - ++lineno; - } - return(0); -} - -static void -load_devtab(void) -{ - int i, val; - int count = resource_count(); - int id = 1; - int dt; - char *name; - int unit; - - uc_devtab = malloc(sizeof(struct uc_device) * (count + 1), M_DEVL, - M_WAITOK | M_ZERO); - dt = 0; - for (i = 0; i < count; i++) { - name = resource_query_name(i); - unit = resource_query_unit(i); - if (unit < 0) - continue; /* skip wildcards */ - uc_devtab[dt].id_id = id++; - resource_int_value(name, unit, "port", &uc_devtab[dt].id_iobase); - val = 0; - resource_int_value(name, unit, "irq", &val); - uc_devtab[dt].id_irq = (1 << val); - resource_int_value(name, unit, "drq", &uc_devtab[dt].id_drq); - resource_int_value(name, unit, "maddr",(int *)&uc_devtab[dt].id_maddr); - resource_int_value(name, unit, "msize", &uc_devtab[dt].id_msize); - uc_devtab[dt].id_unit = unit; - resource_int_value(name, unit, "flags", &uc_devtab[dt].id_flags); - val = 0; - resource_int_value(name, unit, "disabled", &val); - uc_devtab[dt].id_enabled = !val; - uc_devtab[dt].id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(uc_devtab[dt].id_name, name); - dt++; - } -} - -static void -free_devtab(void) -{ - int i; - int count = resource_count(); - - for (i = 0; i < count; i++) - if (uc_devtab[i].id_name) - free(uc_devtab[i].id_name, M_DEVL); - free(uc_devtab, M_DEVL); -} - -static struct uc_device * -find_device(char *devname, int unit) -{ - struct uc_device *ret; - - if ((ret = search_devtable(uc_devtab, devname, unit)) != NULL) - return ret; - return NULL; -} - -static struct uc_device * -search_devtable(struct uc_device *dt, char *devname, int unit) -{ - int i; - - for (i = 0; dt->id_id != 0; dt++) - if (!strcmp(dt->id_name, devname) && dt->id_unit == unit) - return dt; - return NULL; -} - -static void -cngets(char *input, int maxin) -{ - int c, nchars = 0; - - while (1) { - c = getchar(); - /* Treat ^H or ^? as backspace */ - if ((c == '\010' || c == '\177')) { - if (nchars) { - printf("\010 \010"); - *--input = '\0', --nchars; - } - continue; - } - /* Treat ^U or ^X as kill line */ - else if ((c == '\025' || c == '\030')) { - while (nchars) { - printf("\010 \010"); - *--input = '\0', --nchars; - } - continue; - } - printf("%c", c); - if ((++nchars == maxin) || (c == '\n') || (c == '\r') || ( c == -1)) { - *input = '\0'; - break; - } - *input++ = (u_char)c; - } -} - -static void -save_resource(struct uc_device *idev) -{ - char *name; - int unit; - - name = idev->id_name; - unit = idev->id_unit; - resource_set_int(name, unit, "port", idev->id_iobase); - resource_set_int(name, unit, "irq", ffs(idev->id_irq) - 1); - resource_set_int(name, unit, "drq", idev->id_drq); - resource_set_int(name, unit, "maddr", (int)idev->id_maddr); - resource_set_int(name, unit, "msize", idev->id_msize); - resource_set_int(name, unit, "flags", idev->id_flags); - resource_set_int(name, unit, "disabled", !idev->id_enabled); -} - -static int -save_dev(idev) -struct uc_device *idev; -{ - struct uc_device *id_p,*id_pn; - char *name = idev->id_name; - - for (id_p = uc_devlist; id_p; id_p = id_p->id_next) { - if (id_p->id_id == idev->id_id) { - id_pn = id_p->id_next; - if (id_p->id_name) - free(id_p->id_name, M_DEVL); - bcopy(idev,id_p,sizeof(struct uc_device)); - save_resource(idev); - id_p->id_name = malloc(strlen(name)+1, M_DEVL,M_WAITOK); - strcpy(id_p->id_name, name); - id_p->id_next = id_pn; - return 1; - } - } - id_pn = malloc(sizeof(struct uc_device),M_DEVL,M_WAITOK); - bcopy(idev,id_pn,sizeof(struct uc_device)); - save_resource(idev); - id_pn->id_name = malloc(strlen(name) + 1, M_DEVL,M_WAITOK); - strcpy(id_pn->id_name, name); - id_pn->id_next = uc_devlist; - uc_devlist = id_pn; - return 0; -} - - |