%union { char *str; int val; struct file_list *file; } %token ANY %token AT %token BUS %token COMMA %token CONFIG %token CONFIG_MACHINE %token CONFIG_MACHINE_ARCH %token CONFIG_PLATFORM %token CPU %token DEVICE %token DISABLE %token DRIVE %token DRQ %token EQUALS %token FLAGS %token IDENT %token IOMEM %token IOSIZ %token IRQ %token MAXUSERS %token MINUS %token NEXUS %token OPTIONS %token MAKEOPTIONS %token PORT %token PSEUDO_DEVICE %token SEMICOLON %token TARGET %token TTY %token UNIT %token VECTOR %token ID %token NUMBER %token FPNUMBER %type Save_id %type Opt_value %type Dev %type device_name %{ /* * Copyright (c) 1988, 1993 * The Regents of the University of California. 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. * 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. * * @(#)config.y 8.1 (Berkeley) 6/6/93 * $FreeBSD: src/usr.sbin/config/config.y,v 1.42.2.1 2001/01/23 00:09:32 peter Exp $ * $DragonFly: src/usr.sbin/config/config.y,v 1.15 2008/05/01 09:24:42 swildner Exp $ */ #include #include #include #include #include "config.h" static struct device cur; static struct device *curp = 0; struct device *dtab; char *ident; int yyline; struct file_list *ftab; char errbuf[80]; int maxusers; static int connect(char *, int); static void yyerror(const char *s); %} %% Configuration: Many_specs ; Many_specs: Many_specs Spec | /* lambda */ ; Spec: Device_spec SEMICOLON = { newdev(&cur); } | Config_spec SEMICOLON | SEMICOLON | error SEMICOLON ; Config_spec: CONFIG_PLATFORM Save_id = { if (platformname != NULL) { errx(1, "%d: only one platform directive is allowed", yyline); } platformname = $2; } | CONFIG_MACHINE Save_id = { if (machinename != NULL) { errx(1, "%d: only one machine directive is allowed", yyline); } machinename = $2; } | CONFIG_MACHINE_ARCH Save_id = { if (machinearchname != NULL) { errx(1, "%d: only one machine_arch directive is allowed", yyline); } machinearchname = $2; } | CPU Save_id = { struct cputype *cp; cp = malloc(sizeof(struct cputype)); bzero(cp, sizeof(*cp)); cp->cpu_name = $2; cp->cpu_next = cputype; cputype = cp; } | OPTIONS Opt_list | MAKEOPTIONS Mkopt_list | IDENT ID = { ident = $2; } | System_spec | MAXUSERS NUMBER = { maxusers = $2; }; System_spec: CONFIG System_id System_parameter_list = { errx(1,"line %d: root/dump/swap specifications obsolete", yyline);} | CONFIG System_id ; System_id: Save_id = { struct opt *op; op = malloc(sizeof(struct opt)); bzero(op, sizeof(*op)); op->op_name = strdup("KERNEL"); op->op_ownfile = 0; op->op_next = mkopt; op->op_value = $1; op->op_line = yyline + 1; mkopt = op; }; System_parameter_list: System_parameter_list ID | ID ; device_name: Save_id = { $$ = $1; } | Save_id NUMBER = { char buf[80]; snprintf(buf, sizeof(buf), "%s%d", $1, $2); $$ = strdup(buf); free($1); } | Save_id NUMBER ID = { char buf[80]; snprintf(buf, sizeof(buf), "%s%d%s", $1, $2, $3); $$ = strdup(buf); free($1); } | Save_id NUMBER ID NUMBER = { char buf[80]; snprintf(buf, sizeof(buf), "%s%d%s%d", $1, $2, $3, $4); $$ = strdup(buf); free($1); } | Save_id NUMBER ID NUMBER ID = { char buf[80]; snprintf(buf, sizeof(buf), "%s%d%s%d%s", $1, $2, $3, $4, $5); $$ = strdup(buf); free($1); } ; Opt_list: Opt_list COMMA Option | Option ; Option: Save_id = { struct opt *op; op = malloc(sizeof(struct opt)); bzero(op, sizeof(*op)); op->op_name = $1; op->op_next = opt; op->op_value = 0; /* * op->op_line is 1-based; yyline is 0-based but is now 1 * larger than when `Save_id' was lexed. */ op->op_line = yyline; opt = op; if (strchr(op->op_name, '=') != NULL) errx(1, "line %d: The `=' in options should not be quoted", yyline); } | Save_id EQUALS Opt_value = { struct opt *op; op = malloc(sizeof(struct opt)); bzero(op, sizeof(*op)); op->op_name = $1; op->op_next = opt; op->op_value = $3; op->op_line = yyline + 1; opt = op; } ; Opt_value: ID = { $$ = $1; } | NUMBER = { char buf[80]; snprintf(buf, sizeof(buf), "%d", $1); $$ = strdup(buf); } ; Save_id: ID = { $$ = $1; } ; Mkopt_list: Mkopt_list COMMA Mkoption | Mkoption ; Mkoption: Save_id EQUALS Opt_value = { struct opt *op; op = malloc(sizeof(struct opt)); bzero(op, sizeof(*op)); op->op_name = $1; op->op_ownfile = 0; /* for now */ op->op_next = mkopt; op->op_value = $3; op->op_line = yyline + 1; mkopt = op; } ; Dev: ID = { $$ = $1; } ; Device_spec: DEVICE Dev_spec = { cur.d_type = DEVICE; } | PSEUDO_DEVICE Init_dev Dev = { cur.d_name = $3; cur.d_type = PSEUDO_DEVICE; } | PSEUDO_DEVICE Init_dev Dev NUMBER = { cur.d_name = $3; cur.d_type = PSEUDO_DEVICE; cur.d_count = $4; } ; Dev_spec: Init_dev Dev = { cur.d_name = $2; cur.d_unit = UNKNOWN; } | Init_dev Dev NUMBER Dev_info = { cur.d_name = $2; cur.d_unit = $3; }; Init_dev: /* lambda */ = { init_dev(&cur); }; Dev_info: Con_info Info_list | /* lambda */ ; Con_info: AT Dev NUMBER = { connect($2, $3); cur.d_conn = $2; cur.d_connunit = $3; } | AT NEXUS NUMBER = { cur.d_conn = "nexus"; cur.d_connunit = 0; }; Info_list: Info_list Info | /* lambda */ ; Info: BUS NUMBER /* device scbus1 at ahc0 bus 1 - twin channel */ = { cur.d_bus = $2; } | TARGET NUMBER = { cur.d_target = $2; } | UNIT NUMBER = { cur.d_lun = $2; } | DRIVE NUMBER = { cur.d_drive = $2; } | IRQ NUMBER = { cur.d_irq = $2; } | DRQ NUMBER = { cur.d_drq = $2; } | IOMEM NUMBER = { cur.d_maddr = $2; } | IOSIZ NUMBER = { cur.d_msize = $2; } | PORT device_name = { cur.d_port = $2; } | PORT NUMBER = { cur.d_portn = $2; } | FLAGS NUMBER = { cur.d_flags = $2; } | DISABLE = { cur.d_disabled = 1; } %% static void yyerror(const char *s) { errx(1, "line %d: %s", yyline + 1, s); } /* * add a device to the list of devices */ static void newdev(struct device *dp) { struct device *np, *xp; if (dp->d_unit >= 0) { for (xp = dtab; xp != NULL; xp = xp->d_next) { if ((xp->d_unit == dp->d_unit) && !strcmp(xp->d_name, dp->d_name)) { errx(1, "line %d: already seen device %s%d", yyline, xp->d_name, xp->d_unit); } } } np = malloc(sizeof(*np)); bzero(np, sizeof(*np)); *np = *dp; np->d_next = NULL; if (curp == NULL) dtab = np; else curp->d_next = np; curp = np; } /* * find the pointer to connect to the given device and number. * returns 0 if no such device and prints an error message */ static int connect(char *dev, int num) { struct device *dp; if (num == QUES) { for (dp = dtab; dp != NULL; dp = dp->d_next) if (!strcmp(dp->d_name, dev)) break; if (dp == NULL) { snprintf(errbuf, sizeof(errbuf), "no %s's to wildcard", dev); yyerror(errbuf); return(0); } return(1); } for (dp = dtab; dp != NULL; dp = dp->d_next) { if ((num != dp->d_unit) || strcmp(dev, dp->d_name)) continue; if (dp->d_type != DEVICE) { snprintf(errbuf, sizeof(errbuf), "%s connected to non-device", dev); yyerror(errbuf); return(0); } return(1); } snprintf(errbuf, sizeof(errbuf), "%s %d not defined", dev, num); yyerror(errbuf); return(0); } void init_dev(struct device *dp) { dp->d_name = "OHNO!!!"; dp->d_type = DEVICE; dp->d_conn = 0; dp->d_disabled = 0; dp->d_flags = 0; dp->d_bus = dp->d_lun = dp->d_target = dp->d_drive = dp->d_unit = dp->d_count = UNKNOWN; dp->d_port = NULL; dp->d_portn = -1; dp->d_irq = -1; dp->d_drq = -1; dp->d_maddr = 0; dp->d_msize = 0; }