blob: 7943853ff191fe80b9089d4ce8801bb950f9e5f5 [file] [log] [blame]
Serge Bazanskicc25bdf2018-10-25 14:02:58 +02001#!/usr/bin/env bash
2# Copyright 2009 The Go Authors. All rights reserved.
3# Use of this source code is governed by a BSD-style
4# license that can be found in the LICENSE file.
5
6# Generate Go code listing errors and other #defined constant
7# values (ENAMETOOLONG etc.), by asking the preprocessor
8# about the definitions.
9
10unset LANG
11export LC_ALL=C
12export LC_CTYPE=C
13
14if test -z "$GOARCH" -o -z "$GOOS"; then
15 echo 1>&2 "GOARCH or GOOS not defined in environment"
16 exit 1
17fi
18
19# Check that we are using the new build system if we should
20if [[ "$GOOS" = "linux" ]] && [[ "$GOARCH" != "sparc64" ]]; then
21 if [[ "$GOLANG_SYS_BUILD" != "docker" ]]; then
22 echo 1>&2 "In the new build system, mkerrors should not be called directly."
23 echo 1>&2 "See README.md"
24 exit 1
25 fi
26fi
27
28if [[ "$GOOS" = "aix" ]]; then
29 CC=${CC:-gcc}
30else
31 CC=${CC:-cc}
32fi
33
34if [[ "$GOOS" = "solaris" ]]; then
35 # Assumes GNU versions of utilities in PATH.
36 export PATH=/usr/gnu/bin:$PATH
37fi
38
39uname=$(uname)
40
41includes_AIX='
42#include <net/if.h>
43#include <net/netopt.h>
44#include <netinet/ip_mroute.h>
45#include <sys/protosw.h>
46#include <sys/stropts.h>
47#include <sys/mman.h>
48#include <sys/poll.h>
49#include <sys/termio.h>
50#include <termios.h>
51#include <fcntl.h>
52
53#define AF_LOCAL AF_UNIX
54'
55
56includes_Darwin='
57#define _DARWIN_C_SOURCE
58#define KERNEL
59#define _DARWIN_USE_64_BIT_INODE
60#include <stdint.h>
61#include <sys/attr.h>
62#include <sys/types.h>
63#include <sys/event.h>
64#include <sys/ptrace.h>
65#include <sys/socket.h>
66#include <sys/sockio.h>
67#include <sys/sysctl.h>
68#include <sys/mman.h>
69#include <sys/mount.h>
70#include <sys/utsname.h>
71#include <sys/wait.h>
72#include <sys/xattr.h>
73#include <net/bpf.h>
74#include <net/if.h>
75#include <net/if_types.h>
76#include <net/route.h>
77#include <netinet/in.h>
78#include <netinet/ip.h>
79#include <termios.h>
80'
81
82includes_DragonFly='
83#include <sys/types.h>
84#include <sys/event.h>
85#include <sys/socket.h>
86#include <sys/sockio.h>
87#include <sys/stat.h>
88#include <sys/sysctl.h>
89#include <sys/mman.h>
90#include <sys/mount.h>
91#include <sys/wait.h>
92#include <sys/ioctl.h>
93#include <net/bpf.h>
94#include <net/if.h>
95#include <net/if_types.h>
96#include <net/route.h>
97#include <netinet/in.h>
98#include <termios.h>
99#include <netinet/ip.h>
100#include <net/ip_mroute/ip_mroute.h>
101'
102
103includes_FreeBSD='
104#include <sys/capability.h>
105#include <sys/param.h>
106#include <sys/types.h>
107#include <sys/event.h>
108#include <sys/socket.h>
109#include <sys/sockio.h>
110#include <sys/stat.h>
111#include <sys/sysctl.h>
112#include <sys/mman.h>
113#include <sys/mount.h>
114#include <sys/wait.h>
115#include <sys/ioctl.h>
116#include <net/bpf.h>
117#include <net/if.h>
118#include <net/if_types.h>
119#include <net/route.h>
120#include <netinet/in.h>
121#include <termios.h>
122#include <netinet/ip.h>
123#include <netinet/ip_mroute.h>
124#include <sys/extattr.h>
125
126#if __FreeBSD__ >= 10
127#define IFT_CARP 0xf8 // IFT_CARP is deprecated in FreeBSD 10
128#undef SIOCAIFADDR
129#define SIOCAIFADDR _IOW(105, 26, struct oifaliasreq) // ifaliasreq contains if_data
130#undef SIOCSIFPHYADDR
131#define SIOCSIFPHYADDR _IOW(105, 70, struct oifaliasreq) // ifaliasreq contains if_data
132#endif
133'
134
135includes_Linux='
136#define _LARGEFILE_SOURCE
137#define _LARGEFILE64_SOURCE
138#ifndef __LP64__
139#define _FILE_OFFSET_BITS 64
140#endif
141#define _GNU_SOURCE
142
143// <sys/ioctl.h> is broken on powerpc64, as it fails to include definitions of
144// these structures. We just include them copied from <bits/termios.h>.
145#if defined(__powerpc__)
146struct sgttyb {
147 char sg_ispeed;
148 char sg_ospeed;
149 char sg_erase;
150 char sg_kill;
151 short sg_flags;
152};
153
154struct tchars {
155 char t_intrc;
156 char t_quitc;
157 char t_startc;
158 char t_stopc;
159 char t_eofc;
160 char t_brkc;
161};
162
163struct ltchars {
164 char t_suspc;
165 char t_dsuspc;
166 char t_rprntc;
167 char t_flushc;
168 char t_werasc;
169 char t_lnextc;
170};
171#endif
172
173#include <bits/sockaddr.h>
174#include <sys/epoll.h>
175#include <sys/eventfd.h>
176#include <sys/inotify.h>
177#include <sys/ioctl.h>
178#include <sys/mman.h>
179#include <sys/mount.h>
180#include <sys/prctl.h>
181#include <sys/stat.h>
182#include <sys/types.h>
183#include <sys/time.h>
184#include <sys/socket.h>
185#include <sys/xattr.h>
186#include <linux/if.h>
187#include <linux/if_alg.h>
188#include <linux/if_arp.h>
189#include <linux/if_ether.h>
190#include <linux/if_tun.h>
191#include <linux/if_packet.h>
192#include <linux/if_addr.h>
193#include <linux/falloc.h>
194#include <linux/filter.h>
195#include <linux/fs.h>
196#include <linux/kexec.h>
197#include <linux/keyctl.h>
198#include <linux/magic.h>
199#include <linux/memfd.h>
200#include <linux/netfilter/nfnetlink.h>
201#include <linux/netlink.h>
202#include <linux/net_namespace.h>
203#include <linux/perf_event.h>
204#include <linux/random.h>
205#include <linux/reboot.h>
206#include <linux/rtnetlink.h>
207#include <linux/ptrace.h>
208#include <linux/sched.h>
209#include <linux/seccomp.h>
210#include <linux/sockios.h>
211#include <linux/wait.h>
212#include <linux/icmpv6.h>
213#include <linux/serial.h>
214#include <linux/can.h>
215#include <linux/vm_sockets.h>
216#include <linux/taskstats.h>
217#include <linux/genetlink.h>
218#include <linux/watchdog.h>
219#include <linux/hdreg.h>
220#include <linux/rtc.h>
221#include <linux/if_xdp.h>
222#include <mtd/ubi-user.h>
223#include <net/route.h>
224#include <asm/termbits.h>
225
226#ifndef MSG_FASTOPEN
227#define MSG_FASTOPEN 0x20000000
228#endif
229
230#ifndef PTRACE_GETREGS
231#define PTRACE_GETREGS 0xc
232#endif
233
234#ifndef PTRACE_SETREGS
235#define PTRACE_SETREGS 0xd
236#endif
237
238#ifndef SOL_NETLINK
239#define SOL_NETLINK 270
240#endif
241
242#ifdef SOL_BLUETOOTH
243// SPARC includes this in /usr/include/sparc64-linux-gnu/bits/socket.h
244// but it is already in bluetooth_linux.go
245#undef SOL_BLUETOOTH
246#endif
247
248// Certain constants are missing from the fs/crypto UAPI
249#define FS_KEY_DESC_PREFIX "fscrypt:"
250#define FS_KEY_DESC_PREFIX_SIZE 8
251#define FS_MAX_KEY_SIZE 64
252
253// XDP socket constants do not appear to be picked up otherwise.
254// Copied from samples/bpf/xdpsock_user.c.
255#ifndef SOL_XDP
256#define SOL_XDP 283
257#endif
258
259#ifndef AF_XDP
260#define AF_XDP 44
261#endif
262'
263
264includes_NetBSD='
265#include <sys/types.h>
266#include <sys/param.h>
267#include <sys/event.h>
268#include <sys/extattr.h>
269#include <sys/mman.h>
270#include <sys/mount.h>
271#include <sys/socket.h>
272#include <sys/sockio.h>
273#include <sys/sysctl.h>
274#include <sys/termios.h>
275#include <sys/ttycom.h>
276#include <sys/wait.h>
277#include <net/bpf.h>
278#include <net/if.h>
279#include <net/if_types.h>
280#include <net/route.h>
281#include <netinet/in.h>
282#include <netinet/in_systm.h>
283#include <netinet/ip.h>
284#include <netinet/ip_mroute.h>
285#include <netinet/if_ether.h>
286
287// Needed since <sys/param.h> refers to it...
288#define schedppq 1
289'
290
291includes_OpenBSD='
292#include <sys/types.h>
293#include <sys/param.h>
294#include <sys/event.h>
295#include <sys/mman.h>
296#include <sys/mount.h>
297#include <sys/socket.h>
298#include <sys/sockio.h>
299#include <sys/stat.h>
300#include <sys/sysctl.h>
301#include <sys/termios.h>
302#include <sys/ttycom.h>
303#include <sys/unistd.h>
304#include <sys/wait.h>
305#include <net/bpf.h>
306#include <net/if.h>
307#include <net/if_types.h>
308#include <net/if_var.h>
309#include <net/route.h>
310#include <netinet/in.h>
311#include <netinet/in_systm.h>
312#include <netinet/ip.h>
313#include <netinet/ip_mroute.h>
314#include <netinet/if_ether.h>
315#include <net/if_bridge.h>
316
317// We keep some constants not supported in OpenBSD 5.5 and beyond for
318// the promise of compatibility.
319#define EMUL_ENABLED 0x1
320#define EMUL_NATIVE 0x2
321#define IPV6_FAITH 0x1d
322#define IPV6_OPTIONS 0x1
323#define IPV6_RTHDR_STRICT 0x1
324#define IPV6_SOCKOPT_RESERVED1 0x3
325#define SIOCGIFGENERIC 0xc020693a
326#define SIOCSIFGENERIC 0x80206939
327#define WALTSIG 0x4
328'
329
330includes_SunOS='
331#include <limits.h>
332#include <sys/types.h>
333#include <sys/socket.h>
334#include <sys/sockio.h>
335#include <sys/stat.h>
336#include <sys/mman.h>
337#include <sys/wait.h>
338#include <sys/ioctl.h>
339#include <sys/mkdev.h>
340#include <net/bpf.h>
341#include <net/if.h>
342#include <net/if_arp.h>
343#include <net/if_types.h>
344#include <net/route.h>
345#include <netinet/in.h>
346#include <termios.h>
347#include <netinet/ip.h>
348#include <netinet/ip_mroute.h>
349'
350
351
352includes='
353#include <sys/types.h>
354#include <sys/file.h>
355#include <fcntl.h>
356#include <dirent.h>
357#include <sys/socket.h>
358#include <netinet/in.h>
359#include <netinet/ip.h>
360#include <netinet/ip6.h>
361#include <netinet/tcp.h>
362#include <errno.h>
363#include <sys/signal.h>
364#include <signal.h>
365#include <sys/resource.h>
366#include <time.h>
367'
368ccflags="$@"
369
370# Write go tool cgo -godefs input.
371(
372 echo package unix
373 echo
374 echo '/*'
375 indirect="includes_$(uname)"
376 echo "${!indirect} $includes"
377 echo '*/'
378 echo 'import "C"'
379 echo 'import "syscall"'
380 echo
381 echo 'const ('
382
383 # The gcc command line prints all the #defines
384 # it encounters while processing the input
385 echo "${!indirect} $includes" | $CC -x c - -E -dM $ccflags |
386 awk '
387 $1 != "#define" || $2 ~ /\(/ || $3 == "" {next}
388
389 $2 ~ /^E([ABCD]X|[BIS]P|[SD]I|S|FL)$/ {next} # 386 registers
390 $2 ~ /^(SIGEV_|SIGSTKSZ|SIGRT(MIN|MAX))/ {next}
391 $2 ~ /^(SCM_SRCRT)$/ {next}
392 $2 ~ /^(MAP_FAILED)$/ {next}
393 $2 ~ /^ELF_.*$/ {next}# <asm/elf.h> contains ELF_ARCH, etc.
394
395 $2 ~ /^EXTATTR_NAMESPACE_NAMES/ ||
396 $2 ~ /^EXTATTR_NAMESPACE_[A-Z]+_STRING/ {next}
397
398 $2 !~ /^ECCAPBITS/ &&
399 $2 !~ /^ETH_/ &&
400 $2 !~ /^EPROC_/ &&
401 $2 !~ /^EQUIV_/ &&
402 $2 !~ /^EXPR_/ &&
403 $2 ~ /^E[A-Z0-9_]+$/ ||
404 $2 ~ /^B[0-9_]+$/ ||
405 $2 ~ /^(OLD|NEW)DEV$/ ||
406 $2 == "BOTHER" ||
407 $2 ~ /^CI?BAUD(EX)?$/ ||
408 $2 == "IBSHIFT" ||
409 $2 ~ /^V[A-Z0-9]+$/ ||
410 $2 ~ /^CS[A-Z0-9]/ ||
411 $2 ~ /^I(SIG|CANON|CRNL|UCLC|EXTEN|MAXBEL|STRIP|UTF8)$/ ||
412 $2 ~ /^IGN/ ||
413 $2 ~ /^IX(ON|ANY|OFF)$/ ||
414 $2 ~ /^IN(LCR|PCK)$/ ||
415 $2 !~ "X86_CR3_PCID_NOFLUSH" &&
416 $2 ~ /(^FLU?SH)|(FLU?SH$)/ ||
417 $2 ~ /^C(LOCAL|READ|MSPAR|RTSCTS)$/ ||
418 $2 == "BRKINT" ||
419 $2 == "HUPCL" ||
420 $2 == "PENDIN" ||
421 $2 == "TOSTOP" ||
422 $2 == "XCASE" ||
423 $2 == "ALTWERASE" ||
424 $2 == "NOKERNINFO" ||
425 $2 ~ /^PAR/ ||
426 $2 ~ /^SIG[^_]/ ||
427 $2 ~ /^O[CNPFPL][A-Z]+[^_][A-Z]+$/ ||
428 $2 ~ /^(NL|CR|TAB|BS|VT|FF)DLY$/ ||
429 $2 ~ /^(NL|CR|TAB|BS|VT|FF)[0-9]$/ ||
430 $2 ~ /^O?XTABS$/ ||
431 $2 ~ /^TC[IO](ON|OFF)$/ ||
432 $2 ~ /^IN_/ ||
433 $2 ~ /^LOCK_(SH|EX|NB|UN)$/ ||
434 $2 ~ /^(AF|SOCK|SO|SOL|IPPROTO|IP|IPV6|ICMP6|TCP|EVFILT|NOTE|EV|SHUT|PROT|MAP|MFD|T?PACKET|MSG|SCM|MCL|DT|MADV|PR)_/ ||
435 $2 ~ /^TP_STATUS_/ ||
436 $2 ~ /^FALLOC_/ ||
437 $2 == "ICMPV6_FILTER" ||
438 $2 == "SOMAXCONN" ||
439 $2 == "NAME_MAX" ||
440 $2 == "IFNAMSIZ" ||
441 $2 ~ /^CTL_(HW|KERN|MAXNAME|NET|QUERY)$/ ||
442 $2 ~ /^KERN_(HOSTNAME|OS(RELEASE|TYPE)|VERSION)$/ ||
443 $2 ~ /^HW_MACHINE$/ ||
444 $2 ~ /^SYSCTL_VERS/ ||
445 $2 !~ "MNT_BITS" &&
446 $2 ~ /^(MS|MNT|UMOUNT)_/ ||
447 $2 ~ /^TUN(SET|GET|ATTACH|DETACH)/ ||
448 $2 ~ /^(O|F|E?FD|NAME|S|PTRACE|PT)_/ ||
449 $2 ~ /^KEXEC_/ ||
450 $2 ~ /^LINUX_REBOOT_CMD_/ ||
451 $2 ~ /^LINUX_REBOOT_MAGIC[12]$/ ||
452 $2 !~ "NLA_TYPE_MASK" &&
453 $2 ~ /^(NETLINK|NLM|NLMSG|NLA|IFA|IFAN|RT|RTC|RTCF|RTN|RTPROT|RTNH|ARPHRD|ETH_P|NETNSA)_/ ||
454 $2 ~ /^SIOC/ ||
455 $2 ~ /^TIOC/ ||
456 $2 ~ /^TCGET/ ||
457 $2 ~ /^TCSET/ ||
458 $2 ~ /^TC(FLSH|SBRKP?|XONC)$/ ||
459 $2 !~ "RTF_BITS" &&
460 $2 ~ /^(IFF|IFT|NET_RT|RTM|RTF|RTV|RTA|RTAX)_/ ||
461 $2 ~ /^BIOC/ ||
462 $2 ~ /^RUSAGE_(SELF|CHILDREN|THREAD)/ ||
463 $2 ~ /^RLIMIT_(AS|CORE|CPU|DATA|FSIZE|LOCKS|MEMLOCK|MSGQUEUE|NICE|NOFILE|NPROC|RSS|RTPRIO|RTTIME|SIGPENDING|STACK)|RLIM_INFINITY/ ||
464 $2 ~ /^PRIO_(PROCESS|PGRP|USER)/ ||
465 $2 ~ /^CLONE_[A-Z_]+/ ||
466 $2 !~ /^(BPF_TIMEVAL)$/ &&
467 $2 ~ /^(BPF|DLT)_/ ||
468 $2 ~ /^CLOCK_/ ||
469 $2 ~ /^CAN_/ ||
470 $2 ~ /^CAP_/ ||
471 $2 ~ /^ALG_/ ||
472 $2 ~ /^FS_(POLICY_FLAGS|KEY_DESC|ENCRYPTION_MODE|[A-Z0-9_]+_KEY_SIZE|IOC_(GET|SET)_ENCRYPTION)/ ||
473 $2 ~ /^GRND_/ ||
474 $2 ~ /^KEY_(SPEC|REQKEY_DEFL)_/ ||
475 $2 ~ /^KEYCTL_/ ||
476 $2 ~ /^PERF_EVENT_IOC_/ ||
477 $2 ~ /^SECCOMP_MODE_/ ||
478 $2 ~ /^SPLICE_/ ||
479 $2 ~ /^SYNC_FILE_RANGE_/ ||
480 $2 !~ /^AUDIT_RECORD_MAGIC/ &&
481 $2 !~ /IOC_MAGIC/ &&
482 $2 ~ /^[A-Z][A-Z0-9_]+_MAGIC2?$/ ||
483 $2 ~ /^(VM|VMADDR)_/ ||
484 $2 ~ /^IOCTL_VM_SOCKETS_/ ||
485 $2 ~ /^(TASKSTATS|TS)_/ ||
486 $2 ~ /^CGROUPSTATS_/ ||
487 $2 ~ /^GENL_/ ||
488 $2 ~ /^STATX_/ ||
489 $2 ~ /^RENAME/ ||
490 $2 ~ /^UBI_IOC[A-Z]/ ||
491 $2 ~ /^UTIME_/ ||
492 $2 ~ /^XATTR_(CREATE|REPLACE|NO(DEFAULT|FOLLOW|SECURITY)|SHOWCOMPRESSION)/ ||
493 $2 ~ /^ATTR_(BIT_MAP_COUNT|(CMN|VOL|FILE)_)/ ||
494 $2 ~ /^FSOPT_/ ||
495 $2 ~ /^WDIOC_/ ||
496 $2 ~ /^NFN/ ||
497 $2 ~ /^XDP_/ ||
498 $2 ~ /^(HDIO|WIN|SMART)_/ ||
499 $2 !~ "WMESGLEN" &&
500 $2 ~ /^W[A-Z0-9]+$/ ||
501 $2 ~ /^BLK[A-Z]*(GET$|SET$|BUF$|PART$|SIZE)/ {printf("\t%s = C.%s\n", $2, $2)}
502 $2 ~ /^__WCOREFLAG$/ {next}
503 $2 ~ /^__W[A-Z0-9]+$/ {printf("\t%s = C.%s\n", substr($2,3), $2)}
504
505 {next}
506 ' | sort
507
508 echo ')'
509) >_const.go
510
511# Pull out the error names for later.
512errors=$(
513 echo '#include <errno.h>' | $CC -x c - -E -dM $ccflags |
514 awk '$1=="#define" && $2 ~ /^E[A-Z0-9_]+$/ { print $2 }' |
515 sort
516)
517
518# Pull out the signal names for later.
519signals=$(
520 echo '#include <signal.h>' | $CC -x c - -E -dM $ccflags |
521 awk '$1=="#define" && $2 ~ /^SIG[A-Z0-9]+$/ { print $2 }' |
522 egrep -v '(SIGSTKSIZE|SIGSTKSZ|SIGRT|SIGMAX64)' |
523 sort
524)
525
526# Again, writing regexps to a file.
527echo '#include <errno.h>' | $CC -x c - -E -dM $ccflags |
528 awk '$1=="#define" && $2 ~ /^E[A-Z0-9_]+$/ { print "^\t" $2 "[ \t]*=" }' |
529 sort >_error.grep
530echo '#include <signal.h>' | $CC -x c - -E -dM $ccflags |
531 awk '$1=="#define" && $2 ~ /^SIG[A-Z0-9]+$/ { print "^\t" $2 "[ \t]*=" }' |
532 egrep -v '(SIGSTKSIZE|SIGSTKSZ|SIGRT|SIGMAX64)' |
533 sort >_signal.grep
534
535echo '// mkerrors.sh' "$@"
536echo '// Code generated by the command above; see README.md. DO NOT EDIT.'
537echo
538echo "// +build ${GOARCH},${GOOS}"
539echo
540go tool cgo -godefs -- "$@" _const.go >_error.out
541cat _error.out | grep -vf _error.grep | grep -vf _signal.grep
542echo
543echo '// Errors'
544echo 'const ('
545cat _error.out | grep -f _error.grep | sed 's/=\(.*\)/= syscall.Errno(\1)/'
546echo ')'
547
548echo
549echo '// Signals'
550echo 'const ('
551cat _error.out | grep -f _signal.grep | sed 's/=\(.*\)/= syscall.Signal(\1)/'
552echo ')'
553
554# Run C program to print error and syscall strings.
555(
556 echo -E "
557#include <stdio.h>
558#include <stdlib.h>
559#include <errno.h>
560#include <ctype.h>
561#include <string.h>
562#include <signal.h>
563
564#define nelem(x) (sizeof(x)/sizeof((x)[0]))
565
566enum { A = 'A', Z = 'Z', a = 'a', z = 'z' }; // avoid need for single quotes below
567
568struct tuple {
569 int num;
570 const char *name;
571};
572
573struct tuple errors[] = {
574"
575 for i in $errors
576 do
577 echo -E ' {'$i', "'$i'" },'
578 done
579
580 echo -E "
581};
582
583struct tuple signals[] = {
584"
585 for i in $signals
586 do
587 echo -E ' {'$i', "'$i'" },'
588 done
589
590 # Use -E because on some systems bash builtin interprets \n itself.
591 echo -E '
592};
593
594static int
595tuplecmp(const void *a, const void *b)
596{
597 return ((struct tuple *)a)->num - ((struct tuple *)b)->num;
598}
599
600int
601main(void)
602{
603 int i, e;
604 char buf[1024], *p;
605
606 printf("\n\n// Error table\n");
607 printf("var errorList = [...]struct {\n");
608 printf("\tnum syscall.Errno\n");
609 printf("\tname string\n");
610 printf("\tdesc string\n");
611 printf("} {\n");
612 qsort(errors, nelem(errors), sizeof errors[0], tuplecmp);
613 for(i=0; i<nelem(errors); i++) {
614 e = errors[i].num;
615 if(i > 0 && errors[i-1].num == e)
616 continue;
617 strcpy(buf, strerror(e));
618 // lowercase first letter: Bad -> bad, but STREAM -> STREAM.
619 if(A <= buf[0] && buf[0] <= Z && a <= buf[1] && buf[1] <= z)
620 buf[0] += a - A;
621 printf("\t{ %d, \"%s\", \"%s\" },\n", e, errors[i].name, buf);
622 }
623 printf("}\n\n");
624
625 printf("\n\n// Signal table\n");
626 printf("var signalList = [...]struct {\n");
627 printf("\tnum syscall.Signal\n");
628 printf("\tname string\n");
629 printf("\tdesc string\n");
630 printf("} {\n");
631 qsort(signals, nelem(signals), sizeof signals[0], tuplecmp);
632 for(i=0; i<nelem(signals); i++) {
633 e = signals[i].num;
634 if(i > 0 && signals[i-1].num == e)
635 continue;
636 strcpy(buf, strsignal(e));
637 // lowercase first letter: Bad -> bad, but STREAM -> STREAM.
638 if(A <= buf[0] && buf[0] <= Z && a <= buf[1] && buf[1] <= z)
639 buf[0] += a - A;
640 // cut trailing : number.
641 p = strrchr(buf, ":"[0]);
642 if(p)
643 *p = '\0';
644 printf("\t{ %d, \"%s\", \"%s\" },\n", e, signals[i].name, buf);
645 }
646 printf("}\n\n");
647
648 return 0;
649}
650
651'
652) >_errors.c
653
654$CC $ccflags -o _errors _errors.c && $GORUN ./_errors && rm -f _errors.c _errors _const.go _error.grep _signal.grep _error.out