Home | History | Annotate | Line # | Download | only in etc
      1 #!/usr/bin/awk -
      2 #
      3 #	$NetBSD: MAKEDEV.awk,v 1.29 2020/06/13 19:46:23 thorpej Exp $
      4 #
      5 # Copyright (c) 2003 The NetBSD Foundation, Inc.
      6 # All rights reserved.
      7 #
      8 # This code is derived from software contributed to The NetBSD Foundation
      9 # by Jaromir Dolecek.
     10 #
     11 # Redistribution and use in source and binary forms, with or without
     12 # modification, are permitted provided that the following conditions
     13 # are met:
     14 # 1. Redistributions of source code must retain the above copyright
     15 #    notice, this list of conditions and the following disclaimer.
     16 # 2. Redistributions in binary form must reproduce the above copyright
     17 #    notice, this list of conditions and the following disclaimer in the
     18 #    documentation and/or other materials provided with the distribution.
     19 #
     20 # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     21 # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     22 # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     23 # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     24 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     25 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     26 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     27 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     28 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     29 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     30 # POSSIBILITY OF SUCH DAMAGE.
     31 #
     32 
     33 # Script to generate platform MAKEDEV script from MI template, MD
     34 # MAKEDEV.conf and MD/MI major lists
     35 #
     36 # Uses environment variables MACHINE/MACHINE_ARCH to select
     37 # appropriate files, and NETBSDSRCDIR to get root of source tree.
     38 
     39 BEGIN {
     40 	# top of source tree, used to find major number list in kernel
     41 	# sources
     42 	machine = ENVIRON["MACHINE"]
     43 	maarch = ENVIRON["MACHINE_ARCH"]
     44 	srcdir = ENVIRON["NETBSDSRCDIR"]
     45 	if (!machine || !maarch || !srcdir) {
     46 		print "ERROR: 'MACHINE', 'MACHINE_ARCH' and 'NETBSDSRCDIR' must be set in environment" > "/dev/stderr"
     47 		exit 1
     48 	}
     49 	top = srcdir "/sys/"
     50 	if (system("test -d '" top "'") != 0) {
     51 		print "ERROR: can't find top of kernel source tree ('" top "' not a directory)" > "/dev/stderr"
     52 		exit 1
     53 	}
     54 
     55 
     56 	# file with major definitions
     57 	majors[0] = "conf/majors"
     58 	if ((index(maarch, "arm") != 0 || index(maarch, "aarch64")) && system("test -f '" top "arch/" machine "/conf/majors." machine "'") != 0)
     59 		majors[1] = "arch/arm/conf/majors.arm32";
     60 	else if (machine == "sbmips")
     61 		majors[1] = "arch/evbmips/conf/majors.evbmips";
     62 	else if ((maarch == "powerpc" || maarch == "powerpc64") && system("test -f '" top "arch/" machine "/conf/majors." machine "'") != 0)
     63 		majors[1] = "arch/powerpc/conf/majors.powerpc";
     64 	else
     65 		majors[1] = "arch/" machine "/conf/majors." machine;
     66 	nm = 2;
     67 
     68 	# process all files with majors and fill the chr[] and blk[]
     69 	# arrays, used in template processing
     70 	for (m = 0; m < nm; m++) {
     71 		file = top majors[m]
     72 		if (system("test -f '" file "'") != 0) {
     73 			print "ERROR: can't find majors file '" file "'" > "/dev/stderr"
     74 			exit 1
     75 		}
     76 		while (getline < file) {
     77 			if ($1 == "include") {
     78 				majors[nm++] = substr($2, 2, length($2)-2);
     79 			} else if ($1 == "device-major") {
     80 				if ($3 == "char") {
     81 					chr[$2] = $4
     82 					if ($5 == "block")
     83 						blk[$2] = $6
     84 				} else if ($3 == "block")
     85 					blk[$2] = $4
     86 			}
     87 		}
     88 		close(file)
     89 	}
     90 	CONSOLE_CMAJOR = chr["cons"]
     91 	if (CONSOLE_CMAJOR == "") {
     92 		print "ERROR: no entry for 'cons' in majors file" > "/dev/stderr"
     93 		exit 1
     94 	}
     95 
     96 	# read MD config file for MD device targets
     97 	cfgfile = srcdir "/etc/etc." machine "/MAKEDEV.conf"
     98 	if (system("test -f '" cfgfile "'") != 0) {
     99 		print "ERROR: no platform MAKEDEV.conf - '" cfgfile "' doesn't exist" > "/dev/stderr"
    100 		exit 1
    101 	}
    102 	# skip first two lines
    103 	getline CONFRCSID < cfgfile	# RCS Id
    104 	getline < cfgfile		# blank line
    105 	MDDEV = 0		# MD device targets
    106 	while (getline < cfgfile) {
    107 		#
    108 		# Perform the same blk / chr subsitution that happens below.
    109 		#
    110 		md_deventry = $0
    111 		if (match(md_deventry, /%[a-z0-9]*_(blk|chr)%/)) {
    112 			nam = substr(md_deventry, RSTART + 1, RLENGTH - 6);
    113 			typ = substr(md_deventry, RSTART + RLENGTH - 4, 3);
    114 			dev = ""
    115 			if (typ == "blk") {
    116 				if (nam in blk) {
    117 					dev = blk[nam];
    118 				}
    119 			} else {
    120 				if (nam in chr) {
    121 					dev = chr[nam];
    122 				}
    123 			}
    124 			if (dev != "") {
    125 				parsed = substr(md_deventry, 1, RSTART - 1) dev
    126 				md_deventry = substr(md_deventry, RSTART + RLENGTH)
    127 			}
    128 			md_deventry = parsed md_deventry
    129 		}
    130 		if (MDDEV)
    131 			MDDEV = MDDEV "\n" md_deventry
    132 		else
    133 			MDDEV = md_deventry
    134 	}
    135 	close(cfgfile)
    136 
    137 	# determine number of partitions used by platform
    138 	# there are three variants in tree:
    139 	# 1. MAXPARTITIONS = 8
    140 	# 2. MAXPARTITIONS = 16 with no back compat mapping
    141 	# 3. MAXPARTITIONS = 16 with back compat with old limit of 8
    142 	# currently all archs, which moved from 8->16 use same
    143 	# scheme for mapping disk minors, high minor offset
    144 	# if this changes, the below needs to be adjusted and
    145 	# additional makedisk_p16foo needs to be added
    146 	incdir = machine
    147 	diskpartitions = 0
    148 	diskbackcompat = 0
    149 	while (1) {
    150 		inc = top "arch/" incdir "/include/disklabel.h"
    151 		if (system("test -f '" inc "'") != 0) {
    152 			print "ERROR: can't find kernel include file '" inc "'" > "/dev/stderr"
    153 			exit 1
    154 		}
    155 		incdir = 0
    156 		while (getline < inc) {
    157 			if ($1 == "#define" && $2 == "MAXPARTITIONS")
    158 				diskpartitions = $3
    159 			else if ($1 == "#define" && $2 == "OLDMAXPARTITIONS")
    160 				diskbackcompat = $3
    161 			else if ($1 == "#ifndef" && $2 == "RAW_PART" &&
    162 			    RAWDISK_OFF) {
    163 				# special case to ignore #ifndef RAW_PART
    164 				# sections (e.g. in arm/include/disklabel.h,
    165 				# when it is already set in
    166 				# zaurus/include/disklabel.h)
    167 				while (getline < inc) {
    168 					# skip all lines upto the next #endif
    169 					if ($1 == "#endif")
    170 						break;
    171 				}
    172 			} else if ($1 == "#define" && $2 == "RAW_PART")
    173 				RAWDISK_OFF = $3
    174 			else if ($1 == "#include" && 
    175 				 $2 ~ "<.*/disklabel.h>" &&
    176 				 $2 !~ ".*nbinclude.*")
    177 			{
    178 				# wrapper, switch to the right file
    179 				incdir = substr($2, 2)
    180 				sub("/.*", "", incdir)
    181 				break;
    182 			}
    183 		}
    184 		close(inc)
    185 
    186 		if (diskpartitions)
    187 			break;
    188 
    189 		if (!incdir) {
    190 			print "ERROR: can't determine MAXPARTITIONS from include file '" inc "'" > "/dev/stderr"
    191 			exit 1
    192 		}
    193 	}
    194 	MKDISK = "makedisk_p" diskpartitions	# routine to create disk devs
    195 	DISKMINOROFFSET = diskpartitions
    196 	if (diskbackcompat) {
    197 		MKDISK = MKDISK "high"
    198 		DISKMINOROFFSET = diskbackcompat
    199 	}
    200 	RAWDISK_NAME = sprintf("%c", 97 + RAWDISK_OFF)		# a+offset
    201 
    202 	# read etc/master.passwd for user name->UID mapping
    203 	idfile = srcdir "/etc/master.passwd"
    204 	if (system("test -f '" idfile "'") != 0) {
    205 		print "ERROR: can't find password file '" idfile "'" > "/dev/stderr"
    206 		exit 1
    207 	}
    208 	oldFS=FS
    209 	FS=":"
    210 	while (getline < idfile) {
    211 		uid[$1] = $3
    212 	}
    213 	close(idfile)
    214 	FS=oldFS
    215 
    216 	# read etc/group for group name->GID mapping
    217 	idfile = srcdir "/etc/group"
    218 	if (system("test -f '" idfile "'") != 0) {
    219 		print "ERROR: can't find group file '" idfile "'" > "/dev/stderr"
    220 		exit 1
    221 	}
    222 	oldFS=FS
    223 	FS=":"
    224 	while (getline < idfile) {
    225 		gid[$1] = $3
    226 	}
    227 	close(idfile)
    228 	FS=oldFS
    229 
    230 	# initially no substitutions
    231 	devsubst = 0
    232 	deventry = ""
    233 }
    234 
    235 /%MI_DEVICES_BEGIN%/ {
    236 	devsubst = 1;
    237 	next
    238 }
    239 
    240 /%MI_DEVICES_END%/ {
    241 	devsubst = 0;
    242 	next
    243 }
    244 
    245 # output 'Generated from' lines
    246 /\$[N]etBSD/ {
    247 	print "#"
    248 	print "# Generated from:"
    249 
    250 	# MAKEDEV.awk (this script) RCS Id
    251 	ARCSID = "$NetBSD: MAKEDEV.awk,v 1.29 2020/06/13 19:46:23 thorpej Exp $"
    252 	gsub(/\$/, "", ARCSID)
    253 	print "#	" ARCSID
    254 	
    255 	# MAKEDEV.tmpl RCS Id
    256 	gsub(/\$/, "")
    257 	print $0
    258 
    259 	# MD MAKEDEV.conf RCS Id
    260 	# strip leading hash and insert machine subdirectory name
    261 	gsub(/\$/, "", CONFRCSID)
    262 	sub(/^\# /, "", CONFRCSID)
    263 	sub(/MAKEDEV.conf/, "etc." machine "/MAKEDEV.conf", CONFRCSID)
    264 	print "#	" CONFRCSID
    265 
    266 	next # don't print the RCS Id line again
    267 }
    268 
    269 # filter the 'PLEASE RUN ...' paragraph
    270 /^\#   PLEASE RUN/, /^\#\#\#\#\#\#/ {
    271 	next
    272 }
    273  
    274 # filter the device list
    275 /^\# Tapes/,/^$/ {
    276 	next
    277 }
    278 
    279 # filter the two unneeded makedisk_p* routines, leave only
    280 # the one used
    281 /^makedisk_p8\(\) \{/, /^\}/ {
    282 	if (MKDISK != "makedisk_p8")
    283 		next;
    284 }
    285 /^makedisk_p16\(\) \{/, /^\}/ {
    286 	if (MKDISK != "makedisk_p16")
    287 		next;
    288 }
    289 /^makedisk_p16high\(\) \{/, /^\}/ {
    290 	if (MKDISK != "makedisk_p16high")
    291 		next;
    292 }
    293 
    294 # special cases aside, handle normal line
    295 {
    296 	sub(/^%MD_DEVICES%/, MDDEV)
    297 	sub(/%MKDISK%/, MKDISK)
    298 	sub(/%DISKMINOROFFSET%/, DISKMINOROFFSET)
    299 	sub(/%RAWDISK_OFF%/, RAWDISK_OFF)
    300 	sub(/%RAWDISK_NAME%/, RAWDISK_NAME)
    301 	sub(/%CONSOLE_CMAJOR%/, CONSOLE_CMAJOR)
    302 	parsed = ""
    303 	line = $0
    304 	while (match(line, /%[gu]id_[_a-z]*%/)) {
    305 		typ = substr(line, RSTART + 1, 3);
    306 		nam = substr(line, RSTART + 5, RLENGTH - 6);
    307 		if (typ == "uid") {
    308 			if (!(nam in uid)) {
    309 				print "ERROR unmatched uid in `" $0 "'" > \
    310 				    "/dev/stderr"
    311 				exit 1
    312 			} else
    313 				id = uid[nam];
    314 		} else {
    315 			if (!(nam in gid)) {
    316 				print "ERROR unmatched gid in `" $0 "'" > \
    317 				    "/dev/stderr"
    318 				exit 1
    319 			} else
    320 				id = gid[nam];
    321 		}
    322 		parsed = parsed substr(line, 1, RSTART - 1) id
    323 		line = substr(line, RSTART + RLENGTH)
    324 	}
    325 	$0 = parsed line
    326 
    327 	# if device substitutions are not active, do nothing more
    328 	if (!devsubst) {
    329 		print
    330 		next
    331 	}
    332 }
    333 
    334 # first line of device entry
    335 /^[a-z].*\)$/ {
    336 	if (length(deventry) > 0) {
    337 		# We have a previous entry to print. Replace all known
    338 		# character and block devices. If no unknown character
    339 		# or block device definition remains within the entry,
    340 		# print it to output, otherwise scrap it.
    341 		parsed = ""
    342 		while (match(deventry, /%[a-z0-9]*_(blk|chr)%/)) {
    343 			nam = substr(deventry, RSTART + 1, RLENGTH - 6);
    344 			typ = substr(deventry, RSTART + RLENGTH - 4, 3);
    345 			if (typ == "blk") {
    346 				if (!(nam in blk)) {
    347 					deventry = $0
    348 					next
    349 				} else
    350 					dev = blk[nam];
    351 			} else {
    352 				if (!(nam in chr)) {
    353 					deventry = $0
    354 					next
    355 				} else
    356 					dev = chr[nam];
    357 			}
    358 			parsed = parsed substr(deventry, 1, RSTART - 1) dev
    359 			deventry = substr(deventry, RSTART + RLENGTH)
    360 		}
    361 
    362 		print parsed deventry
    363 	}
    364 	deventry = $0
    365 	next
    366 }
    367 
    368 # template line within device substitution section - just keep appending
    369 # to the current entry
    370 {
    371 	deventry = deventry "\n" $0
    372 }
    373