/* $NetBSD: dev_hppa.c,v 1.2 2014/03/26 17:57:17 christos Exp $ */ /* $OpenBSD: dev_hppa.c,v 1.5 1999/04/20 20:01:01 mickey Exp $ */ /* * Copyright (c) 1998-2004 Michael Shalayeff * All rights reserved. * * 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. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR OR HIS RELATIVES 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 MIND, 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. * */ #include "libsa.h" #include #include #include #include #include #include "dev_hppa.h" extern int debug; const char cdevs[][4] = { "ite", "", "", "", "", "", "", "", "", "", "", "", "" }; const int ncdevs = NENTS(cdevs); const struct pdc_devs { char name[3]; int dev_type; } pdc_devs[] = { { "dk", 0 }, { "ct", 1 }, { "lf", 2 }, { "", -1 }, { "rd", -1 }, { "sw", -1 }, { "fl", -1 }, }; /* pass dev_t to the open routines */ int devopen(struct open_file *f, const char *fname, char **file) { struct hppa_dev *hpd; const struct pdc_devs *dp = pdc_devs; int bdev, badapt, bctlr, bunit, bpart; unsigned long n; char *p; int rc = 1; if (!(*file = strchr(fname, ':'))) return ENODEV; else (*file)++; #ifdef DEBUGBUG if (debug) printf("devopen: "); #endif for (dp = pdc_devs; dp < &pdc_devs[NENTS(pdc_devs)]; dp++) if (!strncmp(fname, dp->name, sizeof(dp->name)-1)) break; if (dp >= &pdc_devs[NENTS(pdc_devs)] || dp->dev_type < 0) return ENODEV; bdev = dp->dev_type; n = strtoul(fname + sizeof(dp->name)-1, &p, 10); if (n == ULONG_MAX) return ENODEV; bunit = n & 0xf; bctlr = (n >> 4) & 0xf; badapt = (n >> 8) & 0xf; if (*p >= 'a' && *p < 'a' + MAXPARTITIONS) { bpart = *p - 'a'; } else { bpart = 0; } bootdev = MAKEBOOTDEV(bdev, badapt, bctlr, bunit, bpart); #ifdef DEBUGBUG if (debug) printf("%s\n", dp->name); #endif if (!(hpd = alloc(sizeof *hpd))) { #ifdef DEBUG printf ("devopen: no mem\n"); #endif } else { memset(hpd, 0, sizeof *hpd); hpd->bootdev = bootdev; hpd->buf = (char *)(((u_int)hpd->ua_buf + IODC_MINIOSIZ-1) & ~(IODC_MINIOSIZ-1)); f->f_devdata = hpd; if ((rc = (*devsw[dp->dev_type].dv_open)(f, file)) == 0) { f->f_dev = &devsw[dp->dev_type]; return 0; } dealloc (hpd, 0); f->f_devdata = NULL; } if (!(f->f_flags & F_NODEV)) f->f_dev = &devsw[dp->dev_type]; if (!f->f_devdata) *file = NULL; return rc; } void devboot(btdev_t dev, char *p) { const char *q; if (!dev) { int type, unit; switch (PAGE0->mem_boot.pz_class) { case PCL_RANDOM: type = 0; unit = PAGE0->mem_boot.pz_layers[0]; break; case PCL_SEQU: type = 1; unit = PAGE0->mem_boot.pz_layers[0]; break; case PCL_NET_MASK|PCL_SEQU: type = 2; unit = 0; break; default: type = 0; unit = 0; break; } dev = bootdev = MAKEBOOTDEV(type, 0, 0, unit, 0); } #ifdef _TEST *p++ = '/'; *p++ = 'd'; *p++ = 'e'; *p++ = 'v'; *p++ = '/'; *p++ = 'r'; #endif /* quick copy device name */ for (q = pdc_devs[B_TYPE(dev)].name; (*p++ = *q++);); p[-1] = '0' + B_UNIT(dev); *p++ = 'a' + B_PARTITION(dev); *p = '\0'; } int pch_pos; void putchar(int c) { switch(c) { case '\177': /* DEL erases */ cnputc('\b'); cnputc(' '); case '\b': cnputc('\b'); if (pch_pos) pch_pos--; break; case '\t': do cnputc(' '); while(++pch_pos % 8); break; case '\n': /* * XXX fredette - probably only necessary * when using a serial console? */ cnputc(c); c = '\r'; /* FALLTHROUGH */ case '\r': cnputc(c); pch_pos=0; break; default: cnputc(c); pch_pos++; break; } } int getchar(void) { int c = cngetc(); if (c == '\r') c = '\n'; if ((c < ' ' && c != '\n') || c == '\177') return(c); /* * XXX fredette - probably only unnecessary * when using a serial console? */ #if 0 putchar(c); #endif return(c); } int tgetchar(void) { int c; if ((c = tcngetc()) == 0) return(0); return(getchar()); } #if 0 char ttyname_buf[8]; char * ttyname(int fd) { snprintf(ttyname_buf, sizeof(ttyname_buf), "%s%d", cdevs[major(cn_tab->cn_dev)], minor(cn_tab->cn_dev)); return (ttyname_buf); } dev_t ttydev(char *name) { int i, unit = -1; char *no = name + strlen(name) - 1; while (no >= name && *no >= '0' && *no <= '9') unit = (unit < 0 ? 0 : (unit * 10)) + *no-- - '0'; if (no < name || unit < 0) return (NODEV); for (i = 0; i < ncdevs; i++) if (strncmp(name, cdevs[i], no - name + 1) == 0) return (makedev(i, unit)); return (NODEV); } #endif