ref: 548ce5467e98c95e545be3fd021da014798a942d
dir: /support/syscall-gen/gencalls.awk/
BEGIN { sys = ARGV[1] split(sys, sp, ":") systype = sp[1] sysvers = sp[2] arch = ARGV[2] print "/*"; print " generated-ish source"; print " stitched for "sys" arch:"arch; print " edit with caution."; print " */"; ARGC=1 if (systype =="freebsd" || systype == "linux") { Typefield = 3; Protofield = 4; renamings["sys_exit"] = "exit" if (systype == "linux") { strippfx="^(linux_|linux_new)" renamings["mmap2"] = "mmap"; renamings["sys_futex"] = "futex"; } } else if (systype == "openbsd") { Typefield = 2; Protofield = 3; strippfx="^sys_" } else if (systype == "netbsd") { Typefield = 2; Protofield = 4; } else { print "unknown system " ARGV[1]; exit 1; } # stop processing args as files. loadtypes() loadspecials() } NF == 0 || $1 ~ /^;/ || $1 ~ "^#" { next } END { printf("pkg sys =\n") for(i = 0; i < ntypes; i++) printf("\t%s\n", types[i]) printf("\n") for(i = 0; i < nnums; i++) printf("\t%s\n", scnums[i]) printf("\n") printf("\t/* start manual overrides { */\n"); for(i = 0; i < nspecialdefs; i++) printf("\t%s\n", specialdefs[i]) printf("\t/* } end manual overrides */\n"); printf("\n") for(i = 0; i < nproto; i++) printf("\t%s\n", scproto[i]) printf(";;\n\n") printf("\t/* start manual overrides { */\n"); for(i = 0; i < nspecialbody; i++) printf("\t%s\n", specialbody[i]) printf("\t/* } end manual overrides */\n"); for(i = 0; i < ncall; i++) printf("%s\n", sccall[i]) } $Typefield == "STD" || (systype == "linux" && $Typefield == "NOPROTO") { parse(); if (length("const Sys") + length(name) < 16) tabs="\t\t\t" else if (length("const Sys") + length(name) < 24) tabs="\t\t" else tabs = "\t" num = sprintf("const Sys%s%s: scno = %d", name, tabs, number); scnums[nnums++] = num if (nocode) next; if (length("const ") + length(name) < 16) tabs="\t\t\t" else if (length("const ") + length(name) < 24) tabs="\t\t" else tabs = "\t" proto = sprintf("const %s%s: (", name, tabs); sep=""; for (i = 0; i < argc; i++) { proto = proto sprintf("%s%s : %s", sep, argname[i], argtype[i]) sep = ", "; } proto = proto sprintf(" -> %s)", ret) scproto[nproto++] = proto call = sprintf("const %s\t= {", name); sep=""; for (i = 0; i < argc; i++) { call = call sprintf("%s%s", sep, argname[i]) sep = ", "; } call = call sprintf("\n\t -> (syscall(Sys%s", name) for (i = 1; i < argc; i++) { call = call sprintf(", a(%s)", argname[i]) } call = call sprintf(") : %s)\n}", ret) sccall[ncall++] = call } function err(was, wanted) { printf("%s: line %d: unexpected %s (expected %s)\n", infile, NR, was, wanted); printf("\tnear %s\n", $0); exit 1; } function myrtype(t) { gsub("[[:space:]]", "", t) if (systype == "linux") gsub("^l_", "", t) if (t == "char") t = "byte"; if (t == "unsigned int") t = "uint"; if (t == "unsigned long") t = "uint64"; if (t == "long") t = "int64"; if (t == "u_char") t = "byte"; if (t == "u_int") t = "uint"; if (t == "u_long") t = "uint64"; if (t == "acl_type_t") t = "acltype"; if (t == "acl_tag_t") t = "acltag"; if (t == "acl_perm_t") t = "aclperm"; if (t == "acl_entry_type_t") t = "aclenttype"; if (t == "mode_t") t = "filemode" # myrddin sizes are signed if (t == "ssize_t") t = "size"; if (t == "__socklen_t") t = "int32"; if (t == "socklen_t") t = "int32"; if (t == "fd_set") t = "fdset"; if (t == "__ucontext") t = "ucontext"; if (t == "cap_rights_t") t = "caprights"; if (t == "stack_t") t = "sigstack"; if (t == "thr_param") t = "thrparam"; if (t == "nstat") t = "statbuf"; if (t == "stat") t = "statbuf"; if (t == "suseconds_t") t = "int64"; if (t == "caddr_t") t = "void#"; # linux stuff if (t == "times_argv") t = "tms"; if (t == "newstat") t = "statbuf"; if (t == "stat64") t = "statbuf"; if (t == "new_utsname") t = "utsname" if (t == "user_cap_header") t = "capuserheader"; if (t == "user_cap_data") t = "capuserdata"; if (t == "epoll_event") t = "epollevt"; if (t == "statfs_buf") t = "statfs"; if (t == "linux_robust_list_head") t = "robust_list_head" gsub("_t$", "", t) gsub("^__", "", t) return t; } function cname(t) { # mostly to work around freebsd's linux syscalls.master # being a bit off in places. if (systype == "linux") gsub("^l_", "", t) if (t == "times_argv") t = "tms"; if (t == "newstat") t = "stat"; if (t == "statfs_buf") t = "statfs"; if (t == "linux_robust_list_head") t = "robust_list_head" return t; } function unkeyword(kw) { if (kw == "type") return "kind"; if (kw == "in") return "_in"; return kw; } function parse() { number = $1; type = $Typefield; argc = 0 nocode = 0 if ($NF == "}") { funcalias=""; argalias=""; rettype="int"; end=NF; } else if ($(NF-1) == "}") { funcalias=$(NF-1); argalias=""; end=NF-1; } else { funcalias=$(NF-2); argalias=$(NF-1); rettype=$NF; end=NF-3; } f = Protofield; # openbsd puts flags after the kind of call if (systype == "openbsd" && $f != "{") f++; if ($f != "{") err($f, "{"); f++ if ($end != "}") err($f, "closing }"); end--; if ($end != ";") err($end, ";"); end--; if ($end != ")") err($end, ")"); end--; ret=myrtype($f); f++; if ($f == "*") { $f = $f"#"; f++; } name=$f; if (strippfx) gsub(strippfx, "", name); if (renamings[name]) name=renamings[name]; if (special[name]) nocode = 1 f++; if ($f != "(") err($f, "("); f++; if (f == end) { if ($f != "void") parserr($f, "argument definition") return } while (f <= end) { argtype[argc]="" oldf="" while (f < end && $(f+1) != ",") { if ($f == "...") f++; if (argtype[argc] != "" && oldf != "*" && $f != "*") argtype[argc] = argtype[argc]" "; if ($f == "*") $f="#" if ($f != "const" && $f != "struct" && $f != "__restrict" && $f != "volatile" && $f != "union") { argtype[argc] = myrtype(argtype[argc]$f); if ($f != "#" && !knowntypes[myrtype($f)] && !wanttype[$f]) { t = cname($f) wanttype[t] = 1 printf("%s\n", t) > "want.txt" } } oldf = $f; f++ } if (argtype[argc] == "") err($f, "argument definition") argname[argc]=unkeyword($f); f += 2; # skip name, and any comma argc++ } } function loadtypes() { path="types+"sys"-"arch".frag"; while ((getline ln < path) > 0) { types[ntypes++] = ln; split(ln, a) if (a[1] == "type") { knowntypes[a[2]] = 1 printf("%s\n", a[2]) > "have.txt" } } path="gentypes+"sys"-"arch".frag"; while ((getline ln < path) > 0) { types[ntypes++] = ln; split(ln, a) if (a[1] == "type") { knowntypes[a[2]] = 1 printf("%s\n", a[2]) > "have.txt" } } } function loadspecials() { path="specials+"sys"-"arch".frag"; while ((getline ln < path) > 0) { while (match(ln, "\\\\[ \t]*$")) { getline ln2 < path ln = ln "\n" ln2 } split(ln, sp) off = 0 if (sp[off+1] == "extern") off++; if (sp[off+1] == "const" || sp[off+1] == "generic") { special[sp[off+2]] = 1 if (sp[off+3] == ":") specialdefs[nspecialdefs++] = ln; else if (sp[3] == "=") specialbody[nspecialbody++] = ln; else err(sp[3], ": or =") } else { specialbody[nspecialbody++] = ln; } } }