Deleted Added
sdiff udiff text old ( 9842 ) new ( 10551 )
full compact
1#!/bin/sh -
2#
3# Copyright (c) 1992, 1993
4# The Regents of the University of California. All rights reserved.
5#
6# Redistribution and use in source and binary forms, with or without
7# modification, are permitted provided that the following conditions
8# are met:
9# 1. Redistributions of source code must retain the above copyright
10# notice, this list of conditions and the following disclaimer.
11# 2. Redistributions in binary form must reproduce the above copyright
12# notice, this list of conditions and the following disclaimer in the
13# documentation and/or other materials provided with the distribution.
14# 3. All advertising materials mentioning features or use of this software
15# must display the following acknowledgement:
16# This product includes software developed by the University of
17# California, Berkeley and its contributors.
18# 4. Neither the name of the University nor the names of its contributors
19# may be used to endorse or promote products derived from this software
20# without specific prior written permission.
21#
22# THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25# ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32# SUCH DAMAGE.
33#
34# @(#)vnode_if.sh 8.1 (Berkeley) 6/10/93
35# $Id: vnode_if.sh,v 1.4 1995/08/01 18:50:40 davidg Exp $
36#
37
38# Script to produce VFS front-end sugar.
39#
40# usage: vnode_if.sh srcfile
41# (where srcfile is currently /sys/kern/vnode_if.src)
42#
43# These awk scripts are not particularly well written, specifically they
44# don't use arrays well and figure out the same information repeatedly.
45# Please rewrite them if you actually understand how to use awk. Note,
46# they use nawk extensions and gawk's toupper.
47
48if [ $# -ne 1 ] ; then
49 echo 'usage: vnode_if.sh srcfile'
50 exit 1
51fi
52
53# Name of the source file.
54SRC=$1
55
56# Names of the created files.
57CFILE=vnode_if.c
58HEADER=vnode_if.h
59
60# Awk program (must support nawk extensions and gawk's "toupper")
61# Use "awk" at Berkeley, "gawk" elsewhere.
62AWK=awk
63
64# Print out header information for vnode_if.h.
65cat << END_OF_LEADING_COMMENT > $HEADER
66/*
67 * This file is produced automatically.
68 * Do not modify anything in here by hand.
69 *
70 * Created from @(#)vnode_if.sh 8.1 (Berkeley) 6/10/93
71 */
72
73extern struct vnodeop_desc vop_default_desc;
74
75#include <vm/vm.h>
76#include <vm/vm_page.h>
77END_OF_LEADING_COMMENT
78
79# Awk script to take vnode_if.src and turn it into vnode_if.h.
80$AWK '
81 NF == 0 || $0 ~ "^#" {
82 next;
83 }
84 {
85 # Get the function name.
86 name = $1;
87 uname = toupper(name);
88
89 # Get the function arguments.
90 for (c1 = 0;; ++c1) {
91 if (getline <= 0)
92 exit
93 if ($0 ~ "^};")
94 break;
95 a[c1] = $0;
96 }
97
98 # Print out the vop_F_args structure.
99 printf("struct %s_args {\n\tstruct vnodeop_desc *a_desc;\n",
100 name);
101 for (c2 = 0; c2 < c1; ++c2) {
102 c3 = split(a[c2], t);
103 printf("\t");
104 if (t[2] ~ "WILLRELE")
105 c4 = 3;
106 else
107 c4 = 2;
108 for (; c4 < c3; ++c4)
109 printf("%s ", t[c4]);
110 beg = match(t[c3], "[^*]");
111 printf("%sa_%s\n",
112 substr(t[c4], 0, beg - 1), substr(t[c4], beg));
113 }
114 printf("};\n");
115
116 # Print out extern declaration.
117 printf("extern struct vnodeop_desc %s_desc;\n", name);
118
119 # Print out inline struct.
120 printf("static inline int %s(", uname);
121 sep = ", ";
122 for (c2 = 0; c2 < c1; ++c2) {
123 if (c2 == c1 - 1)
124 sep = ")\n";
125 c3 = split(a[c2], t);
126 beg = match(t[c3], "[^*]");
127 end = match(t[c3], ";");
128 printf("%s%s", substr(t[c3], beg, end - beg), sep);
129 }
130 for (c2 = 0; c2 < c1; ++c2) {
131 c3 = split(a[c2], t);
132 printf("\t");
133 if (t[2] ~ "WILLRELE")
134 c4 = 3;
135 else
136 c4 = 2;
137 for (; c4 < c3; ++c4)
138 printf("%s ", t[c4]);
139 beg = match(t[c3], "[^*]");
140 printf("%s%s\n",
141 substr(t[c4], 0, beg - 1), substr(t[c4], beg));
142 }
143 printf("{\n\tstruct %s_args a;\n\n", name);
144 printf("\ta.a_desc = VDESC(%s);\n", name);
145 for (c2 = 0; c2 < c1; ++c2) {
146 c3 = split(a[c2], t);
147 printf("\t");
148 beg = match(t[c3], "[^*]");
149 end = match(t[c3], ";");
150 printf("a.a_%s = %s\n",
151 substr(t[c3], beg, end - beg), substr(t[c3], beg));
152 }
153 c1 = split(a[0], t);
154 beg = match(t[c1], "[^*]");
155 end = match(t[c1], ";");
156 printf("\treturn (VCALL(%s, VOFFSET(%s), &a));\n}\n",
157 substr(t[c1], beg, end - beg), name);
158 }' < $SRC >> $HEADER
159
160# Print out header information for vnode_if.c.
161cat << END_OF_LEADING_COMMENT > $CFILE
162/*
163 * This file is produced automatically.
164 * Do not modify anything in here by hand.
165 *
166 * Created from @(#)vnode_if.sh 8.1 (Berkeley) 6/10/93
167 */
168
169#include <sys/param.h>
170#include <sys/mount.h>
171#include <sys/vnode.h>
172#include <vm/vm.h>
173#include <vm/vm_page.h>
174
175struct vnodeop_desc vop_default_desc = {
176 0,
177 "default",
178 0,
179 NULL,
180 VDESC_NO_OFFSET,
181 VDESC_NO_OFFSET,
182 VDESC_NO_OFFSET,
183 VDESC_NO_OFFSET,
184 NULL,
185};
186
187END_OF_LEADING_COMMENT
188
189# Awk script to take vnode_if.src and turn it into vnode_if.c.
190$AWK 'function kill_surrounding_ws (s) {
191 sub (/^[ \t]*/, "", s);
192 sub (/[ \t]*$/, "", s);
193 return s;
194 }
195
196 function read_args() {
197 numargs = 0;
198 while (getline ln) {
199 if (ln ~ /}/) {
200 break;
201 };
202
203 # Delete comments, if any.
204 gsub (/\/\*.*\*\//, "", ln);
205
206 # Delete leading/trailing space.
207 ln = kill_surrounding_ws(ln);
208
209 # Pick off direction.
210 if (1 == sub(/^INOUT[ \t]+/, "", ln))
211 dir = "INOUT";
212 else if (1 == sub(/^IN[ \t]+/, "", ln))
213 dir = "IN";
214 else if (1 == sub(/^OUT[ \t]+/, "", ln))
215 dir = "OUT";
216 else
217 bail("No IN/OUT direction for \"" ln "\".");
218
219 # check for "WILLRELE"
220 if (1 == sub(/^WILLRELE[ \t]+/, "", ln)) {
221 rele = "WILLRELE";
222 } else {
223 rele = "WONTRELE";
224 };
225
226 # kill trailing ;
227 if (1 != sub (/;$/, "", ln)) {
228 bail("Missing end-of-line ; in \"" ln "\".");
229 };
230
231 # pick off variable name
232 if (!(i = match(ln, /[A-Za-z0-9_]+$/))) {
233 bail("Missing var name \"a_foo\" in \"" ln "\".");
234 };
235 arg = substr (ln, i);
236 # Want to <<substr(ln, i) = "";>>, but nawk cannot.
237 # Hack around this.
238 ln = substr(ln, 1, i-1);
239
240 # what is left must be type
241 # (put clean it up some)
242 type = ln;
243 gsub (/[ \t]+/, " ", type); # condense whitespace
244 type = kill_surrounding_ws(type);
245
246 # (boy this was easier in Perl)
247
248 numargs++;
249 dirs[numargs] = dir;
250 reles[numargs] = rele;
251 types[numargs] = type;
252 args[numargs] = arg;
253 };
254 }
255
256 function generate_operation_vp_offsets() {
257 printf ("int %s_vp_offsets[] = {\n", name);
258 # as a side effect, figure out the releflags
259 releflags = "";
260 vpnum = 0;
261 for (i=1; i<=numargs; i++) {
262 if (types[i] == "struct vnode *") {
263 printf ("\tVOPARG_OFFSETOF(struct %s_args,a_%s),\n",
264 name, args[i]);
265 if (reles[i] == "WILLRELE") {
266 releflags = releflags "|VDESC_VP" vpnum "_WILLRELE";
267 };
268 vpnum++;
269 };
270 };
271 sub (/^\|/, "", releflags);
272 print "\tVDESC_NO_OFFSET";
273 print "};";
274 }
275
276 function find_arg_with_type (type) {
277 for (i=1; i<=numargs; i++) {
278 if (types[i] == type) {
279 return "VOPARG_OFFSETOF(struct " name "_args,a_" args[i] ")";
280 };
281 };
282 return "VDESC_NO_OFFSET";
283 }
284
285 function generate_operation_desc() {
286 printf ("struct vnodeop_desc %s_desc = {\n", name);
287 # offset
288 printf ("\t0,\n");
289 # printable name
290 printf ("\t\"%s\",\n", name);
291 # flags
292 vppwillrele = "";
293 for (i=1; i<=numargs; i++) {
294 if (types[i] == "struct vnode **" &&
295 (reles[i] == "WILLRELE")) {
296 vppwillrele = "|VDESC_VPP_WILLRELE";
297 };
298 };
299 if (releflags == "") {
300 printf ("\t0%s,\n", vppwillrele);
301 } else {
302 printf ("\t%s%s,\n", releflags, vppwillrele);
303 };
304 # vp offsets
305 printf ("\t%s_vp_offsets,\n", name);
306 # vpp (if any)
307 printf ("\t%s,\n", find_arg_with_type("struct vnode **"));
308 # cred (if any)
309 printf ("\t%s,\n", find_arg_with_type("struct ucred *"));
310 # proc (if any)
311 printf ("\t%s,\n", find_arg_with_type("struct proc *"));
312 # componentname
313 printf ("\t%s,\n", find_arg_with_type("struct componentname *"));
314 # transport layer information
315 printf ("\tNULL,\n};\n");
316 }
317
318 NF == 0 || $0 ~ "^#" {
319 next;
320 }
321 {
322 # get the function name
323 name = $1;
324
325 # get the function arguments
326 read_args();
327
328 # Print out the vop_F_vp_offsets structure. This all depends
329 # on naming conventions and nothing else.
330 generate_operation_vp_offsets();
331
332 # Print out the vnodeop_desc structure.
333 generate_operation_desc();
334
335 printf "\n";
336
337 }' < $SRC >> $CFILE
338# THINGS THAT DON'T WORK RIGHT YET.
339#
340# Two existing BSD vnodeops (bwrite and strategy) don't take any vnodes as
341# arguments. This means that these operations can't function successfully
342# through a bypass routine.
343#
344# Bwrite and strategy will be replaced when the VM page/buffer cache
345# integration happens.
346#
347# To get around this problem for now we handle these ops as special cases.
348
349cat << END_OF_SPECIAL_CASES >> $HEADER
350#include <sys/buf.h>
351struct vop_strategy_args {
352 struct vnodeop_desc *a_desc;
353 struct buf *a_bp;
354};
355extern struct vnodeop_desc vop_strategy_desc;
356static inline int VOP_STRATEGY(bp)
357 struct buf *bp;
358{
359 struct vop_strategy_args a;
360
361 a.a_desc = VDESC(vop_strategy);
362 a.a_bp = bp;
363 return (VCALL((bp)->b_vp, VOFFSET(vop_strategy), &a));
364}
365
366struct vop_bwrite_args {
367 struct vnodeop_desc *a_desc;
368 struct buf *a_bp;
369};
370extern struct vnodeop_desc vop_bwrite_desc;
371static inline int VOP_BWRITE(bp)
372 struct buf *bp;
373{
374 struct vop_bwrite_args a;
375
376 a.a_desc = VDESC(vop_bwrite);
377 a.a_bp = bp;
378 return (VCALL((bp)->b_vp, VOFFSET(vop_bwrite), &a));
379}
380END_OF_SPECIAL_CASES
381
382cat << END_OF_SPECIAL_CASES >> $CFILE
383int vop_strategy_vp_offsets[] = {
384 VDESC_NO_OFFSET
385};
386struct vnodeop_desc vop_strategy_desc = {
387 0,
388 "vop_strategy",
389 0,
390 vop_strategy_vp_offsets,
391 VDESC_NO_OFFSET,
392 VDESC_NO_OFFSET,
393 VDESC_NO_OFFSET,
394 VDESC_NO_OFFSET,
395 NULL,
396};
397int vop_bwrite_vp_offsets[] = {
398 VDESC_NO_OFFSET
399};
400struct vnodeop_desc vop_bwrite_desc = {
401 0,
402 "vop_bwrite",
403 0,
404 vop_bwrite_vp_offsets,
405 VDESC_NO_OFFSET,
406 VDESC_NO_OFFSET,
407 VDESC_NO_OFFSET,
408 VDESC_NO_OFFSET,
409 NULL,
410};
411END_OF_SPECIAL_CASES
412
413# Add the vfs_op_descs array to the C file.
414$AWK '
415 BEGIN {
416 printf("\nstruct vnodeop_desc *vfs_op_descs[] = {\n");
417 printf("\t&vop_default_desc, /* MUST BE FIRST */\n");
418 printf("\t&vop_strategy_desc, /* XXX: SPECIAL CASE */\n");
419 printf("\t&vop_bwrite_desc, /* XXX: SPECIAL CASE */\n");
420 }
421 END {
422 printf("\tNULL\n};\n");
423 }
424 NF == 0 || $0 ~ "^#" {
425 next;
426 }
427 {
428 # Get the function name.
429 printf("\t&%s_desc,\n", $1);
430
431 # Skip the function arguments.
432 for (;;) {
433 if (getline <= 0)
434 exit
435 if ($0 ~ "^};")
436 break;
437 }
438 }' < $SRC >> $CFILE
439