1 #pragma lib "libc.a" 2 #pragma src "/sys/src/libc" 3 4 #define nelem(x) (sizeof(x)/sizeof((x)[0])) 5 #define offsetof(s, m) (ulong)(&(((s*)0)->m)) 6 #define assert(x) if(x){}else _assert("x") 7 8 /* 9 * mem routines 10 */ 11 extern void* memccpy(void*, void*, int, ulong); 12 extern void* memset(void*, int, ulong); 13 extern int memcmp(void*, void*, ulong); 14 extern void* memcpy(void*, void*, ulong); 15 extern void* memmove(void*, void*, ulong); 16 extern void* memchr(void*, int, ulong); 17 18 /* 19 * string routines 20 */ 21 extern char* strcat(char*, char*); 22 extern char* strchr(char*, int); 23 extern int strcmp(char*, char*); 24 extern char* strcpy(char*, char*); 25 extern char* strecpy(char*, char*, char*); 26 extern char* strdup(char*); 27 extern char* strncat(char*, char*, long); 28 extern char* strncpy(char*, char*, long); 29 extern int strncmp(char*, char*, long); 30 extern char* strpbrk(char*, char*); 31 extern char* strrchr(char*, int); 32 extern char* strtok(char*, char*); 33 extern long strlen(char*); 34 extern long strspn(char*, char*); 35 extern long strcspn(char*, char*); 36 extern char* strstr(char*, char*); 37 extern int cistrncmp(char*, char*, int); 38 extern int cistrcmp(char*, char*); 39 extern char* cistrstr(char*, char*); 40 extern int tokenize(char*, char**, int); 41 42 enum 43 { 44 UTFmax = 3, /* maximum bytes per rune */ 45 Runesync = 0x80, /* cannot represent part of a UTF sequence (<) */ 46 Runeself = 0x80, /* rune and UTF sequences are the same (<) */ 47 Runeerror = 0xFFFD, /* decoding error in UTF */ 48 }; 49 50 /* 51 * rune routines 52 */ 53 extern int runetochar(char*, Rune*); 54 extern int chartorune(Rune*, char*); 55 extern int runelen(long); 56 extern int runenlen(Rune*, int); 57 extern int fullrune(char*, int); 58 extern int utflen(char*); 59 extern int utfnlen(char*, long); 60 extern char* utfrune(char*, long); 61 extern char* utfrrune(char*, long); 62 extern char* utfutf(char*, char*); 63 extern char* utfecpy(char*, char*, char*); 64 65 extern Rune* runestrcat(Rune*, Rune*); 66 extern Rune* runestrchr(Rune*, Rune); 67 extern int runestrcmp(Rune*, Rune*); 68 extern Rune* runestrcpy(Rune*, Rune*); 69 extern Rune* runestrncpy(Rune*, Rune*, long); 70 extern Rune* runestrecpy(Rune*, Rune*, Rune*); 71 extern Rune* runestrdup(Rune*); 72 extern Rune* runestrncat(Rune*, Rune*, long); 73 extern int runestrncmp(Rune*, Rune*, long); 74 extern Rune* runestrrchr(Rune*, Rune); 75 extern long runestrlen(Rune*); 76 extern Rune* runestrstr(Rune*, Rune*); 77 78 extern Rune tolowerrune(Rune); 79 extern Rune totitlerune(Rune); 80 extern Rune toupperrune(Rune); 81 extern int isalpharune(Rune); 82 extern int islowerrune(Rune); 83 extern int isspacerune(Rune); 84 extern int istitlerune(Rune); 85 extern int isupperrune(Rune); 86 87 /* 88 * malloc 89 */ 90 extern void* malloc(ulong); 91 extern void* mallocz(ulong, int); 92 extern void free(void*); 93 extern ulong msize(void*); 94 extern void* mallocalign(ulong, ulong, long, ulong); 95 extern void* calloc(ulong, ulong); 96 extern void* realloc(void*, ulong); 97 extern void setmalloctag(void*, ulong); 98 extern void setrealloctag(void*, ulong); 99 extern ulong getmalloctag(void*); 100 extern ulong getrealloctag(void*); 101 extern void* malloctopoolblock(void*); 102 103 /* 104 * print routines 105 */ 106 typedef struct Fmt Fmt; 107 struct Fmt{ 108 uchar runes; /* output buffer is runes or chars? */ 109 void *start; /* of buffer */ 110 void *to; /* current place in the buffer */ 111 void *stop; /* end of the buffer; overwritten if flush fails */ 112 int (*flush)(Fmt *); /* called when to == stop */ 113 void *farg; /* to make flush a closure */ 114 int nfmt; /* num chars formatted so far */ 115 va_list args; /* args passed to dofmt */ 116 int r; /* % format Rune */ 117 int width; 118 int prec; 119 ulong flags; 120 }; 121 122 enum{ 123 FmtWidth = 1, 124 FmtLeft = FmtWidth << 1, 125 FmtPrec = FmtLeft << 1, 126 FmtSharp = FmtPrec << 1, 127 FmtSpace = FmtSharp << 1, 128 FmtSign = FmtSpace << 1, 129 FmtZero = FmtSign << 1, 130 FmtUnsigned = FmtZero << 1, 131 FmtShort = FmtUnsigned << 1, 132 FmtLong = FmtShort << 1, 133 FmtVLong = FmtLong << 1, 134 FmtComma = FmtVLong << 1, 135 FmtByte = FmtComma << 1, 136 137 FmtFlag = FmtByte << 1 138 }; 139 140 extern int print(char*, ...); 141 extern char* seprint(char*, char*, char*, ...); 142 extern char* vseprint(char*, char*, char*, va_list); 143 extern int snprint(char*, int, char*, ...); 144 extern int vsnprint(char*, int, char*, va_list); 145 extern char* smprint(char*, ...); 146 extern char* vsmprint(char*, va_list); 147 extern int sprint(char*, char*, ...); 148 extern int fprint(int, char*, ...); 149 extern int vfprint(int, char*, va_list); 150 151 extern int runesprint(Rune*, char*, ...); 152 extern int runesnprint(Rune*, int, char*, ...); 153 extern int runevsnprint(Rune*, int, char*, va_list); 154 extern Rune* runeseprint(Rune*, Rune*, char*, ...); 155 extern Rune* runevseprint(Rune*, Rune*, char*, va_list); 156 extern Rune* runesmprint(char*, ...); 157 extern Rune* runevsmprint(char*, va_list); 158 159 extern int fmtfdinit(Fmt*, int, char*, int); 160 extern int fmtfdflush(Fmt*); 161 extern int fmtstrinit(Fmt*); 162 extern char* fmtstrflush(Fmt*); 163 extern int runefmtstrinit(Fmt*); 164 extern Rune* runefmtstrflush(Fmt*); 165 166 #pragma varargck argpos fmtprint 2 167 #pragma varargck argpos fprint 2 168 #pragma varargck argpos print 1 169 #pragma varargck argpos runeseprint 3 170 #pragma varargck argpos runesmprint 1 171 #pragma varargck argpos runesnprint 3 172 #pragma varargck argpos runesprint 2 173 #pragma varargck argpos seprint 3 174 #pragma varargck argpos smprint 1 175 #pragma varargck argpos snprint 3 176 #pragma varargck argpos sprint 2 177 178 #pragma varargck type "lld" vlong 179 #pragma varargck type "llx" vlong 180 #pragma varargck type "lld" uvlong 181 #pragma varargck type "llx" uvlong 182 #pragma varargck type "ld" long 183 #pragma varargck type "lx" long 184 #pragma varargck type "ld" ulong 185 #pragma varargck type "lx" ulong 186 #pragma varargck type "d" int 187 #pragma varargck type "x" int 188 #pragma varargck type "c" int 189 #pragma varargck type "C" int 190 #pragma varargck type "d" uint 191 #pragma varargck type "x" uint 192 #pragma varargck type "c" uint 193 #pragma varargck type "C" uint 194 #pragma varargck type "f" double 195 #pragma varargck type "e" double 196 #pragma varargck type "g" double 197 #pragma varargck type "s" char* 198 #pragma varargck type "q" char* 199 #pragma varargck type "S" Rune* 200 #pragma varargck type "Q" Rune* 201 #pragma varargck type "r" void 202 #pragma varargck type "%" void 203 #pragma varargck type "n" int* 204 #pragma varargck type "p" uintptr 205 #pragma varargck type "p" void* 206 #pragma varargck flag ',' 207 #pragma varargck type "<" void* 208 #pragma varargck type "[" void* 209 #pragma varargck type "H" void* 210 #pragma varargck type "lH" void* 211 212 extern int fmtinstall(int, int (*)(Fmt*)); 213 extern int dofmt(Fmt*, char*); 214 extern int dorfmt(Fmt*, Rune*); 215 extern int fmtprint(Fmt*, char*, ...); 216 extern int fmtvprint(Fmt*, char*, va_list); 217 extern int fmtrune(Fmt*, int); 218 extern int fmtstrcpy(Fmt*, char*); 219 extern int fmtrunestrcpy(Fmt*, Rune*); 220 /* 221 * error string for %r 222 * supplied on per os basis, not part of fmt library 223 */ 224 extern int errfmt(Fmt *f); 225 226 /* 227 * quoted strings 228 */ 229 extern char *unquotestrdup(char*); 230 extern Rune *unquoterunestrdup(Rune*); 231 extern char *quotestrdup(char*); 232 extern Rune *quoterunestrdup(Rune*); 233 extern int quotestrfmt(Fmt*); 234 extern int quoterunestrfmt(Fmt*); 235 extern void quotefmtinstall(void); 236 extern int (*doquote)(int); 237 extern int needsrcquote(int); 238 239 /* 240 * random number 241 */ 242 extern void srand(long); 243 extern int rand(void); 244 extern int nrand(int); 245 extern long lrand(void); 246 extern long lnrand(long); 247 extern double frand(void); 248 extern ulong truerand(void); /* uses /dev/random */ 249 extern ulong ntruerand(ulong); /* uses /dev/random */ 250 251 /* 252 * math 253 */ 254 extern ulong getfcr(void); 255 extern void setfsr(ulong); 256 extern ulong getfsr(void); 257 extern void setfcr(ulong); 258 extern double NaN(void); 259 extern double Inf(int); 260 extern int isNaN(double); 261 extern int isInf(double, int); 262 extern ulong umuldiv(ulong, ulong, ulong); 263 extern long muldiv(long, long, long); 264 265 extern double pow(double, double); 266 extern double atan2(double, double); 267 extern double fabs(double); 268 extern double atan(double); 269 extern double log(double); 270 extern double log10(double); 271 extern double exp(double); 272 extern double floor(double); 273 extern double ceil(double); 274 extern double hypot(double, double); 275 extern double sin(double); 276 extern double cos(double); 277 extern double tan(double); 278 extern double asin(double); 279 extern double acos(double); 280 extern double sinh(double); 281 extern double cosh(double); 282 extern double tanh(double); 283 extern double sqrt(double); 284 extern double fmod(double, double); 285 286 #define HUGE 3.4028234e38 287 #define PIO2 1.570796326794896619231e0 288 #define PI (PIO2+PIO2) 289 290 /* 291 * Time-of-day 292 */ 293 294 typedef 295 struct Tm 296 { 297 int sec; 298 int min; 299 int hour; 300 int mday; 301 int mon; 302 int year; 303 int wday; 304 int yday; 305 char zone[4]; 306 int tzoff; 307 } Tm; 308 309 extern Tm* gmtime(long); 310 extern Tm* localtime(long); 311 extern char* asctime(Tm*); 312 extern char* ctime(long); 313 extern double cputime(void); 314 extern long times(long*); 315 extern long tm2sec(Tm*); 316 extern vlong nsec(void); 317 318 extern void cycles(uvlong*); /* 64-bit value of the cycle counter if there is one, 0 if there isn't */ 319 320 /* 321 * one-of-a-kind 322 */ 323 enum 324 { 325 PNPROC = 1, 326 PNGROUP = 2, 327 }; 328 329 extern void _assert(char*); 330 extern int abs(int); 331 extern int atexit(void(*)(void)); 332 extern void atexitdont(void(*)(void)); 333 extern int atnotify(int(*)(void*, char*), int); 334 extern double atof(char*); 335 extern int atoi(char*); 336 extern long atol(char*); 337 extern vlong atoll(char*); 338 extern double charstod(int(*)(void*), void*); 339 extern char* cleanname(char*); 340 extern int decrypt(void*, void*, int); 341 extern int encrypt(void*, void*, int); 342 extern int dec64(uchar*, int, char*, int); 343 extern int enc64(char*, int, uchar*, int); 344 extern int dec32(uchar*, int, char*, int); 345 extern int enc32(char*, int, uchar*, int); 346 extern int dec16(uchar*, int, char*, int); 347 extern int enc16(char*, int, uchar*, int); 348 extern int encodefmt(Fmt*); 349 extern void exits(char*); 350 extern double frexp(double, int*); 351 extern uintptr getcallerpc(void*); 352 extern char* getenv(char*); 353 extern int getfields(char*, char**, int, int, char*); 354 extern int gettokens(char *, char **, int, char *); 355 extern char* getuser(void); 356 extern char* getwd(char*, int); 357 extern int iounit(int); 358 extern long labs(long); 359 extern double ldexp(double, int); 360 extern void longjmp(jmp_buf, int); 361 extern char* mktemp(char*); 362 extern double modf(double, double*); 363 extern int netcrypt(void*, void*); 364 extern void notejmp(void*, jmp_buf, int); 365 extern void perror(char*); 366 extern int postnote(int, int, char *); 367 extern double pow10(int); 368 extern int putenv(char*, char*); 369 extern void qsort(void*, long, long, int (*)(void*, void*)); 370 extern int setjmp(jmp_buf); 371 extern double strtod(char*, char**); 372 extern long strtol(char*, char**, int); 373 extern ulong strtoul(char*, char**, int); 374 extern vlong strtoll(char*, char**, int); 375 extern uvlong strtoull(char*, char**, int); 376 extern void sysfatal(char*, ...); 377 #pragma varargck argpos sysfatal 1 378 extern void syslog(int, char*, char*, ...); 379 #pragma varargck argpos syslog 3 380 extern long time(long*); 381 extern int tolower(int); 382 extern int toupper(int); 383 384 /* 385 * profiling 386 */ 387 enum { 388 Profoff, /* No profiling */ 389 Profuser, /* Measure user time only (default) */ 390 Profkernel, /* Measure user + kernel time */ 391 Proftime, /* Measure total time */ 392 Profsample, /* Use clock interrupt to sample (default when there is no cycle counter) */ 393 }; /* what */ 394 extern void prof(void (*fn)(void*), void *arg, int entries, int what); 395 396 /* 397 * synchronization 398 */ 399 typedef 400 struct Lock { 401 int val; 402 } Lock; 403 404 extern int _tas(int*); 405 406 extern void lock(Lock*); 407 extern void unlock(Lock*); 408 extern int canlock(Lock*); 409 410 typedef struct QLp QLp; 411 struct QLp 412 { 413 int inuse; 414 QLp *next; 415 char state; 416 }; 417 418 typedef 419 struct QLock 420 { 421 Lock lock; 422 int locked; 423 QLp *head; 424 QLp *tail; 425 } QLock; 426 427 extern void qlock(QLock*); 428 extern void qunlock(QLock*); 429 extern int canqlock(QLock*); 430 extern void _qlockinit(void* (*)(void*, void*)); /* called only by the thread library */ 431 432 typedef 433 struct RWLock 434 { 435 Lock lock; 436 int readers; /* number of readers */ 437 int writer; /* number of writers */ 438 QLp *head; /* list of waiting processes */ 439 QLp *tail; 440 } RWLock; 441 442 extern void rlock(RWLock*); 443 extern void runlock(RWLock*); 444 extern int canrlock(RWLock*); 445 extern void wlock(RWLock*); 446 extern void wunlock(RWLock*); 447 extern int canwlock(RWLock*); 448 449 typedef 450 struct Rendez 451 { 452 QLock *l; 453 QLp *head; 454 QLp *tail; 455 } Rendez; 456 457 extern void rsleep(Rendez*); /* unlocks r->l, sleeps, locks r->l again */ 458 extern int rwakeup(Rendez*); 459 extern int rwakeupall(Rendez*); 460 extern void** privalloc(void); 461 extern void privfree(void**); 462 463 /* 464 * network dialing 465 */ 466 #define NETPATHLEN 40 467 extern int accept(int, char*); 468 extern int announce(char*, char*); 469 extern int dial(char*, char*, char*, int*); 470 extern void setnetmtpt(char*, int, char*); 471 extern int hangup(int); 472 extern int listen(char*, char*); 473 extern char* netmkaddr(char*, char*, char*); 474 extern int reject(int, char*, char*); 475 476 /* 477 * encryption 478 */ 479 extern int pushssl(int, char*, char*, char*, int*); 480 extern int pushtls(int, char*, char*, int, char*, char*); 481 482 /* 483 * network services 484 */ 485 typedef struct NetConnInfo NetConnInfo; 486 struct NetConnInfo 487 { 488 char *dir; /* connection directory */ 489 char *root; /* network root */ 490 char *spec; /* binding spec */ 491 char *lsys; /* local system */ 492 char *lserv; /* local service */ 493 char *rsys; /* remote system */ 494 char *rserv; /* remote service */ 495 char *laddr; /* local address */ 496 char *raddr; /* remote address */ 497 }; 498 extern NetConnInfo* getnetconninfo(char*, int); 499 extern void freenetconninfo(NetConnInfo*); 500 501 /* 502 * system calls 503 * 504 */ 505 #define STATMAX 65535U /* max length of machine-independent stat structure */ 506 #define DIRMAX (sizeof(Dir)+STATMAX) /* max length of Dir structure */ 507 #define ERRMAX 128 /* max length of error string */ 508 509 #define MORDER 0x0003 /* mask for bits defining order of mounting */ 510 #define MREPL 0x0000 /* mount replaces object */ 511 #define MBEFORE 0x0001 /* mount goes before others in union directory */ 512 #define MAFTER 0x0002 /* mount goes after others in union directory */ 513 #define MCREATE 0x0004 /* permit creation in mounted directory */ 514 #define MCACHE 0x0010 /* cache some data */ 515 #define MMASK 0x0017 /* all bits on */ 516 517 #define OREAD 0 /* open for read */ 518 #define OWRITE 1 /* write */ 519 #define ORDWR 2 /* read and write */ 520 #define OEXEC 3 /* execute, == read but check execute permission */ 521 #define OTRUNC 16 /* or'ed in (except for exec), truncate file first */ 522 #define OCEXEC 32 /* or'ed in, close on exec */ 523 #define ORCLOSE 64 /* or'ed in, remove on close */ 524 #define OEXCL 0x1000 /* or'ed in, exclusive use (create only) */ 525 526 #define AEXIST 0 /* accessible: exists */ 527 #define AEXEC 1 /* execute access */ 528 #define AWRITE 2 /* write access */ 529 #define AREAD 4 /* read access */ 530 531 /* Segattch */ 532 #define SG_RONLY 0040 /* read only */ 533 #define SG_CEXEC 0100 /* detach on exec */ 534 535 #define NCONT 0 /* continue after note */ 536 #define NDFLT 1 /* terminate after note */ 537 #define NSAVE 2 /* clear note but hold state */ 538 #define NRSTR 3 /* restore saved state */ 539 540 /* bits in Qid.type */ 541 #define QTDIR 0x80 /* type bit for directories */ 542 #define QTAPPEND 0x40 /* type bit for append only files */ 543 #define QTEXCL 0x20 /* type bit for exclusive use files */ 544 #define QTMOUNT 0x10 /* type bit for mounted channel */ 545 #define QTAUTH 0x08 /* type bit for authentication file */ 546 #define QTTMP 0x04 /* type bit for not-backed-up file */ 547 #define QTFILE 0x00 /* plain file */ 548 549 /* bits in Dir.mode */ 550 #define DMDIR 0x80000000 /* mode bit for directories */ 551 #define DMAPPEND 0x40000000 /* mode bit for append only files */ 552 #define DMEXCL 0x20000000 /* mode bit for exclusive use files */ 553 #define DMMOUNT 0x10000000 /* mode bit for mounted channel */ 554 #define DMAUTH 0x08000000 /* mode bit for authentication file */ 555 #define DMTMP 0x04000000 /* mode bit for non-backed-up files */ 556 #define DMREAD 0x4 /* mode bit for read permission */ 557 #define DMWRITE 0x2 /* mode bit for write permission */ 558 #define DMEXEC 0x1 /* mode bit for execute permission */ 559 560 /* rfork */ 561 enum 562 { 563 RFNAMEG = (1<<0), 564 RFENVG = (1<<1), 565 RFFDG = (1<<2), 566 RFNOTEG = (1<<3), 567 RFPROC = (1<<4), 568 RFMEM = (1<<5), 569 RFNOWAIT = (1<<6), 570 RFCNAMEG = (1<<10), 571 RFCENVG = (1<<11), 572 RFCFDG = (1<<12), 573 RFREND = (1<<13), 574 RFNOMNT = (1<<14) 575 }; 576 577 typedef 578 struct Qid 579 { 580 uvlong path; 581 ulong vers; 582 uchar type; 583 } Qid; 584 585 typedef 586 struct Dir { 587 /* system-modified data */ 588 ushort type; /* server type */ 589 uint dev; /* server subtype */ 590 /* file data */ 591 Qid qid; /* unique id from server */ 592 ulong mode; /* permissions */ 593 ulong atime; /* last read time */ 594 ulong mtime; /* last write time */ 595 vlong length; /* file length */ 596 char *name; /* last element of path */ 597 char *uid; /* owner name */ 598 char *gid; /* group name */ 599 char *muid; /* last modifier name */ 600 } Dir; 601 602 /* keep /sys/src/ape/lib/ap/plan9/sys9.h in sync with this -rsc */ 603 typedef 604 struct Waitmsg 605 { 606 int pid; /* of loved one */ 607 ulong time[3]; /* of loved one & descendants */ 608 char *msg; 609 } Waitmsg; 610 611 typedef 612 struct IOchunk 613 { 614 void *addr; 615 ulong len; 616 } IOchunk; 617 618 extern void _exits(char*); 619 620 extern void abort(void); 621 extern int access(char*, int); 622 extern long alarm(ulong); 623 extern int await(char*, int); 624 extern int bind(char*, char*, int); 625 extern int brk(void*); 626 extern int chdir(char*); 627 extern int close(int); 628 extern int create(char*, int, ulong); 629 extern int dup(int, int); 630 extern int errstr(char*, uint); 631 extern int exec(char*, char*[]); 632 extern int execl(char*, ...); 633 extern int fork(void); 634 extern int rfork(int); 635 extern int fauth(int, char*); 636 extern int fstat(int, uchar*, int); 637 extern int fwstat(int, uchar*, int); 638 extern int fversion(int, int, char*, int); 639 extern int mount(int, int, char*, int, char*); 640 extern int unmount(char*, char*); 641 extern int noted(int); 642 extern int notify(void(*)(void*, char*)); 643 extern int open(char*, int); 644 extern int fd2path(int, char*, int); 645 extern int pipe(int*); 646 extern long pread(int, void*, long, vlong); 647 extern long preadv(int, IOchunk*, int, vlong); 648 extern long pwrite(int, void*, long, vlong); 649 extern long pwritev(int, IOchunk*, int, vlong); 650 extern long read(int, void*, long); 651 extern long readn(int, void*, long); 652 extern long readv(int, IOchunk*, int); 653 extern int remove(char*); 654 extern void* sbrk(ulong); 655 extern long oseek(int, long, int); 656 extern vlong seek(int, vlong, int); 657 extern void* segattach(int, char*, void*, ulong); 658 extern void* segbrk(void*, void*); 659 extern int segdetach(void*); 660 extern int segflush(void*, ulong); 661 extern int segfree(void*, ulong); 662 extern int semacquire(long*, int); 663 extern long semrelease(long*, long); 664 extern int sleep(long); 665 extern int stat(char*, uchar*, int); 666 extern Waitmsg* wait(void); 667 extern int waitpid(void); 668 extern long write(int, void*, long); 669 extern long writev(int, IOchunk*, int); 670 extern int wstat(char*, uchar*, int); 671 extern void* rendezvous(void*, void*); 672 673 extern Dir* dirstat(char*); 674 extern Dir* dirfstat(int); 675 extern int dirwstat(char*, Dir*); 676 extern int dirfwstat(int, Dir*); 677 extern long dirread(int, Dir**); 678 extern void nulldir(Dir*); 679 extern long dirreadall(int, Dir**); 680 extern int getpid(void); 681 extern int getppid(void); 682 extern void rerrstr(char*, uint); 683 extern char* sysname(void); 684 extern void werrstr(char*, ...); 685 #pragma varargck argpos werrstr 1 686 687 extern char *argv0; 688 #define ARGBEGIN for((argv0||(argv0=*argv)),argv++,argc--;\ 689 argv[0] && argv[0][0]=='-' && argv[0][1];\ 690 argc--, argv++) {\ 691 char *_args, *_argt;\ 692 Rune _argc;\ 693 _args = &argv[0][1];\ 694 if(_args[0]=='-' && _args[1]==0){\ 695 argc--; argv++; break;\ 696 }\ 697 _argc = 0;\ 698 while(*_args && (_args += chartorune(&_argc, _args)))\ 699 switch(_argc) 700 #define ARGEND SET(_argt);USED(_argt,_argc,_args);}USED(argv, argc); 701 #define ARGF() (_argt=_args, _args="",\ 702 (*_argt? _argt: argv[1]? (argc--, *++argv): 0)) 703 #define EARGF(x) (_argt=_args, _args="",\ 704 (*_argt? _argt: argv[1]? (argc--, *++argv): ((x), abort(), (char*)0))) 705 706 #define ARGC() _argc 707 708 /* this is used by sbrk and brk, it's a really bad idea to redefine it */ 709 extern char end[]; 710