#!/usr/bin/awk - # # $NetBSD: MAKEDEV.awk,v 1.20 2008/04/30 13:10:49 martin Exp $ # # Copyright (c) 2003 The NetBSD Foundation, Inc. # All rights reserved. # # This code is derived from software contributed to The NetBSD Foundation # by Jaromir Dolecek. # # 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 NETBSD FOUNDATION, INC. 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 FOUNDATION 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. # # Script to generate platform MAKEDEV script from MI template, MD # MAKEDEV.conf and MD/MI major lists # # Uses environment variables MACHINE/MACHINE_ARCH to select # appropriate files, and NETBSDSRCDIR to get root of source tree. BEGIN { # top of source tree, used to find major number list in kernel # sources machine = ENVIRON["MACHINE"] maarch = ENVIRON["MACHINE_ARCH"] srcdir = ENVIRON["NETBSDSRCDIR"] if (!machine || !maarch || !srcdir) { print "ERROR: 'MACHINE', 'MACHINE_ARCH' and 'NETBSDSRCDIR' must be set in environment" > "/dev/stderr" exit 1 } top = srcdir "/sys/" if (system("test -d '" top "'") != 0) { print "ERROR: can't find top of kernel source tree ('" top "' not a directory)" > "/dev/stderr" exit 1 } # file with major definitions majors[0] = "conf/majors" if ((maarch == "arm" || maarch == "armeb") && system("test -f '" top "arch/" machine "/conf/majors." machine "'") != 0) majors[1] = "arch/arm/conf/majors.arm32"; else if (machine == "sbmips") majors[1] = "arch/evbmips/conf/majors.evbmips"; else if ((maarch == "powerpc") && system("test -f '" top "arch/" machine "/conf/majors." machine "'") != 0) majors[1] = "arch/powerpc/conf/majors.powerpc"; else majors[1] = "arch/" machine "/conf/majors." machine; # process all files with majors and fill the chr[] and blk[] # arrays, used in template processing for (m in majors) { file = top majors[m] if (system("test -f '" file "'") != 0) { print "ERROR: can't find majors file '" file "'" > "/dev/stderr" exit 1 } while (getline < file) { if ($1 == "device-major") { if ($3 == "char") { chr[$2] = $4 if ($5 == "block") blk[$2] = $6 } else if ($3 == "block") blk[$2] = $4 } } close(file) } CONSOLE_CMAJOR = chr["cons"] if (CONSOLE_CMAJOR == "") { print "ERROR: no entry for 'cons' in majors file" > "/dev/stderr" exit 1 } # read MD config file for MD device targets cfgfile = srcdir "/etc/etc." machine "/MAKEDEV.conf" if (system("test -f '" cfgfile "'") != 0) { print "ERROR: no platform MAKEDEV.conf - '" cfgfile "' doesn't exist" > "/dev/stderr" exit 1 } # skip first two lines getline CONFRCSID < cfgfile # RCS Id getline < cfgfile # blank line MDDEV = 0 # MD device targets while (getline < cfgfile) { if (MDDEV) MDDEV = MDDEV "\n" $0 else MDDEV = $0 } close(cfgfile) # determine number of partitions used by platform # there are three variants in tree: # 1. MAXPARTITIONS = 8 # 2. MAXPARTITIONS = 16 with no back compat mapping # 3. MAXPARTITIONS = 16 with back compat with old limit of 8 # currently all archs, which moved from 8->16 use same # scheme for mapping disk minors, high minor offset # if this changes, the below needs to be adjusted and # additional makedisk_p16foo needs to be added incdir = machine diskpartitions = 0 diskbackcompat = 0 while (1) { inc = top "arch/" incdir "/include/disklabel.h" if (system("test -f '" inc "'") != 0) { print "ERROR: can't find kernel include file '" inc "'" > "/dev/stderr" exit 1 } incdir = 0 while (getline < inc) { if ($1 == "#define" && $2 == "MAXPARTITIONS") diskpartitions = $3 else if ($1 == "#define" && $2 == "OLDMAXPARTITIONS") diskbackcompat = $3 else if ($1 == "#define" && $2 == "RAW_PART") RAWDISK_OFF = $3 else if ($1 == "#include" && $2 ~ "<.*/disklabel.h>" && $2 !~ ".*nbinclude.*") { # wrapper, switch to the right file incdir = substr($2, 2) sub("/.*", "", incdir) break; } } close(inc) if (diskpartitions) break; if (!incdir) { print "ERROR: can't determine MAXPARTITIONS from include file '" inc "'" > "/dev/stderr" exit 1 } } MKDISK = "makedisk_p" diskpartitions # routine to create disk devs DISKMINOROFFSET = diskpartitions if (diskbackcompat) { MKDISK = MKDISK "high" DISKMINOROFFSET = diskbackcompat } RAWDISK_NAME = sprintf("%c", 97 + RAWDISK_OFF) # a+offset # read etc/master.passwd for user name->UID mapping idfile = srcdir "/etc/master.passwd" if (system("test -f '" idfile "'") != 0) { print "ERROR: can't find password file '" idfile "'" > "/dev/stderr" exit 1 } oldFS=FS FS=":" while (getline < idfile) { uid[$1] = $3 } close(idfile) FS=oldFS # read etc/group for group name->GID mapping idfile = srcdir "/etc/group" if (system("test -f '" idfile "'") != 0) { print "ERROR: can't find group file '" idfile "'" > "/dev/stderr" exit 1 } oldFS=FS FS=":" while (getline < idfile) { gid[$1] = $3 } close(idfile) FS=oldFS # initially no substitutions devsubst = 0 deventry = "" } /%MI_DEVICES_BEGIN%/ { devsubst = 1; next } /%MI_DEVICES_END%/ { devsubst = 0; next } # output 'Generated from' lines /\$[N]etBSD/ { print "#" print "# Generated from:" # MAKEDEV.awk (this script) RCS Id ARCSID = "$NetBSD: MAKEDEV.awk,v 1.20 2008/04/30 13:10:49 martin Exp $" gsub(/\$/, "", ARCSID) print "# " ARCSID # MAKEDEV.tmpl RCS Id gsub(/\$/, "") print $0 # MD MAKEDEV.conf RCS Id # strip leading hash and insert machine subdirectory name gsub(/\$/, "", CONFRCSID) sub(/^\# /, "", CONFRCSID) sub(/MAKEDEV.conf/, "etc." machine "/MAKEDEV.conf", CONFRCSID) print "# " CONFRCSID next # don't print the RCS Id line again } # filter the 'PLEASE RUN ...' paragraph /^\# PLEASE RUN/, /^\#\#\#\#\#\#/ { next } # filter the device list /^\# Tapes/,/^$/ { next } # filter the two unneeded makedisk_p* routines, leave only # the one used /^makedisk_p8\(\) \{/, /^\}/ { if (MKDISK != "makedisk_p8") next; } /^makedisk_p16\(\) \{/, /^\}/ { if (MKDISK != "makedisk_p16") next; } /^makedisk_p16high\(\) \{/, /^\}/ { if (MKDISK != "makedisk_p16high") next; } # special cases aside, handle normal line { sub(/^%MD_DEVICES%/, MDDEV) sub(/%MKDISK%/, MKDISK) sub(/%DISKMINOROFFSET%/, DISKMINOROFFSET) sub(/%RAWDISK_OFF%/, RAWDISK_OFF) sub(/%RAWDISK_NAME%/, RAWDISK_NAME) sub(/%CONSOLE_CMAJOR%/, CONSOLE_CMAJOR) parsed = "" line = $0 while (match(line, /%[gu]id_[a-z]*%/)) { typ = substr(line, RSTART + 1, 3); nam = substr(line, RSTART + 5, RLENGTH - 6); if (typ == "uid") { if (!(nam in uid)) { print "ERROR unmatched uid in `" $0 "'" > \ "/dev/stderr" exit 1 } else id = uid[nam]; } else { if (!(nam in gid)) { print "ERROR unmatched gid in `" $0 "'" > \ "/dev/stderr" exit 1 } else id = gid[nam]; } parsed = parsed substr(line, 1, RSTART - 1) id line = substr(line, RSTART + RLENGTH) } $0 = parsed line # if device substitutions are not active, do nothing more if (!devsubst) { print next } } # first line of device entry /^[a-z].*\)$/ { if (length(deventry) > 0) { # We have a previous entry to print. Replace all known # character and block devices. If no unknown character # or block device definition remains within the entry, # print it to output, otherwise scrap it. parsed = "" while (match(deventry, /%[a-z]*_(blk|chr)%/)) { nam = substr(deventry, RSTART + 1, RLENGTH - 6); typ = substr(deventry, RSTART + RLENGTH - 4, 3); if (typ == "blk") { if (!(nam in blk)) { deventry = $0 next } else dev = blk[nam]; } else { if (!(nam in chr)) { deventry = $0 next } else dev = chr[nam]; } parsed = parsed substr(deventry, 1, RSTART - 1) dev deventry = substr(deventry, RSTART + RLENGTH) } print parsed deventry } deventry = $0 next } # template line within device substitution section - just keep appending # to the current entry { deventry = deventry "\n" $0 }