mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 02:30:57 +00:00 
			
		
		
		
	Add glob and some finer tuning of documentation
This commit is contained in:
		
							parent
							
								
									799e24a87b
								
							
						
					
					
						commit
						d51409ccd9
					
				
					 77 changed files with 1321 additions and 736 deletions
				
			
		
							
								
								
									
										5
									
								
								Makefile
									
										
									
									
									
								
							
							
						
						
									
										5
									
								
								Makefile
									
										
									
									
									
								
							|  | @ -231,8 +231,9 @@ HTAGS:	o/$(MODE)/hdrs.txt $(HDRS) | ||||||
| 	@rm -f $@ | 	@rm -f $@ | ||||||
| 	@ACTION=TAGS TARGET=$@ build/do build/htags -L $< -o $@ | 	@ACTION=TAGS TARGET=$@ build/do build/htags -L $< -o $@ | ||||||
| 
 | 
 | ||||||
| loc:;	find -name \*.h -or -name \*.c -or -name \*.S | \ | loc: o/$(MODE)/tool/build/summy.com | ||||||
| 	$(XARGS) wc -l | grep total | awk '{print $$1}' | summy | 	find -name \*.h -or -name \*.c -or -name \*.S | \
 | ||||||
|  | 	$(XARGS) wc -l | grep total | awk '{print $$1}' | $< | ||||||
| 
 | 
 | ||||||
| # UNSPECIFIED PREREQUISITES TUTORIAL
 | # UNSPECIFIED PREREQUISITES TUTORIAL
 | ||||||
| #
 | #
 | ||||||
|  |  | ||||||
							
								
								
									
										13
									
								
								NOTICE
									
										
									
									
									
								
							
							
						
						
									
										13
									
								
								NOTICE
									
										
									
									
									
								
							|  | @ -1,16 +1,3 @@ | ||||||
| Cosmopolitan is Free Software licensed under the GPLv2. The build config |  | ||||||
| **will embed all linked sources inside your binaries** so the compliance |  | ||||||
| is easy while facilitating trust and transparency similar to JavaScript. |  | ||||||
| You can audit your source filesystem using ZIP GUIs e.g. WIN10, InfoZip. |  | ||||||
| 
 |  | ||||||
| If you want to be able to distribute binaries in binary form only, then |  | ||||||
| please send $1,000 to jtunney@gmail.com on PayPal for a license lasting |  | ||||||
| 1 year. Please be sure to provide your contact information, and details |  | ||||||
| on whether or not the license is for an individual or a corporation. If |  | ||||||
| you want your license to last 5 years, send $10,000 to the author above |  | ||||||
| who is Justine Tunney <jtunney@gmail.com>. Reach out, for more details. |  | ||||||
| 
 |  | ||||||
| ──────────────────────────────────────────────────────────────────────── |  | ||||||
| 
 | 
 | ||||||
| 		    GNU GENERAL PUBLIC LICENSE | 		    GNU GENERAL PUBLIC LICENSE | ||||||
| 		       Version 2, June 1991 | 		       Version 2, June 1991 | ||||||
|  |  | ||||||
							
								
								
									
										44
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										44
									
								
								README.md
									
										
									
									
									
								
							|  | @ -18,17 +18,19 @@ Cosmopolitan provides a native development environment similar to those | ||||||
| currently being offered by RedHat, Google, Apple, Microsoft, etc. We're | currently being offered by RedHat, Google, Apple, Microsoft, etc. We're | ||||||
| different in two ways: (1) we're not a platform therefore we don't have | different in two ways: (1) we're not a platform therefore we don't have | ||||||
| any commercial interest in making our tooling work better on one rather | any commercial interest in making our tooling work better on one rather | ||||||
| than another; and (2) we're only focused on catering towards interfaces | than another; and (2) we cater primarily towards features having gained | ||||||
| all platforms agree upon. Goal is software that stands the test of time | cross-platform agreement. Goal is software that stands the test of time | ||||||
| without the costs and restrictions cross-platform distribution entails. | without the costs and restrictions cross-platform distribution entails. | ||||||
|  | 
 | ||||||
| That makes Cosmopolitan an excellent fit for writing small CLI programs | That makes Cosmopolitan an excellent fit for writing small CLI programs | ||||||
| that do things like heavyweight numerical computations as a subprocess. | that do things like matrix multiplication relu stdio as a subprocess or | ||||||
|  | perhaps a web server having the minimum surface area that you require. | ||||||
| 
 | 
 | ||||||
| ## Getting Started | ## Getting Started | ||||||
| 
 | 
 | ||||||
| Just clone the repository and put your own folder in it. Please choose a | Just clone the repository and put your own folder in it. Please choose a | ||||||
| name that's based on a .com or .org domain name registration you control | name that's based on a .com or .org domain name registration you control | ||||||
| which is scout's honor but might change to require TXT records in future | which is scout's honor but could change to verify TXT records in future. | ||||||
| 
 | 
 | ||||||
| We provide a script that makes it easy to start a new package: | We provide a script that makes it easy to start a new package: | ||||||
| 
 | 
 | ||||||
|  | @ -37,11 +39,11 @@ We provide a script that makes it easy to start a new package: | ||||||
|     make o//com/github/user/project/program.com |     make o//com/github/user/project/program.com | ||||||
|     o//com/github/user/project/program.com |     o//com/github/user/project/program.com | ||||||
| 
 | 
 | ||||||
| Please note GNU make is awesome. Little known fact: make is a functional | Please note GNU Make is awesome. Little known fact: make is a functional | ||||||
| programming language. We improved upon it too! In [tool/build/package.c] | programming language. We improved upon it too! In [tool/build/package.c] | ||||||
| which performs strict dependency checking, to correct Google's published | which performs strict dependency checking, to correct Google's published | ||||||
| mistakes c. 2006 which was when they switched from using a GNU Make repo | mistakes c. 2006 which was when they switched from using a GNU Make repo | ||||||
| in favor of an inhouse derivative called Blaze which does strictness too | in favor of an inhouse derivative called Blaze which does graph checking | ||||||
| thereby allowing the repository to grow gracefully with any requirements | thereby allowing the repository to grow gracefully with any requirements | ||||||
| 
 | 
 | ||||||
| ## Performance | ## Performance | ||||||
|  | @ -66,18 +68,20 @@ Furthermore Cosmopolitan provides you with Intel's official instruction | ||||||
| length decoder Xed ravaged down to 3kb in size using Tim Patterson code | length decoder Xed ravaged down to 3kb in size using Tim Patterson code | ||||||
| stunts. So you can absolutely code high-performance lightweight fabless | stunts. So you can absolutely code high-performance lightweight fabless | ||||||
| x86 microprocessors using any hobbyist board without royalties, because | x86 microprocessors using any hobbyist board without royalties, because | ||||||
| Xed tells us which parts of the encoding space now belong to the people | Xed tells us which parts of the encoding space now belong to the people. | ||||||
|  | Please see [third_party/xed/x86ild.greg.c] to learn more. | ||||||
| 
 | 
 | ||||||
| ## Integrated Development Environment | ## Integrated Development Environment | ||||||
| 
 | 
 | ||||||
| Your Cosmopolitan IDE is based on Emacs. When used alongside `make tags` | Your Cosmopolitan IDE is based on whichever editor you love most. Emacs | ||||||
| it can automate certain toilsome problems such as adding an include line | configs are provided, showing how `make tags` can be used to automate | ||||||
| by typing `C-c C-h` with the cursor over a symbol. Please install these: | certain toil, such as adding include lines by typing `C-c C-h` over a | ||||||
|  | symbol. We recommend trying the following: | ||||||
| 
 | 
 | ||||||
|     sudo bash -c "$(wget -O - https://apt.llvm.org/llvm.sh)" |     sudo bash -c "$(wget -O - https://apt.llvm.org/llvm.sh)" | ||||||
|     sudo apt install gdb ragel ctags clang-format-10  # <-recommended |     sudo apt install gdb ragel ctags clang-format-10 | ||||||
|     git clone git@github.com:jart/cosmopolitan.git && cd cosmopolitan |     git clone git@github.com:jart/cosmopolitan.git && cd cosmopolitan | ||||||
|     tool/scripts/install-emacs.sh    # recommended over apt |     tool/scripts/install-emacs.sh    # Emacs 26.1 has a bug | ||||||
|     tool/scripts/configure-emacs.sh  # adds load statements |     tool/scripts/configure-emacs.sh  # adds load statements | ||||||
|     make tags                        # index all the symbol |     make tags                        # index all the symbol | ||||||
|     emacs                            # for power and glory! |     emacs                            # for power and glory! | ||||||
|  | @ -95,10 +99,9 @@ should explain how easy it is to do pe+sh+elf+macho+bootloader overlays | ||||||
| enabling gnu ld.bfd to produce a simple portable 12kb .com program file | enabling gnu ld.bfd to produce a simple portable 12kb .com program file | ||||||
| 
 | 
 | ||||||
| Please send feedback and concerns to <jtunney@gmail.com> since we would | Please send feedback and concerns to <jtunney@gmail.com> since we would | ||||||
| love to do an even better job. Please also report GPL abuse the address | love to do an even better job. Please also let us know if some goofball | ||||||
| above as well. We believe this binary structure is novel, and therefore | is distributing without bundling their source code, so we can reach out | ||||||
| easily recognizable. So if some goofball isn't bundling the source code | and ask if they would like to purchase a commercial license. | ||||||
| please let us know, so we can reach out and ask if they want a license. |  | ||||||
| 
 | 
 | ||||||
|     make -j10 -O MODE=tiny CPPFLAGS=-DIM_FEELING_NAUGHTY |     make -j10 -O MODE=tiny CPPFLAGS=-DIM_FEELING_NAUGHTY | ||||||
|     o/tiny/tool/viz/bing.com <o/tiny/examples/life.com |  |     o/tiny/tool/viz/bing.com <o/tiny/examples/life.com |  | ||||||
|  | @ -244,6 +247,7 @@ Cosmopolitan is Free Software licensed under the GPLv2. The build config | ||||||
| **will embed all linked sources inside your binaries** so the compliance | **will embed all linked sources inside your binaries** so the compliance | ||||||
| is easy while facilitating trust and transparency similar to JavaScript. | is easy while facilitating trust and transparency similar to JavaScript. | ||||||
| You can audit your source filesystem using ZIP GUIs e.g. Win10, InfoZip. | You can audit your source filesystem using ZIP GUIs e.g. Win10, InfoZip. | ||||||
|  | That works easiest by changing the filename extension from .com to .zip. | ||||||
| 
 | 
 | ||||||
| ### Commercial Support | ### Commercial Support | ||||||
| 
 | 
 | ||||||
|  | @ -276,9 +280,9 @@ ideas, aside from taking care of low-level build system toil, so coding | ||||||
| can become more beautifully pleasant. | can become more beautifully pleasant. | ||||||
| 
 | 
 | ||||||
| Cosmopolitan is maintained by Justine Tunney, who previously worked on | Cosmopolitan is maintained by Justine Tunney, who previously worked on | ||||||
| projects like TensorFlow, [Operation Rosehub], [Nomulus], and Google's | projects like [TensorFlow], [Operation Rosehub], [Nomulus], and Google's | ||||||
| tape backups sre team. She's also helped radical activists by operating | tape backups SRE team. She's also helped activists by operating the | ||||||
| the [Occupy Wall Street] website. Justine Tunney currently isn't on the | [Occupy Wall Street] website. Justine Tunney currently isn't on the | ||||||
| payroll of any company. She's been focusing on Cosmopolitan because she | payroll of any company. She's been focusing on Cosmopolitan because she | ||||||
| wants to give back to Free Software which helped her be successful. See | wants to give back to Free Software which helped her be successful. See | ||||||
| her [LinkedIn] profile for further details on her professional history. | her [LinkedIn] profile for further details on her professional history. | ||||||
|  | @ -288,6 +292,7 @@ her [LinkedIn] profile for further details on her professional history. | ||||||
| [Nomulus]: https://github.com/google/nomulus | [Nomulus]: https://github.com/google/nomulus | ||||||
| [Occupy Wall Street]: https://www.newyorker.com/magazine/2011/11/28/pre-occupied | [Occupy Wall Street]: https://www.newyorker.com/magazine/2011/11/28/pre-occupied | ||||||
| [Operation Rosehub]: https://opensource.googleblog.com/2017/03/operation-rosehub.html | [Operation Rosehub]: https://opensource.googleblog.com/2017/03/operation-rosehub.html | ||||||
|  | [TensorFlow]: https://github.com/tensorflow/tensorflow | ||||||
| [ansitrinsics library]: libc/intrin | [ansitrinsics library]: libc/intrin | ||||||
| [ape/ape.lds]: ape/ape.lds | [ape/ape.lds]: ape/ape.lds | ||||||
| [dsp/scale/cdecimate2xuint8x8.c]: dsp/scale/cdecimate2xuint8x8.c | [dsp/scale/cdecimate2xuint8x8.c]: dsp/scale/cdecimate2xuint8x8.c | ||||||
|  | @ -297,6 +302,7 @@ her [LinkedIn] profile for further details on her professional history. | ||||||
| [libc/nexgen32e/x86feature.h]: libc/nexgen32e/x86feature.h | [libc/nexgen32e/x86feature.h]: libc/nexgen32e/x86feature.h | ||||||
| [libc/sysv/consts.sh]: libc/sysv/consts.sh | [libc/sysv/consts.sh]: libc/sysv/consts.sh | ||||||
| [libc/sysv/syscalls.sh]: libc/sysv/syscalls.sh | [libc/sysv/syscalls.sh]: libc/sysv/syscalls.sh | ||||||
|  | [third_party/xed/x86ild.greg.c]: third_party/xed/x86ild.greg.c | ||||||
| [tool/build/package.c]: tool/build/package.c | [tool/build/package.c]: tool/build/package.c | ||||||
| [tool/emacs/cosmo-stuff.el]: tool/emacs/cosmo-stuff.el | [tool/emacs/cosmo-stuff.el]: tool/emacs/cosmo-stuff.el | ||||||
| [tool/viz/lib/ycbcr2rgb3.c]: tool/viz/lib/ycbcr2rgb3.c | [tool/viz/lib/ycbcr2rgb3.c]: tool/viz/lib/ycbcr2rgb3.c | ||||||
|  |  | ||||||
|  | @ -141,12 +141,37 @@ endif | ||||||
| 
 | 
 | ||||||
| # ANSI Mode
 | # ANSI Mode
 | ||||||
| #
 | #
 | ||||||
| # Folks who want it deserve to get it, good and hard.
 | # These flags cause GCC to predefine __STRICT_ANSI__. Please be warned
 | ||||||
|  | # that Cosmopolitan headers are written to comply with that request if
 | ||||||
|  | # it's possible to do so. Consider the following example:
 | ||||||
|  | #
 | ||||||
|  | #   make -j12 -O o//tool/viz/printvideo.i
 | ||||||
|  | #   clang-format-10 -i o//tool/viz/printvideo.i
 | ||||||
|  | #   less o//tool/viz/printvideo.i
 | ||||||
|  | #
 | ||||||
|  | # You'll notice functions like memcpy(), ioctl(), etc. get expanded into
 | ||||||
|  | # wild-eyed gnu-style performance hacks. You can turn it off as follows:
 | ||||||
|  | #
 | ||||||
|  | #   make -j12 -O MODE=ansi o/ansi/tool/viz/printvideo.i
 | ||||||
|  | #   clang-format-10 -i o/ansi/tool/viz/printvideo.i
 | ||||||
|  | #   less o/ansi/tool/viz/printvideo.i
 | ||||||
|  | #
 | ||||||
|  | # Here it becomes clear that ANSI mode can help you decouple your source
 | ||||||
|  | # from Cosmopolitan, by turning it into plain ordinary textbook C code.
 | ||||||
|  | #
 | ||||||
|  | # Another potential use case is distributing code to folks using tools
 | ||||||
|  | # such as MSVC or XCode. You can run your binary objects through a tool
 | ||||||
|  | # like objconv to convert them to COFF or MachO. Then use ANSI mode to
 | ||||||
|  | # rollup one header file that'll enable linkage with minimal issues.
 | ||||||
|  | #
 | ||||||
|  | # Lastly note that in some cases, such as gc(), there simply isn't any
 | ||||||
|  | # ANSI workaround available. It's only in cases like that when we'll use
 | ||||||
|  | # the __asm__() header workaround, rather than simply removing it. We do
 | ||||||
|  | # however try to do that much less often than mainstream C libraries.
 | ||||||
| 
 | 
 | ||||||
| ifeq ($(MODE), ansi) | ifeq ($(MODE), ansi) | ||||||
| 
 | 
 | ||||||
| CONFIG_CCFLAGS +=		\
 | CONFIG_CFLAGS += -std=c11 | ||||||
| 	-std=c11		\
 | CONFIG_CXXFLAGS += -std=c++11 | ||||||
| 	-Og |  | ||||||
| 
 | 
 | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | @ -96,4 +96,3 @@ build/bootstrap/%.c.gz: %.rl | ||||||
| %.svgz: %.rl | %.svgz: %.rl | ||||||
| 	@$(RAGEL) -V -p $< | $(DOT) -Tsvg | $(GZ) $(ZFLAGS) >$@ | 	@$(RAGEL) -V -p $< | $(DOT) -Tsvg | $(GZ) $(ZFLAGS) >$@ | ||||||
| 
 | 
 | ||||||
| 	$(LIBC__A_SRCS:%=o/$(MODE)/%.zip.o)	\
 |  | ||||||
|  |  | ||||||
|  | @ -1,37 +0,0 @@ | ||||||
| #if 0 |  | ||||||
| /*─────────────────────────────────────────────────────────────────╗
 |  | ||||||
| │ To the extent possible under law, Justine Tunney has waived      │ |  | ||||||
| │ all copyright and related or neighboring rights to this file,    │ |  | ||||||
| │ as it is written in the following disclaimers:                   │ |  | ||||||
| │   • http://unlicense.org/                                        │
 |  | ||||||
| │   • http://creativecommons.org/publicdomain/zero/1.0/            │
 |  | ||||||
| ╚─────────────────────────────────────────────────────────────────*/ |  | ||||||
| #endif |  | ||||||
| #include "libc/dce.h" |  | ||||||
| #include "libc/nt/dll.h" |  | ||||||
| #include "libc/nt/nt/process.h" |  | ||||||
| #include "libc/runtime/runtime.h" |  | ||||||
| #include "libc/stdio/stdio.h" |  | ||||||
| #include "libc/sysv/consts/exit.h" |  | ||||||
| 
 |  | ||||||
| int main() { |  | ||||||
|   if (IsWindows()) { |  | ||||||
|     /*
 |  | ||||||
|      * Cosmopolitan imports this automatically if it's referenced the |  | ||||||
|      * normal way. But Microsoft wants us to use loose coupling when |  | ||||||
|      * referencing their internal APIs. Don't think for a moment they |  | ||||||
|      * won't break open source projects that fail to heed the warning. |  | ||||||
|      */ |  | ||||||
|     typeof(NtTerminateProcess) *NtTerminateProcess_ = |  | ||||||
|         GetProcAddress(GetModuleHandle("ntdll.dll"), "NtTerminateProcess"); |  | ||||||
|     printf("%ld\n", GetModuleHandle("ntdll.dll")); |  | ||||||
|     printf("%p\n", NtTerminateProcess_); |  | ||||||
|     if (NtTerminateProcess_) { |  | ||||||
|       fflush(stdout); |  | ||||||
|       for (;;) NtTerminateProcess_(-1, 42); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     fprintf(stderr, "error: intended for windows\n"); |  | ||||||
|   } |  | ||||||
|   return EXIT_FAILURE; |  | ||||||
| } |  | ||||||
|  | @ -13,17 +13,14 @@ void *bsearch_r(const void *, const void *, size_t, size_t, | ||||||
|                 int cmp(const void *, const void *, void *), void *) |                 int cmp(const void *, const void *, void *), void *) | ||||||
|     paramsnonnull((1, 2, 5)) nothrow nosideeffect; |     paramsnonnull((1, 2, 5)) nothrow nosideeffect; | ||||||
| void djbsort(size_t n, int32_t[n]); | void djbsort(size_t n, int32_t[n]); | ||||||
| void qsort(void *items, size_t itemcount, size_t itemsize, | void qsort(void *, size_t, size_t, int (*)(const void *, const void *)) | ||||||
|            int cmp(const void *, const void *)) paramsnonnull(); |     paramsnonnull(); | ||||||
| void qsort_r(void *items, size_t itemcount, size_t itemsize, | void qsort_r(void *, size_t, size_t, | ||||||
|              int cmp(const void *, const void *, void *), void *arg) |              int cmp(const void *, const void *, void *), void *arg) | ||||||
|     paramsnonnull((1, 4)); |     paramsnonnull((1, 4)); | ||||||
| int tarjan(uint32_t vertex_count, const uint32_t (*edges)[2], | int tarjan(uint32_t, const uint32_t (*)[2], uint32_t, uint32_t[], uint32_t[], | ||||||
|            uint32_t edge_count, uint32_t out_sorted[], |            uint32_t *) paramsnonnull((2, 4)) nocallback nothrow; | ||||||
|            uint32_t out_opt_components[], uint32_t *out_opt_componentcount) | void heapsortcar(int32_t (*)[2], unsigned) paramsnonnull() nocallback nothrow; | ||||||
|     paramsnonnull((2, 4)) nocallback nothrow; |  | ||||||
| void heapsortcar(int32_t (*A)[2], unsigned n) |  | ||||||
|     paramsnonnull() nocallback nothrow; |  | ||||||
| 
 | 
 | ||||||
| void *memmem(const void *, size_t, const void *, size_t) | void *memmem(const void *, size_t, const void *, size_t) | ||||||
|     paramsnonnull() nothrow nocallback nosideeffect; |     paramsnonnull() nothrow nocallback nosideeffect; | ||||||
|  | @ -56,16 +53,16 @@ struct critbit0 { | ||||||
|   size_t count; |   size_t count; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| bool critbit0_contains(struct critbit0 *t, const char *u) nothrow nosideeffect | bool critbit0_contains(struct critbit0 *, const char *) nothrow nosideeffect | ||||||
|     paramsnonnull(); |     paramsnonnull(); | ||||||
| bool critbit0_insert(struct critbit0 *t, const char *u) paramsnonnull(); | bool critbit0_insert(struct critbit0 *, const char *) paramsnonnull(); | ||||||
| bool critbit0_delete(struct critbit0 *t, const char *u) nothrow paramsnonnull(); | bool critbit0_delete(struct critbit0 *, const char *) nothrow paramsnonnull(); | ||||||
| void critbit0_clear(struct critbit0 *t) nothrow paramsnonnull(); | void critbit0_clear(struct critbit0 *) nothrow paramsnonnull(); | ||||||
| char *critbit0_get(struct critbit0 *t, const char *u); | char *critbit0_get(struct critbit0 *, const char *); | ||||||
| intptr_t critbit0_allprefixed(struct critbit0 *t, const char *prefix, | intptr_t critbit0_allprefixed(struct critbit0 *, const char *, | ||||||
|                               intptr_t (*callback)(const char *elem, void *arg), |                               intptr_t (*)(const char *, void *), void *) | ||||||
|                               void *arg) paramsnonnull((1, 2, 3)) nothrow; |     paramsnonnull((1, 2, 3)) nothrow; | ||||||
| bool critbit0_emplace(struct critbit0 *t, char *u, size_t ulen) paramsnonnull(); | bool critbit0_emplace(struct critbit0 *, char *, size_t) paramsnonnull(); | ||||||
| 
 | 
 | ||||||
| /*───────────────────────────────────────────────────────────────────────────│─╗
 | /*───────────────────────────────────────────────────────────────────────────│─╗
 | ||||||
| │ cosmopolitan § algorithms » comparators                                  ─╬─│┼ | │ cosmopolitan § algorithms » comparators                                  ─╬─│┼ | ||||||
|  |  | ||||||
|  | @ -76,6 +76,7 @@ static textwindows noinline DIR *opendir$nt(const char *name) { | ||||||
|  * |  * | ||||||
|  * @returns newly allocated DIR object, or NULL w/ errno |  * @returns newly allocated DIR object, or NULL w/ errno | ||||||
|  * @errors ENOENT, ENOTDIR, EACCES, EMFILE, ENFILE, ENOMEM |  * @errors ENOENT, ENOTDIR, EACCES, EMFILE, ENFILE, ENOMEM | ||||||
|  |  * @see glob() | ||||||
|  */ |  */ | ||||||
| DIR *opendir(const char *name) { | DIR *opendir(const char *name) { | ||||||
|   if (!IsWindows() && !IsXnu()) { |   if (!IsWindows() && !IsXnu()) { | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ int ioctl(int, uint64_t, void *); | ||||||
| /*───────────────────────────────────────────────────────────────────────────│─╗
 | /*───────────────────────────────────────────────────────────────────────────│─╗
 | ||||||
| │ cosmopolitan § system calls » input output control » undiamonding        ─╬─│┼ | │ cosmopolitan § system calls » input output control » undiamonding        ─╬─│┼ | ||||||
| ╚────────────────────────────────────────────────────────────────────────────│*/ | ╚────────────────────────────────────────────────────────────────────────────│*/ | ||||||
| #ifdef __GNUC__ | #if defined(__GNUC__) && !defined(__STRICT_ANSI__) | ||||||
| #include "libc/macros.h" | #include "libc/macros.h" | ||||||
| #include "libc/sysv/consts/termios.h" | #include "libc/sysv/consts/termios.h" | ||||||
| 
 | 
 | ||||||
|  | @ -46,7 +46,7 @@ forceinline int ioctl$dispatch(int fd, uint64_t request, void *memory) { | ||||||
|   return ioctl$default(fd, request, memory); |   return ioctl$default(fd, request, memory); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #endif /* GNUC */ | #endif /* GNUC && !ANSI */ | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
| #endif /* COSMOPOLITAN_LIBC_CALLS_IOCTL_H_ */ | #endif /* COSMOPOLITAN_LIBC_CALLS_IOCTL_H_ */ | ||||||
|  |  | ||||||
|  | @ -28,8 +28,9 @@ int unlockpt(int); | ||||||
| int posix_openpt(int) nodiscard; | int posix_openpt(int) nodiscard; | ||||||
| 
 | 
 | ||||||
| /*───────────────────────────────────────────────────────────────────────────│─╗
 | /*───────────────────────────────────────────────────────────────────────────│─╗
 | ||||||
| │ cosmopolitan § teletypewriter » optimizations                            ─╬─│┼ | │ cosmopolitan § teletypewriter » undiamonding                             ─╬─│┼ | ||||||
| ╚────────────────────────────────────────────────────────────────────────────│*/ | ╚────────────────────────────────────────────────────────────────────────────│*/ | ||||||
|  | #if defined(__GNUC__) && !defined(__STRICT_ANSI__) | ||||||
| 
 | 
 | ||||||
| #define tcsetattr(FD, OPT, TIO) tcsetattr$dispatch(FD, OPT, TIO) | #define tcsetattr(FD, OPT, TIO) tcsetattr$dispatch(FD, OPT, TIO) | ||||||
| forceinline int tcsetattr$dispatch(int fd, int opt, const struct termios *tio) { | forceinline int tcsetattr$dispatch(int fd, int opt, const struct termios *tio) { | ||||||
|  | @ -44,6 +45,7 @@ forceinline int tcgetattr$dispatch(int fd, const struct termios *tio) { | ||||||
|   return ioctl(fd, TCGETS, (void *)tio); |   return ioctl(fd, TCGETS, (void *)tio); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #endif /* GNUC && !ANSI */ | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
| #endif /* COSMOPOLITAN_LIBC_CALLS_TERMIOS_H_ */ | #endif /* COSMOPOLITAN_LIBC_CALLS_TERMIOS_H_ */ | ||||||
|  |  | ||||||
|  | @ -3,7 +3,7 @@ | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * @fileoverview Types for scoundrels. |  * @fileoverview Types we'd prefer hadn't been invented. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #define blkcnt_t      int64_t | #define blkcnt_t      int64_t | ||||||
|  |  | ||||||
|  | @ -49,7 +49,7 @@ aes_block_t unrijndael(uint32_t, aes_block_t, const struct Rijndael *); | ||||||
| aligned(64) extern const uint8_t kAesSbox[256]; | aligned(64) extern const uint8_t kAesSbox[256]; | ||||||
| aligned(64) extern const uint8_t kAesSboxInverse[256]; | aligned(64) extern const uint8_t kAesSboxInverse[256]; | ||||||
| 
 | 
 | ||||||
| aes_block_t InvMixColumns(aes_block_t x) hidden; | aes_block_t InvMixColumns(aes_block_t) hidden; | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  |  | ||||||
|  | @ -18,12 +18,10 @@ COSMOPOLITAN_C_START_ | ||||||
| 
 | 
 | ||||||
| struct MappedFile; | struct MappedFile; | ||||||
| 
 | 
 | ||||||
| Elf64_Ehdr *mapelfread(const char *filename, struct MappedFile *mf); | Elf64_Ehdr *mapelfread(const char *, struct MappedFile *); | ||||||
| char *getelfstringtable(const Elf64_Ehdr *elf, size_t mapsize); | char *getelfstringtable(const Elf64_Ehdr *, size_t); | ||||||
| Elf64_Sym *getelfsymboltable(const Elf64_Ehdr *elf, size_t mapsize, | Elf64_Sym *getelfsymboltable(const Elf64_Ehdr *, size_t, Elf64_Xword *); | ||||||
|                              Elf64_Xword *out_count); | Elf64_Shdr *getelfsectionbyaddress(const Elf64_Ehdr *, size_t, void *); | ||||||
| Elf64_Shdr *getelfsectionbyaddress(const Elf64_Ehdr *elf, size_t mapsize, |  | ||||||
|                                    void *addr); |  | ||||||
| 
 | 
 | ||||||
| forceinline void checkelfaddress(const Elf64_Ehdr *elf, size_t mapsize, | forceinline void checkelfaddress(const Elf64_Ehdr *elf, size_t mapsize, | ||||||
|                                  intptr_t addr, size_t addrsize) { |                                  intptr_t addr, size_t addrsize) { | ||||||
|  | @ -66,8 +64,9 @@ static inline Elf64_Shdr *getelfsectionheaderaddress(const Elf64_Ehdr *elf, | ||||||
| 
 | 
 | ||||||
| static inline void *getelfsectionaddress(const Elf64_Ehdr *elf, size_t mapsize, | static inline void *getelfsectionaddress(const Elf64_Ehdr *elf, size_t mapsize, | ||||||
|                                          const Elf64_Shdr *shdr) { |                                          const Elf64_Shdr *shdr) { | ||||||
|   intptr_t addr = (intptr_t)elf + (intptr_t)shdr->sh_offset; |   intptr_t addr, size; | ||||||
|   intptr_t size = (intptr_t)shdr->sh_size; |   addr = (intptr_t)elf + (intptr_t)shdr->sh_offset; | ||||||
|  |   size = (intptr_t)shdr->sh_size; | ||||||
|   checkelfaddress(elf, mapsize, addr, size); |   checkelfaddress(elf, mapsize, addr, size); | ||||||
|   return (void *)addr; |   return (void *)addr; | ||||||
| } | } | ||||||
|  | @ -83,13 +82,16 @@ static inline void getelfvirtualaddressrange(const Elf64_Ehdr *elf, | ||||||
|                                              size_t elfsize, |                                              size_t elfsize, | ||||||
|                                              intptr_t *out_start, |                                              intptr_t *out_start, | ||||||
|                                              intptr_t *out_end) { |                                              intptr_t *out_end) { | ||||||
|   intptr_t start = INTPTR_MAX; |   unsigned i; | ||||||
|   intptr_t end = 0; |   Elf64_Phdr *phdr; | ||||||
|   for (unsigned i = 0; i < elf->e_phnum; ++i) { |   intptr_t start, end, pstart, pend; | ||||||
|     Elf64_Phdr *phdr = getelfsegmentheaderaddress(elf, elfsize, i); |   start = INTPTR_MAX; | ||||||
|  |   end = 0; | ||||||
|  |   for (i = 0; i < elf->e_phnum; ++i) { | ||||||
|  |     phdr = getelfsegmentheaderaddress(elf, elfsize, i); | ||||||
|     if (phdr->p_type != PT_LOAD) continue; |     if (phdr->p_type != PT_LOAD) continue; | ||||||
|     intptr_t pstart = phdr->p_vaddr; |     pstart = phdr->p_vaddr; | ||||||
|     intptr_t pend = phdr->p_vaddr + phdr->p_memsz; |     pend = phdr->p_vaddr + phdr->p_memsz; | ||||||
|     if (pstart < start) start = pstart; |     if (pstart < start) start = pstart; | ||||||
|     if (pend > end) end = pend; |     if (pend > end) end = pend; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -23,9 +23,10 @@ | ||||||
| Elf64_Shdr *getelfsectionbyaddress(const Elf64_Ehdr *elf, size_t mapsize, | Elf64_Shdr *getelfsectionbyaddress(const Elf64_Ehdr *elf, size_t mapsize, | ||||||
|                                    void *addr) { |                                    void *addr) { | ||||||
|   Elf64_Half i; |   Elf64_Half i; | ||||||
|  |   Elf64_Shdr *shdr; | ||||||
|   if (elf) { |   if (elf) { | ||||||
|     for (i = elf->e_shnum; i > 0; --i) { |     for (i = elf->e_shnum; i > 0; --i) { | ||||||
|       Elf64_Shdr *shdr = getelfsectionheaderaddress(elf, mapsize, i - 1); |       shdr = getelfsectionheaderaddress(elf, mapsize, i - 1); | ||||||
|       if ((intptr_t)addr >= shdr->sh_addr && |       if ((intptr_t)addr >= shdr->sh_addr && | ||||||
|           (intptr_t)addr < shdr->sh_addr + shdr->sh_size) { |           (intptr_t)addr < shdr->sh_addr + shdr->sh_size) { | ||||||
|         return shdr; |         return shdr; | ||||||
|  |  | ||||||
|  | @ -7,20 +7,15 @@ COSMOPOLITAN_C_START_ | ||||||
| │ cosmopolitan § escaping                                                  ─╬─│┼ | │ cosmopolitan § escaping                                                  ─╬─│┼ | ||||||
| ╚────────────────────────────────────────────────────────────────────────────│*/ | ╚────────────────────────────────────────────────────────────────────────────│*/ | ||||||
| 
 | 
 | ||||||
| unsigned cescapec(int c); | unsigned cescapec(int); | ||||||
| int escapec(char *buf, unsigned size, const char *unescaped, unsigned length) | int escapec(char *, unsigned, const char *, unsigned) | ||||||
|     paramsnonnull((3)) nothrow nocallback; |     paramsnonnull((3)) nothrow nocallback; | ||||||
| int escapesh(char *buf, unsigned size, const char *unescaped, unsigned length) | int escapesh(char *, unsigned, const char *, unsigned) paramsnonnull((3)); | ||||||
|     paramsnonnull((3)); | bool escapedos(char16_t *, unsigned, const char16_t *, unsigned); | ||||||
| bool escapedos(char16_t *buffer, unsigned buflen, const char16_t *unquoted, | int aescapec(char **, const char *, unsigned) paramsnonnull(); | ||||||
|                unsigned len); | int aescapesh(char **, const char *, unsigned) paramsnonnull(); | ||||||
| int aescapec(char **escaped, const char *unescaped, unsigned length) | int aescape(char **, size_t, const char *, unsigned, | ||||||
|     paramsnonnull(); |             int (*)(char *, unsigned, const char *, unsigned)) hidden; | ||||||
| int aescapesh(char **escaped, const char *unescaped, unsigned length) |  | ||||||
|     paramsnonnull(); |  | ||||||
| int aescape(char **escaped, size_t size, const char *unescaped, unsigned length, |  | ||||||
|             int impl(char *escaped, unsigned size, const char *unescaped, |  | ||||||
|                      unsigned length)) hidden; |  | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  |  | ||||||
|  | @ -20,11 +20,6 @@ | ||||||
| #include "libc/escape/escape.h" | #include "libc/escape/escape.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| 
 | 
 | ||||||
| asm(".ident\t\"\\n\\n\
 |  | ||||||
| NSSM (Public Domain)\\n\ |  | ||||||
| Credit: Iain Patterson\n\ |  | ||||||
| http://iain.cx/\"");
 |  | ||||||
| 
 |  | ||||||
| static textwindows bool shouldescapedos(const char16_t c) { | static textwindows bool shouldescapedos(const char16_t c) { | ||||||
|   if (c == u'"') return true; |   if (c == u'"') return true; | ||||||
|   if (c == u'&') return true; |   if (c == u'&') return true; | ||||||
|  | @ -48,6 +43,7 @@ static textwindows bool shouldquotedos(const char16_t c) { | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Escapes command so DOS can run it. |  * Escapes command so DOS can run it. | ||||||
|  |  * @see Iain Patterson's NSSM for original code in public domain | ||||||
|  */ |  */ | ||||||
| textwindows bool escapedos(char16_t *buffer, unsigned buflen, | textwindows bool escapedos(char16_t *buffer, unsigned buflen, | ||||||
|                            const char16_t *unquoted, unsigned len) { |                            const char16_t *unquoted, unsigned len) { | ||||||
|  |  | ||||||
|  | @ -170,138 +170,138 @@ kErrnoNames: | ||||||
| /	@assume	linker invoked as LC_ALL=C ld ...
 | /	@assume	linker invoked as LC_ALL=C ld ...
 | ||||||
| /	@see	libc/sysv/systemfive.S
 | /	@see	libc/sysv/systemfive.S
 | ||||||
| /	@see	libc/sysv/consts/syscon.h
 | /	@see	libc/sysv/consts/syscon.h
 | ||||||
| 	yoink	E2BIG | 	.yoink	E2BIG
 | ||||||
| 	yoink	EACCES | 	.yoink	EACCES
 | ||||||
| 	yoink	EADDRINUSE | 	.yoink	EADDRINUSE
 | ||||||
| 	yoink	EADDRNOTAVAIL | 	.yoink	EADDRNOTAVAIL
 | ||||||
| 	yoink	EADV | 	.yoink	EADV
 | ||||||
| 	yoink	EAFNOSUPPORT | 	.yoink	EAFNOSUPPORT
 | ||||||
| 	yoink	EAGAIN | 	.yoink	EAGAIN
 | ||||||
| 	yoink	EALREADY | 	.yoink	EALREADY
 | ||||||
| 	yoink	EBADE | 	.yoink	EBADE
 | ||||||
| 	yoink	EBADF | 	.yoink	EBADF
 | ||||||
| 	yoink	EBADFD | 	.yoink	EBADFD
 | ||||||
| 	yoink	EBADMSG | 	.yoink	EBADMSG
 | ||||||
| 	yoink	EBADR | 	.yoink	EBADR
 | ||||||
| 	yoink	EBADRQC | 	.yoink	EBADRQC
 | ||||||
| 	yoink	EBADSLT | 	.yoink	EBADSLT
 | ||||||
| 	yoink	EBFONT | 	.yoink	EBFONT
 | ||||||
| 	yoink	EBUSY | 	.yoink	EBUSY
 | ||||||
| 	yoink	ECANCELED | 	.yoink	ECANCELED
 | ||||||
| 	yoink	ECHILD | 	.yoink	ECHILD
 | ||||||
| 	yoink	ECHRNG | 	.yoink	ECHRNG
 | ||||||
| 	yoink	ECOMM | 	.yoink	ECOMM
 | ||||||
| 	yoink	ECONNABORTED | 	.yoink	ECONNABORTED
 | ||||||
| 	yoink	ECONNREFUSED | 	.yoink	ECONNREFUSED
 | ||||||
| 	yoink	ECONNRESET | 	.yoink	ECONNRESET
 | ||||||
| 	yoink	EDEADLK | 	.yoink	EDEADLK
 | ||||||
| 	yoink	EDESTADDRREQ | 	.yoink	EDESTADDRREQ
 | ||||||
| 	yoink	EDOM | 	.yoink	EDOM
 | ||||||
| 	yoink	EDOTDOT | 	.yoink	EDOTDOT
 | ||||||
| 	yoink	EDQUOT | 	.yoink	EDQUOT
 | ||||||
| 	yoink	EEXIST | 	.yoink	EEXIST
 | ||||||
| 	yoink	EFAULT | 	.yoink	EFAULT
 | ||||||
| 	yoink	EFBIG | 	.yoink	EFBIG
 | ||||||
| 	yoink	EHOSTDOWN | 	.yoink	EHOSTDOWN
 | ||||||
| 	yoink	EHOSTUNREACH | 	.yoink	EHOSTUNREACH
 | ||||||
| 	yoink	EHWPOISON | 	.yoink	EHWPOISON
 | ||||||
| 	yoink	EIDRM | 	.yoink	EIDRM
 | ||||||
| 	yoink	EILSEQ | 	.yoink	EILSEQ
 | ||||||
| 	yoink	EINPROGRESS | 	.yoink	EINPROGRESS
 | ||||||
| 	yoink	EINTR | 	.yoink	EINTR
 | ||||||
| 	yoink	EINVAL | 	.yoink	EINVAL
 | ||||||
| 	yoink	EIO | 	.yoink	EIO
 | ||||||
| 	yoink	EISCONN | 	.yoink	EISCONN
 | ||||||
| 	yoink	EISDIR | 	.yoink	EISDIR
 | ||||||
| 	yoink	EISNAM | 	.yoink	EISNAM
 | ||||||
| 	yoink	EKEYEXPIRED | 	.yoink	EKEYEXPIRED
 | ||||||
| 	yoink	EKEYREJECTED | 	.yoink	EKEYREJECTED
 | ||||||
| 	yoink	EKEYREVOKED | 	.yoink	EKEYREVOKED
 | ||||||
| 	yoink	EL2HLT | 	.yoink	EL2HLT
 | ||||||
| 	yoink	EL2NSYNC | 	.yoink	EL2NSYNC
 | ||||||
| 	yoink	EL3HLT | 	.yoink	EL3HLT
 | ||||||
| 	yoink	EL3RST | 	.yoink	EL3RST
 | ||||||
| 	yoink	ELIBACC | 	.yoink	ELIBACC
 | ||||||
| 	yoink	ELIBBAD | 	.yoink	ELIBBAD
 | ||||||
| 	yoink	ELIBEXEC | 	.yoink	ELIBEXEC
 | ||||||
| 	yoink	ELIBMAX | 	.yoink	ELIBMAX
 | ||||||
| 	yoink	ELIBSCN | 	.yoink	ELIBSCN
 | ||||||
| 	yoink	ELNRNG | 	.yoink	ELNRNG
 | ||||||
| 	yoink	ELOOP | 	.yoink	ELOOP
 | ||||||
| 	yoink	EMEDIUMTYPE | 	.yoink	EMEDIUMTYPE
 | ||||||
| 	yoink	EMFILE | 	.yoink	EMFILE
 | ||||||
| 	yoink	EMLINK | 	.yoink	EMLINK
 | ||||||
| 	yoink	EMSGSIZE | 	.yoink	EMSGSIZE
 | ||||||
| 	yoink	EMULTIHOP | 	.yoink	EMULTIHOP
 | ||||||
| 	yoink	ENAMETOOLONG | 	.yoink	ENAMETOOLONG
 | ||||||
| 	yoink	ENAVAIL | 	.yoink	ENAVAIL
 | ||||||
| 	yoink	ENETDOWN | 	.yoink	ENETDOWN
 | ||||||
| 	yoink	ENETRESET | 	.yoink	ENETRESET
 | ||||||
| 	yoink	ENETUNREACH | 	.yoink	ENETUNREACH
 | ||||||
| 	yoink	ENFILE | 	.yoink	ENFILE
 | ||||||
| 	yoink	ENOANO | 	.yoink	ENOANO
 | ||||||
| 	yoink	ENOBUFS | 	.yoink	ENOBUFS
 | ||||||
| 	yoink	ENOCSI | 	.yoink	ENOCSI
 | ||||||
| 	yoink	ENODATA | 	.yoink	ENODATA
 | ||||||
| 	yoink	ENODEV | 	.yoink	ENODEV
 | ||||||
| 	yoink	ENOENT | 	.yoink	ENOENT
 | ||||||
| 	yoink	ENOEXEC | 	.yoink	ENOEXEC
 | ||||||
| 	yoink	ENOKEY | 	.yoink	ENOKEY
 | ||||||
| 	yoink	ENOLCK | 	.yoink	ENOLCK
 | ||||||
| 	yoink	ENOLINK | 	.yoink	ENOLINK
 | ||||||
| 	yoink	ENOMEDIUM | 	.yoink	ENOMEDIUM
 | ||||||
| 	yoink	ENOMEM | 	.yoink	ENOMEM
 | ||||||
| 	yoink	ENOMSG | 	.yoink	ENOMSG
 | ||||||
| 	yoink	ENONET | 	.yoink	ENONET
 | ||||||
| 	yoink	ENOPKG | 	.yoink	ENOPKG
 | ||||||
| 	yoink	ENOPROTOOPT | 	.yoink	ENOPROTOOPT
 | ||||||
| 	yoink	ENOSPC | 	.yoink	ENOSPC
 | ||||||
| 	yoink	ENOSR | 	.yoink	ENOSR
 | ||||||
| 	yoink	ENOSTR | 	.yoink	ENOSTR
 | ||||||
| 	yoink	ENOSYS | 	.yoink	ENOSYS
 | ||||||
| 	yoink	ENOTBLK | 	.yoink	ENOTBLK
 | ||||||
| 	yoink	ENOTCONN | 	.yoink	ENOTCONN
 | ||||||
| 	yoink	ENOTDIR | 	.yoink	ENOTDIR
 | ||||||
| 	yoink	ENOTEMPTY | 	.yoink	ENOTEMPTY
 | ||||||
| 	yoink	ENOTNAM | 	.yoink	ENOTNAM
 | ||||||
| 	yoink	ENOTRECOVERABLE | 	.yoink	ENOTRECOVERABLE
 | ||||||
| 	yoink	ENOTSOCK | 	.yoink	ENOTSOCK
 | ||||||
| 	yoink	ENOTSUP | 	.yoink	ENOTSUP
 | ||||||
| 	yoink	ENOTTY | 	.yoink	ENOTTY
 | ||||||
| 	yoink	ENOTUNIQ | 	.yoink	ENOTUNIQ
 | ||||||
| 	yoink	ENXIO | 	.yoink	ENXIO
 | ||||||
| 	yoink	EOPNOTSUPP | 	.yoink	EOPNOTSUPP
 | ||||||
| 	yoink	EOVERFLOW | 	.yoink	EOVERFLOW
 | ||||||
| 	yoink	EOWNERDEAD | 	.yoink	EOWNERDEAD
 | ||||||
| 	yoink	EPERM | 	.yoink	EPERM
 | ||||||
| 	yoink	EPFNOSUPPORT | 	.yoink	EPFNOSUPPORT
 | ||||||
| 	yoink	EPIPE | 	.yoink	EPIPE
 | ||||||
| 	yoink	EPROTO | 	.yoink	EPROTO
 | ||||||
| 	yoink	EPROTONOSUPPORT | 	.yoink	EPROTONOSUPPORT
 | ||||||
| 	yoink	EPROTOTYPE | 	.yoink	EPROTOTYPE
 | ||||||
| 	yoink	ERANGE | 	.yoink	ERANGE
 | ||||||
| 	yoink	EREMCHG | 	.yoink	EREMCHG
 | ||||||
| 	yoink	EREMOTE | 	.yoink	EREMOTE
 | ||||||
| 	yoink	EREMOTEIO | 	.yoink	EREMOTEIO
 | ||||||
| 	yoink	ERESTART | 	.yoink	ERESTART
 | ||||||
| 	yoink	ERFKILL | 	.yoink	ERFKILL
 | ||||||
| 	yoink	EROFS | 	.yoink	EROFS
 | ||||||
| 	yoink	ESHUTDOWN | 	.yoink	ESHUTDOWN
 | ||||||
| 	yoink	ESOCKTNOSUPPORT | 	.yoink	ESOCKTNOSUPPORT
 | ||||||
| 	yoink	ESPIPE | 	.yoink	ESPIPE
 | ||||||
| 	yoink	ESRCH | 	.yoink	ESRCH
 | ||||||
| 	yoink	ESRMNT | 	.yoink	ESRMNT
 | ||||||
| 	yoink	ESTALE | 	.yoink	ESTALE
 | ||||||
| 	yoink	ESTRPIPE | 	.yoink	ESTRPIPE
 | ||||||
| 	yoink	ETIME | 	.yoink	ETIME
 | ||||||
| 	yoink	ETIMEDOUT | 	.yoink	ETIMEDOUT
 | ||||||
| 	yoink	ETOOMANYREFS | 	.yoink	ETOOMANYREFS
 | ||||||
| 	yoink	ETXTBSY | 	.yoink	ETXTBSY
 | ||||||
| 	yoink	EUCLEAN | 	.yoink	EUCLEAN
 | ||||||
| 	yoink	EUNATCH | 	.yoink	EUNATCH
 | ||||||
| 	yoink	EUSERS | 	.yoink	EUSERS
 | ||||||
| 	yoink	EXDEV | 	.yoink	EXDEV
 | ||||||
| 	yoink	EXFULL | 	.yoink	EXFULL
 | ||||||
| 
 | 
 | ||||||
| 	.type	kErrnoStart,@object
 | 	.type	kErrnoStart,@object
 | ||||||
| 	.type	kErrnoEnd,@object
 | 	.type	kErrnoEnd,@object
 | ||||||
|  |  | ||||||
|  | @ -37,6 +37,12 @@ | ||||||
| static const int kPow10[] = {1,      10,      100,      1000,      10000, | static const int kPow10[] = {1,      10,      100,      1000,      10000, | ||||||
|                              100000, 1000000, 10000000, 100000000, 1000000000}; |                              100000, 1000000, 10000000, 100000000, 1000000000}; | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Formats floating point number. | ||||||
|  |  * | ||||||
|  |  * @see xdtoa() for higher precision at the cost of bloat | ||||||
|  |  * @see palandprintf() which is intended caller | ||||||
|  |  */ | ||||||
| int ftoa(int out(int, void *), void *arg, long double value, unsigned long prec, | int ftoa(int out(int, void *), void *arg, long double value, unsigned long prec, | ||||||
|          unsigned long width, unsigned long flags) { |          unsigned long width, unsigned long flags) { | ||||||
|   long whole, frac; |   long whole, frac; | ||||||
|  |  | ||||||
|  | @ -56,7 +56,52 @@ static unsigned ppatoi(const char **str) { | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Implements {,v}{,s{,n},{,{,x}as},f,d}printf state machine. |  * Implements {,v}{,s{,n},{,{,x}as},f,d}printf state machine. | ||||||
|  * @see libc/stdio/printf.c for documentation |  * | ||||||
|  |  * Type Specifiers | ||||||
|  |  * | ||||||
|  |  *   - `%s`   char * (thompson-pike unicode) | ||||||
|  |  *   - `%ls`  wchar_t * (32-bit unicode → thompson-pike unicode) | ||||||
|  |  *   - `%hs`  char16_t * (16-bit unicode → thompson-pike unicode) | ||||||
|  |  *   - `%b`   int (radix 2 binary) | ||||||
|  |  *   - `%o`   int (radix 8 octal) | ||||||
|  |  *   - `%d`   int (radix 10 decimal) | ||||||
|  |  *   - `%x`   int (radix 16 hexadecimal) | ||||||
|  |  *   - `%X`   int (radix 16 hexadecimal uppercase) | ||||||
|  |  *   - `%u`   unsigned | ||||||
|  |  *   - `%f`   double | ||||||
|  |  *   - `%Lf`  long double | ||||||
|  |  *   - `%p`   pointer (48-bit hexadecimal) | ||||||
|  |  * | ||||||
|  |  * Length Modifiers | ||||||
|  |  * | ||||||
|  |  *   - `%hhd` char (8-bit) | ||||||
|  |  *   - `%hd`  short (16-bit) | ||||||
|  |  *   - `%ld`  long (64-bit) | ||||||
|  |  *   - `%lu`  unsigned long (64-bit) | ||||||
|  |  *   - `%lx`  unsigned long (64-bit hexadecimal) | ||||||
|  |  *   - `%jd`  intmax_t (128-bit) | ||||||
|  |  * | ||||||
|  |  * Width Modifiers | ||||||
|  |  * | ||||||
|  |  *   - `%08d` fixed columns w/ zero leftpadding | ||||||
|  |  *   - `%8d`  fixed columns w/ space leftpadding | ||||||
|  |  *   - `%*s`  variable column string (thompson-pike) | ||||||
|  |  *   - `%.*s` variable column data (ignore nul terminator) | ||||||
|  |  * | ||||||
|  |  * Formatting Modifiers | ||||||
|  |  * | ||||||
|  |  *   - `%,d`  thousands separators | ||||||
|  |  *   - `%'s`  escaped c string literal | ||||||
|  |  *   - `%'c`  escaped c character literal | ||||||
|  |  *   - `%`s`  escaped double quoted c string literal | ||||||
|  |  *   - `%`c`  escaped double quoted c character literal | ||||||
|  |  *   - `%+d`  plus leftpad if positive (aligns w/ negatives) | ||||||
|  |  *   - `% d`  space leftpad if positive (aligns w/ negatives) | ||||||
|  |  *   - `%#s`  datum (radix 256 null-terminated ibm cp437) | ||||||
|  |  *   - `%#x`  int (radix 16 hexadecimal w/ 0x prefix if not zero) | ||||||
|  |  * | ||||||
|  |  * @note implementation detail of printf(), snprintf(), etc. | ||||||
|  |  * @see printf() for wordier documentation | ||||||
|  */ |  */ | ||||||
| hidden int palandprintf(void *fn, void *arg, const char *format, va_list va) { | hidden int palandprintf(void *fn, void *arg, const char *format, va_list va) { | ||||||
|   int (*out)(int, void *); |   int (*out)(int, void *); | ||||||
|  |  | ||||||
|  | @ -50,8 +50,18 @@ | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #if defined(__GNUC__) && __GNUC__ < 6 | #if defined(__GNUC__) && __GNUC__ < 6 | ||||||
|  | /*
 | ||||||
|  |  * Compilers don't understand the features we've added to the format | ||||||
|  |  * string DSL, such as c string escaping, therefore we can't use it. | ||||||
|  |  * Ideally compilers should grant us more flexibility to define DSLs | ||||||
|  |  * | ||||||
|  |  * The recommended approach to turning this back on is `CFLAGS=-std=c11` | ||||||
|  |  * which puts the compiler in __STRICT_ANSI__ mode, which Cosmopolitan | ||||||
|  |  * respects by disabling all the esoteric tuning in headers like this. | ||||||
|  |  */ | ||||||
| #pragma GCC diagnostic ignored "-Wformat-security" | #pragma GCC diagnostic ignored "-Wformat-security" | ||||||
| #endif /* __GNUC__ + 0 < 6 */ | #endif /* __GNUC__ + 0 < 6 */ | ||||||
|  | 
 | ||||||
| #else | #else | ||||||
| #define PFLINK(FMT) FMT | #define PFLINK(FMT) FMT | ||||||
| #define SFLINK(FMT) FMT | #define SFLINK(FMT) FMT | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ | ||||||
|  * @return number of bytes written, excluding the NUL terminator; or, |  * @return number of bytes written, excluding the NUL terminator; or, | ||||||
|  *     if the output buffer wasn't passed, or was too short, then the |  *     if the output buffer wasn't passed, or was too short, then the | ||||||
|  *     number of characters that *would* have been written is returned |  *     number of characters that *would* have been written is returned | ||||||
|  * @see libc/fmt/fmt.h for documentation |  * @see palandprintf() and printf() for detailed documentation | ||||||
|  */ |  */ | ||||||
| int(snprintf)(char* buf, size_t count, const char* fmt, ...) { | int(snprintf)(char* buf, size_t count, const char* fmt, ...) { | ||||||
|   int rc; |   int rc; | ||||||
|  |  | ||||||
|  | @ -20,6 +20,12 @@ | ||||||
| #include "libc/fmt/fmt.h" | #include "libc/fmt/fmt.h" | ||||||
| #include "libc/limits.h" | #include "libc/limits.h" | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Formats string to buffer that's hopefully large enough. | ||||||
|  |  * | ||||||
|  |  * @see palandprintf() and printf() for detailed documentation | ||||||
|  |  * @see snprintf() for same w/ buf size param | ||||||
|  |  */ | ||||||
| int(sprintf)(char *buf, const char *fmt, ...) { | int(sprintf)(char *buf, const char *fmt, ...) { | ||||||
|   int rc; |   int rc; | ||||||
|   va_list va; |   va_list va; | ||||||
|  |  | ||||||
|  | @ -74,6 +74,8 @@ forceinline int emitquote(int out(int, void *), void *arg, unsigned flags, | ||||||
|  * This function is used by palandprintf() to implement the %s and %c |  * This function is used by palandprintf() to implement the %s and %c | ||||||
|  * directives. The content outputted to the array is always UTF-8, but |  * directives. The content outputted to the array is always UTF-8, but | ||||||
|  * the input may be UTF-16 or UTF-32. |  * the input may be UTF-16 or UTF-32. | ||||||
|  |  * | ||||||
|  |  * @see palandprintf() | ||||||
|  */ |  */ | ||||||
| int stoa(int out(int, void *), void *arg, void *data, unsigned long flags, | int stoa(int out(int, void *), void *arg, void *data, unsigned long flags, | ||||||
|          unsigned long precision, unsigned long width, unsigned char signbit, |          unsigned long precision, unsigned long width, unsigned char signbit, | ||||||
|  |  | ||||||
|  | @ -34,6 +34,9 @@ | ||||||
|  * 8-bit through 128-bit integers (with validation), floating point |  * 8-bit through 128-bit integers (with validation), floating point | ||||||
|  * numbers, etc. It can also be used to convert UTF-8 to UTF-16/32. |  * numbers, etc. It can also be used to convert UTF-8 to UTF-16/32. | ||||||
|  * |  * | ||||||
|  |  *   - `%d`  parses integer | ||||||
|  |  *   - `%ms` parses string allocating buffer assigning pointer | ||||||
|  |  * | ||||||
|  * @param callback supplies UTF-8 characters using -1 sentinel |  * @param callback supplies UTF-8 characters using -1 sentinel | ||||||
|  * @param fmt is a computer program embedded inside a c string, written |  * @param fmt is a computer program embedded inside a c string, written | ||||||
|  *     in a domain-specific programming language that, by design, lacks |  *     in a domain-specific programming language that, by design, lacks | ||||||
|  |  | ||||||
|  | @ -40,7 +40,7 @@ static int vsnprintfputchar(unsigned char c, struct SprintfStr *str) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Formats string to buffer. |  * Formats string to buffer w/ preexisting vararg state. | ||||||
|  * |  * | ||||||
|  * @param buf stores output and a NUL-terminator is always written, |  * @param buf stores output and a NUL-terminator is always written, | ||||||
|  *     provided buf!=NULL && size!=0 |  *     provided buf!=NULL && size!=0 | ||||||
|  | @ -50,7 +50,7 @@ static int vsnprintfputchar(unsigned char c, struct SprintfStr *str) { | ||||||
|  *     number of characters that *would* have been written is returned |  *     number of characters that *would* have been written is returned | ||||||
|  * @throw EOVERFLOW when a formatted field exceeds its limit, which can |  * @throw EOVERFLOW when a formatted field exceeds its limit, which can | ||||||
|  *     be checked by setting errno to 0 before calling |  *     be checked by setting errno to 0 before calling | ||||||
|  * @see libc/fmt/fmt.h for further documentation |  * @see palandprintf() and printf() for detailed documentation | ||||||
|  */ |  */ | ||||||
| int(vsnprintf)(char *buf, size_t size, const char *fmt, va_list va) { | int(vsnprintf)(char *buf, size_t size, const char *fmt, va_list va) { | ||||||
|   struct SprintfStr str = {buf, 0, size}; |   struct SprintfStr str = {buf, 0, size}; | ||||||
|  |  | ||||||
|  | @ -20,6 +20,12 @@ | ||||||
| #include "libc/fmt/fmt.h" | #include "libc/fmt/fmt.h" | ||||||
| #include "libc/limits.h" | #include "libc/limits.h" | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Formats string to buffer hopefully large enough w/ vararg state. | ||||||
|  |  * | ||||||
|  |  * @see palandprintf() and printf() for detailed documentation | ||||||
|  |  * @see vsnprintf() for modern alternative w/ buf size param | ||||||
|  |  */ | ||||||
| int(vsprintf)(char *buf, const char *fmt, va_list va) { | int(vsprintf)(char *buf, const char *fmt, va_list va) { | ||||||
|   return (vsnprintf)(buf, INT_MAX, fmt, va); |   return (vsnprintf)(buf, INT_MAX, fmt, va); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -21,7 +21,7 @@ | ||||||
| #include "libc/fmt/vsscanf.h" | #include "libc/fmt/vsscanf.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * String decoder. |  * Decodes string. | ||||||
|  * |  * | ||||||
|  * @see libc/fmt/vcscanf.h (for docs and implementation) |  * @see libc/fmt/vcscanf.h (for docs and implementation) | ||||||
|  * @note to offer the best performance, we assume small codebases |  * @note to offer the best performance, we assume small codebases | ||||||
|  |  | ||||||
|  | @ -1015,8 +1015,9 @@ typedef uint64_t uintmax_t; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Pulls source of object being compiled into zip. |  * Pulls source of object being compiled into zip. | ||||||
|  * @note automates compliance with gpl terms |  * @note automates most compliance with gpl terms | ||||||
|  * @see -DIM_FEELING_NAUGHTY |  * @see libc/zipos/zipcentraldir.S | ||||||
|  |  * @see ape/ape.lds | ||||||
|  */ |  */ | ||||||
| #ifdef __BASE_FILE__
 | #ifdef __BASE_FILE__
 | ||||||
| STATIC_YOINK_SOURCE(__BASE_FILE__); | STATIC_YOINK_SOURCE(__BASE_FILE__); | ||||||
|  |  | ||||||
|  | @ -50,8 +50,7 @@ void exit(int) noreturn; | ||||||
| void quick_exit(int) noreturn; | void quick_exit(int) noreturn; | ||||||
| void _exit(int) libcesque noreturn; | void _exit(int) libcesque noreturn; | ||||||
| void _Exit(int) libcesque noreturn; | void _Exit(int) libcesque noreturn; | ||||||
| void abort(void) noreturn noinstrument forcealignargpointer; | void abort(void) noreturn noinstrument; | ||||||
| void abort_(void) asm("abort") noreturn noinstrument privileged; |  | ||||||
| void panic(void) noreturn noinstrument privileged; | void panic(void) noreturn noinstrument privileged; | ||||||
| void triplf(void) noreturn noinstrument privileged; | void triplf(void) noreturn noinstrument privileged; | ||||||
| int __cxa_atexit(void *, void *, void *) libcesque; | int __cxa_atexit(void *, void *, void *) libcesque; | ||||||
|  |  | ||||||
|  | @ -44,7 +44,7 @@ size_t fread(void *buf, size_t stride, size_t count, FILE *f) { | ||||||
|       if (i % stride != 0) abort(); /* todo(jart) */ |       if (i % stride != 0) abort(); /* todo(jart) */ | ||||||
|       return i / stride; |       return i / stride; | ||||||
|     } |     } | ||||||
|     p[i] = (unsigned char)c; |     p[i] = c & 0xff; | ||||||
|   } |   } | ||||||
|   return count; |   return count; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -12,13 +12,13 @@ extern unsigned char g_stderrbuf[BUFSIZ]; | ||||||
| int fflushregister(FILE *) hidden; | int fflushregister(FILE *) hidden; | ||||||
| void fflushunregister(FILE *) hidden; | void fflushunregister(FILE *) hidden; | ||||||
| 
 | 
 | ||||||
| int freadbuf(FILE *f) hidden; | int freadbuf(FILE *) hidden; | ||||||
| int fwritebuf(FILE *f) hidden; | int fwritebuf(FILE *) hidden; | ||||||
| int fsreadbuf(FILE *f) hidden; | int fsreadbuf(FILE *) hidden; | ||||||
| int fswritebuf(FILE *f) hidden; | int fswritebuf(FILE *) hidden; | ||||||
| long fseteof(FILE *f) hidden; | long fseteof(FILE *) hidden; | ||||||
| long fseterrno(FILE *f) hidden; | long fseterrno(FILE *) hidden; | ||||||
| long fseterr(FILE *f, int err) hidden; | long fseterr(FILE *, int) hidden; | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  |  | ||||||
|  | @ -59,6 +59,7 @@ | ||||||
|  *   it implies the quoting modifier, wraps the value with {,u,L}['"] |  *   it implies the quoting modifier, wraps the value with {,u,L}['"] | ||||||
|  *   quotes, displays NULL as "NULL" rather than "(null)", etc. |  *   quotes, displays NULL as "NULL" rather than "(null)", etc. | ||||||
|  * |  * | ||||||
|  |  * @see palandprintf() for intuitive reference documentation | ||||||
|  * @see {,v}{,s{,n},{,{,x}as},f,d}printf |  * @see {,v}{,s{,n},{,{,x}as},f,d}printf | ||||||
|  */ |  */ | ||||||
| int(printf)(const char* fmt, ...) { | int(printf)(const char* fmt, ...) { | ||||||
|  |  | ||||||
|  | @ -4,29 +4,29 @@ | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
| COSMOPOLITAN_C_START_ | COSMOPOLITAN_C_START_ | ||||||
| 
 | 
 | ||||||
| int getc_unlocked(FILE *f) paramsnonnull(); | int getc_unlocked(FILE *) paramsnonnull(); | ||||||
| int getchar_unlocked(void); | int getchar_unlocked(void); | ||||||
| int putc_unlocked(int c, FILE *f) paramsnonnull(); | int putc_unlocked(int, FILE *) paramsnonnull(); | ||||||
| int putchar_unlocked(int c); | int putchar_unlocked(int); | ||||||
| void clearerr_unlocked(FILE *f); | void clearerr_unlocked(FILE *); | ||||||
| int feof_unlocked(FILE *f); | int feof_unlocked(FILE *); | ||||||
| int ferror_unlocked(FILE *f); | int ferror_unlocked(FILE *); | ||||||
| int fileno_unlocked(FILE *f); | int fileno_unlocked(FILE *); | ||||||
| int fflush_unlocked(FILE *f); | int fflush_unlocked(FILE *); | ||||||
| int fgetc_unlocked(FILE *f); | int fgetc_unlocked(FILE *); | ||||||
| int fputc_unlocked(int c, FILE *f); | int fputc_unlocked(int, FILE *); | ||||||
| size_t fread_unlocked(void *ptr, size_t size, size_t n, FILE *f); | size_t fread_unlocked(void *, size_t, size_t, FILE *); | ||||||
| size_t fwrite_unlocked(const void *ptr, size_t size, size_t n, FILE *f); | size_t fwrite_unlocked(const void *, size_t, size_t, FILE *); | ||||||
| char *fgets_unlocked(char *s, int n, FILE *f); | char *fgets_unlocked(char *, int, FILE *); | ||||||
| int fputs_unlocked(const char *s, FILE *f); | int fputs_unlocked(const char *, FILE *); | ||||||
| wint_t getwc_unlocked(FILE *f); | wint_t getwc_unlocked(FILE *); | ||||||
| wint_t getwchar_unlocked(void); | wint_t getwchar_unlocked(void); | ||||||
| wint_t fgetwc_unlocked(FILE *f); | wint_t fgetwc_unlocked(FILE *); | ||||||
| wint_t fputwc_unlocked(wchar_t wc, FILE *f); | wint_t fputwc_unlocked(wchar_t, FILE *); | ||||||
| wint_t putwc_unlocked(wchar_t wc, FILE *f); | wint_t putwc_unlocked(wchar_t, FILE *); | ||||||
| wint_t putwchar_unlocked(wchar_t wc); | wint_t putwchar_unlocked(wchar_t); | ||||||
| wchar_t *fgetws_unlocked(wchar_t *ws, int n, FILE *f); | wchar_t *fgetws_unlocked(wchar_t *, int, FILE *); | ||||||
| int fputws_unlocked(const wchar_t *ws, FILE *f); | int fputws_unlocked(const wchar_t *, FILE *); | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| clearerr_unlocked: | clearerr_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	clearerr | 	call	clearerr | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	clearerr_unlocked,globl | 	.endfn	clearerr_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| feof_unlocked: | feof_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	feof | 	call	feof | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	feof_unlocked,globl | 	.endfn	feof_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| ferror_unlocked: | ferror_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	ferror | 	call	ferror | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	ferror_unlocked,globl | 	.endfn	ferror_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fflush_unlocked: | fflush_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fflush | 	call	fflush | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fflush_unlocked,globl | 	.endfn	fflush_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fgetc_unlocked: | fgetc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fgetc | 	call	fgetc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fgetc_unlocked,globl | 	.endfn	fgetc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fgets_unlocked: | fgets_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fgets | 	call	fgets | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fgets_unlocked,globl | 	.endfn	fgets_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fgetwc_unlocked: | fgetwc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fgetwc | 	call	fgetwc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fgetwc_unlocked,globl | 	.endfn	fgetwc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fgetws_unlocked: | fgetws_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fgetws | 	call	fgetws | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fgetws_unlocked,globl | 	.endfn	fgetws_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fileno_unlocked: | fileno_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fileno | 	call	fileno | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fileno_unlocked,globl | 	.endfn	fileno_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fputc_unlocked: | fputc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fputc | 	call	fputc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fputc_unlocked,globl | 	.endfn	fputc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fputs_unlocked: | fputs_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fputs | 	call	fputs | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fputs_unlocked,globl | 	.endfn	fputs_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fputwc_unlocked: | fputwc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fputwc | 	call	fputwc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fputwc_unlocked,globl | 	.endfn	fputwc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fputws_unlocked: | fputws_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fputws | 	call	fputws | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fputws_unlocked,globl | 	.endfn	fputws_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fread_unlocked: | fread_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fread | 	call	fread | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fread_unlocked,globl | 	.endfn	fread_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| fwrite_unlocked: | fwrite_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fwrite | 	call	fwrite | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	fwrite_unlocked,globl | 	.endfn	fwrite_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| getc_unlocked: | getc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	getc | 	call	getc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	getc_unlocked,globl | 	.endfn	getc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| getchar_unlocked: | getchar_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	getchar | 	call	getchar | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	getchar_unlocked,globl | 	.endfn	getchar_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| getwc_unlocked: | getwc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fgetwc_unlocked | 	call	fgetwc_unlocked | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	getwc_unlocked,globl | 	.endfn	getwc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| getwchar_unlocked: | getwchar_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	getwchar | 	call	getwchar | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	getwchar_unlocked,globl | 	.endfn	getwchar_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| putc_unlocked: | putc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	putc | 	call	putc | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	putc_unlocked,globl | 	.endfn	putc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| putchar_unlocked: | putchar_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	putchar | 	call	putchar | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	putchar_unlocked,globl | 	.endfn	putchar_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -21,8 +21,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| putwc_unlocked: | putwc_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	fputwc_unlocked | 	call	fputwc_unlocked | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	putwc_unlocked,globl | 	.endfn	putwc_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -22,8 +22,10 @@ | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
| 
 | 
 | ||||||
| putwchar_unlocked: | putwchar_unlocked: | ||||||
| 	.leafprologue | 	push	%rbp | ||||||
| 	.profilable		# so we can fix code supporting this abomination | 	mov	%rsp,%rbp | ||||||
|  | 	.profilable		# note: no consensus for threads exists in abis | ||||||
| 	call	putwchar | 	call	putwchar | ||||||
| 	.leafepilogue | 	pop	%rbp | ||||||
|  | 	ret | ||||||
| 	.endfn	putwchar_unlocked,globl | 	.endfn	putwchar_unlocked,globl | ||||||
|  |  | ||||||
|  | @ -1760,18 +1760,6 @@ syscon	misc	READ_POSITION				52			0			0			0			0 | ||||||
| syscon	misc	READ_REVERSE				15			0			0			0			0 | syscon	misc	READ_REVERSE				15			0			0			0			0 | ||||||
| syscon	misc	READ_TOC				67			0			0			0			0 | syscon	misc	READ_TOC				67			0			0			0			0 | ||||||
| 
 | 
 | ||||||
| syscon	glob	GLOB_NOCHECK				0x10			0x10			0x10			0x10			0			# unix consensus |  | ||||||
| syscon	glob	GLOB_ABORTED				2			-2			-2			-2			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_APPEND				0x20			1			1			1			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_DOOFFS				8			2			2			2			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_ERR				1			4			4			4			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_MARK				2			8			8			8			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_NOMATCH				3			-3			-3			-3			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_NOSORT				4			0x20			0x20			0x20			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_NOSPACE				1			-1			-1			-1			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_NOSYS				4			-4			-4			-4			0			# bsd consensus |  | ||||||
| syscon	glob	GLOB_NOESCAPE				0x40			0x2000			0x2000			0x1000			0 |  | ||||||
| 
 |  | ||||||
| #	getpriority() / setpriority() magnums (a.k.a. nice) | #	getpriority() / setpriority() magnums (a.k.a. nice) | ||||||
| # | # | ||||||
| #	group	name					GNU/Systemd		XNU's Not UNIX		FreeBSD			OpenBSD			XENIX			Commentary | #	group	name					GNU/Systemd		XNU's Not UNIX		FreeBSD			OpenBSD			XENIX			Commentary | ||||||
|  |  | ||||||
|  | @ -1,34 +0,0 @@ | ||||||
| #ifndef COSMOPOLITAN_LIBC_SYSV_CONSTS_GLOB_H_ |  | ||||||
| #define COSMOPOLITAN_LIBC_SYSV_CONSTS_GLOB_H_ |  | ||||||
| #include "libc/runtime/symbolic.h" |  | ||||||
| 
 |  | ||||||
| #define GLOB_ABORTED SYMBOLIC(GLOB_ABORTED) |  | ||||||
| #define GLOB_APPEND SYMBOLIC(GLOB_APPEND) |  | ||||||
| #define GLOB_DOOFFS SYMBOLIC(GLOB_DOOFFS) |  | ||||||
| #define GLOB_ERR SYMBOLIC(GLOB_ERR) |  | ||||||
| #define GLOB_MARK SYMBOLIC(GLOB_MARK) |  | ||||||
| #define GLOB_NOCHECK SYMBOLIC(GLOB_NOCHECK) |  | ||||||
| #define GLOB_NOESCAPE SYMBOLIC(GLOB_NOESCAPE) |  | ||||||
| #define GLOB_NOMATCH SYMBOLIC(GLOB_NOMATCH) |  | ||||||
| #define GLOB_NOSORT SYMBOLIC(GLOB_NOSORT) |  | ||||||
| #define GLOB_NOSPACE SYMBOLIC(GLOB_NOSPACE) |  | ||||||
| #define GLOB_NOSYS SYMBOLIC(GLOB_NOSYS) |  | ||||||
| 
 |  | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) |  | ||||||
| COSMOPOLITAN_C_START_ |  | ||||||
| 
 |  | ||||||
| hidden extern const long GLOB_ABORTED; |  | ||||||
| hidden extern const long GLOB_APPEND; |  | ||||||
| hidden extern const long GLOB_DOOFFS; |  | ||||||
| hidden extern const long GLOB_ERR; |  | ||||||
| hidden extern const long GLOB_MARK; |  | ||||||
| hidden extern const long GLOB_NOCHECK; |  | ||||||
| hidden extern const long GLOB_NOESCAPE; |  | ||||||
| hidden extern const long GLOB_NOMATCH; |  | ||||||
| hidden extern const long GLOB_NOSORT; |  | ||||||
| hidden extern const long GLOB_NOSPACE; |  | ||||||
| hidden extern const long GLOB_NOSYS; |  | ||||||
| 
 |  | ||||||
| COSMOPOLITAN_C_END_ |  | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ |  | ||||||
| #endif /* COSMOPOLITAN_LIBC_SYSV_CONSTS_GLOB_H_ */ |  | ||||||
|  | @ -60,6 +60,7 @@ void *xvalloc(size_t) attributeallocsize((1)) _XMALPG; | ||||||
| void *xmemalign(size_t, size_t) attributeallocalign((1)) | void *xmemalign(size_t, size_t) attributeallocalign((1)) | ||||||
|     attributeallocsize((2)) _XMAL; |     attributeallocsize((2)) _XMAL; | ||||||
| char *xstrdup(const char *) _XPNN _XMAL; | char *xstrdup(const char *) _XPNN _XMAL; | ||||||
|  | char *xstrndup(const char *, size_t) _XPNN _XMAL; | ||||||
| char *xstrcat(const char *, ...) paramsnonnull((1)) nullterminated() _XMAL; | char *xstrcat(const char *, ...) paramsnonnull((1)) nullterminated() _XMAL; | ||||||
| char *xstrmul(const char *, size_t) paramsnonnull((1)) _XMAL; | char *xstrmul(const char *, size_t) paramsnonnull((1)) _XMAL; | ||||||
| char *xinet_ntop(int, const void *) _XPNN _XMAL; | char *xinet_ntop(int, const void *) _XPNN _XMAL; | ||||||
|  |  | ||||||
							
								
								
									
										36
									
								
								libc/x/xstrndup.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								libc/x/xstrndup.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | ||||||
|  | /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||||
|  | │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||||
|  | ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||||
|  | │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ This program is free software; you can redistribute it and/or modify         │ | ||||||
|  | │ it under the terms of the GNU General Public License as published by         │ | ||||||
|  | │ the Free Software Foundation; version 2 of the License.                      │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ This program is distributed in the hope that it will be useful, but          │ | ||||||
|  | │ WITHOUT ANY WARRANTY; without even the implied warranty of                   │ | ||||||
|  | │ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU             │ | ||||||
|  | │ General Public License for more details.                                     │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ You should have received a copy of the GNU General Public License            │ | ||||||
|  | │ along with this program; if not, write to the Free Software                  │ | ||||||
|  | │ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||||
|  | │ 02110-1301 USA                                                               │ | ||||||
|  | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
|  | #include "libc/log/log.h" | ||||||
|  | #include "libc/mem/mem.h" | ||||||
|  | #include "libc/x/x.h" | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Allocates new copy of string, with byte limit. | ||||||
|  |  * | ||||||
|  |  * @param s is a NUL-terminated byte string | ||||||
|  |  * @param n if less than strlen(s) will truncate the string | ||||||
|  |  * @return new string or NULL w/ errno | ||||||
|  |  * @error ENOMEM | ||||||
|  |  */ | ||||||
|  | char *xstrndup(const char *s, size_t n) { | ||||||
|  |   void *res = strndup(s, n); | ||||||
|  |   if (!res) die(); | ||||||
|  |   return res; | ||||||
|  | } | ||||||
|  | @ -41,6 +41,7 @@ TEST(SUITE(sprintf), testCharacterCounting) { | ||||||
| TEST(SUITE(snprintf), testTableFlip) { | TEST(SUITE(snprintf), testTableFlip) { | ||||||
|   EXPECT_STREQ("Table flip          ", Format("%-20ls", L"Table flip")); |   EXPECT_STREQ("Table flip          ", Format("%-20ls", L"Table flip")); | ||||||
|   EXPECT_STREQ("(╯°□°)╯︵ ┻━┻       ", Format("%-20ls", L"(╯°□°)╯︵ ┻━┻")); |   EXPECT_STREQ("(╯°□°)╯︵ ┻━┻       ", Format("%-20ls", L"(╯°□°)╯︵ ┻━┻")); | ||||||
|  |   EXPECT_STREQ("(╯°□°)╯︵ ┻━┻       ", Format("%-20hs", u"(╯°□°)╯︵ ┻━┻")); | ||||||
|   EXPECT_STREQ("ちゃぶ台返し        ", Format("%-20ls", L"ちゃぶ台返し")); |   EXPECT_STREQ("ちゃぶ台返し        ", Format("%-20ls", L"ちゃぶ台返し")); | ||||||
|   EXPECT_STREQ("       (╯°□°)╯︵ ┻━┻", Format("%20ls", L"(╯°□°)╯︵ ┻━┻")); |   EXPECT_STREQ("       (╯°□°)╯︵ ┻━┻", Format("%20ls", L"(╯°□°)╯︵ ┻━┻")); | ||||||
|   EXPECT_STREQ("        ちゃぶ台返し", Format("%20ls", L"ちゃぶ台返し")); |   EXPECT_STREQ("        ちゃぶ台返し", Format("%20ls", L"ちゃぶ台返し")); | ||||||
|  |  | ||||||
|  | @ -27,8 +27,8 @@ | ||||||
|  * |  * | ||||||
|  * It is demonstrated that our 3.5kb x86 parser supports all legal x86 |  * It is demonstrated that our 3.5kb x86 parser supports all legal x86 | ||||||
|  * instruction set architectures and addressing modes since the 1970's, |  * instruction set architectures and addressing modes since the 1970's, | ||||||
|  * including the really complicated ones, e.g. avx512; or the unpopular |  * including the really sophisticated ones, e.g. avx512, as well as the | ||||||
|  * ones, e.g. amd 3dnow. |  * less popular ones, e.g. 3dnow. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| TEST(x86ild, testSomeThingsNeverChange) { | TEST(x86ild, testSomeThingsNeverChange) { | ||||||
|  |  | ||||||
							
								
								
									
										9
									
								
								third_party/avir/avir1.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								third_party/avir/avir1.h
									
										
									
									
										vendored
									
									
								
							|  | @ -7,11 +7,10 @@ struct avir1 { | ||||||
|   void *p; |   void *p; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| void avir1init(struct avir1 *self); | void avir1init(struct avir1 *); | ||||||
| void avir1free(struct avir1 *self); | void avir1free(struct avir1 *); | ||||||
| void avir1(struct avir1 *resizer, size_t dyn, size_t dxn, void *dst, | void avir1(struct avir1 *, size_t, size_t, void *, size_t, size_t, size_t, | ||||||
|            size_t dstsize, size_t syn, size_t sxn, size_t ssw, const void *src, |            size_t, const void *, size_t); | ||||||
|            size_t srcsize); |  | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								third_party/avir/avir_float4_sse.h
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								third_party/avir/avir_float4_sse.h
									
										
									
									
										vendored
									
									
								
							|  | @ -20,6 +20,7 @@ | ||||||
| #include "libc/bits/mmintrin.h" | #include "libc/bits/mmintrin.h" | ||||||
| #include "libc/bits/xmmintrin.h" | #include "libc/bits/xmmintrin.h" | ||||||
| #include "libc/bits/xmmintrin.h" | #include "libc/bits/xmmintrin.h" | ||||||
|  | #include "libc/bits/xmmintrin.h" | ||||||
| #include "libc/bits/emmintrin.h" | #include "libc/bits/emmintrin.h" | ||||||
| 
 | 
 | ||||||
| namespace avir { | namespace avir { | ||||||
|  |  | ||||||
							
								
								
									
										368
									
								
								third_party/musl/fnmatch.c
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										368
									
								
								third_party/musl/fnmatch.c
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,368 @@ | ||||||
|  | /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||||
|  | │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||||
|  | ╚──────────────────────────────────────────────────────────────────────────────╝ | ||||||
|  | │                                                                              │ | ||||||
|  | │  Musl Libc                                                                   │ | ||||||
|  | │  Copyright © 2005-2014 Rich Felker, et al.                                   │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  Permission is hereby granted, free of charge, to any person obtaining       │ | ||||||
|  | │  a copy of this software and associated documentation files (the             │ | ||||||
|  | │  "Software"), to deal in the Software without restriction, including         │ | ||||||
|  | │  without limitation the rights to use, copy, modify, merge, publish,         │ | ||||||
|  | │  distribute, sublicense, and/or sell copies of the Software, and to          │ | ||||||
|  | │  permit persons to whom the Software is furnished to do so, subject to       │ | ||||||
|  | │  the following conditions:                                                   │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  The above copyright notice and this permission notice shall be              │ | ||||||
|  | │  included in all copies or substantial portions of the Software.             │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,             │ | ||||||
|  | │  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF          │ | ||||||
|  | │  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.      │ | ||||||
|  | │  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY        │ | ||||||
|  | │  CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,        │ | ||||||
|  | │  TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE           │ | ||||||
|  | │  SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                      │ | ||||||
|  | │                                                                              │ | ||||||
|  | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
|  | #include "libc/limits.h" | ||||||
|  | #include "libc/str/str.h" | ||||||
|  | #include "third_party/musl/fnmatch.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * An implementation of what I call the "Sea of Stars" algorithm for | ||||||
|  |  * POSIX fnmatch(). The basic idea is that we factor the pattern into | ||||||
|  |  * a head component (which we match first and can reject without ever | ||||||
|  |  * measuring the length of the string), an optional tail component | ||||||
|  |  * (which only exists if the pattern contains at least one star), and | ||||||
|  |  * an optional "sea of stars", a set of star-separated components | ||||||
|  |  * between the head and tail. After the head and tail matches have | ||||||
|  |  * been removed from the input string, the components in the "sea of | ||||||
|  |  * stars" are matched sequentially by searching for their first | ||||||
|  |  * occurrence past the end of the previous match. | ||||||
|  |  * | ||||||
|  |  * - Rich Felker, April 2012 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | asm(".ident\t\"\\n\\n\
 | ||||||
|  | Musl libc (MIT License)\\n\ | ||||||
|  | Copyright 2005-2014 Rich Felker, et. al.\""); | ||||||
|  | asm(".include \"libc/disclaimer.inc\""); | ||||||
|  | 
 | ||||||
|  | #define END         0 | ||||||
|  | #define UNMATCHABLE -2 | ||||||
|  | #define BRACKET     -3 | ||||||
|  | #define QUESTION    -4 | ||||||
|  | #define STAR        -5 | ||||||
|  | 
 | ||||||
|  | static int FnmatchNextString(const char *str, size_t n, size_t *step) { | ||||||
|  |   if (!n) { | ||||||
|  |     *step = 0; | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   if (str[0] >= 128U) { | ||||||
|  |     wchar_t wc; | ||||||
|  |     int k = mbtowc(&wc, str, n); | ||||||
|  |     if (k < 0) { | ||||||
|  |       *step = 1; | ||||||
|  |       return -1; | ||||||
|  |     } | ||||||
|  |     *step = k; | ||||||
|  |     return wc; | ||||||
|  |   } | ||||||
|  |   *step = 1; | ||||||
|  |   return str[0]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int FnmatchNextPattern(const char *pat, size_t m, size_t *step, | ||||||
|  |                               int flags) { | ||||||
|  |   int esc = 0; | ||||||
|  |   if (!m || !*pat) { | ||||||
|  |     *step = 0; | ||||||
|  |     return END; | ||||||
|  |   } | ||||||
|  |   *step = 1; | ||||||
|  |   if (pat[0] == '\\' && pat[1] && !(flags & FNM_NOESCAPE)) { | ||||||
|  |     *step = 2; | ||||||
|  |     pat++; | ||||||
|  |     esc = 1; | ||||||
|  |     goto escaped; | ||||||
|  |   } | ||||||
|  |   if (pat[0] == '[') { | ||||||
|  |     size_t k = 1; | ||||||
|  |     if (k < m) | ||||||
|  |       if (pat[k] == '^' || pat[k] == '!') k++; | ||||||
|  |     if (k < m) | ||||||
|  |       if (pat[k] == ']') k++; | ||||||
|  |     for (; k < m && pat[k] && pat[k] != ']'; k++) { | ||||||
|  |       if (k + 1 < m && pat[k + 1] && pat[k] == '[' && | ||||||
|  |           (pat[k + 1] == ':' || pat[k + 1] == '.' || pat[k + 1] == '=')) { | ||||||
|  |         int z = pat[k + 1]; | ||||||
|  |         k += 2; | ||||||
|  |         if (k < m && pat[k]) k++; | ||||||
|  |         while (k < m && pat[k] && (pat[k - 1] != z || pat[k] != ']')) k++; | ||||||
|  |         if (k == m || !pat[k]) break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (k == m || !pat[k]) { | ||||||
|  |       *step = 1; | ||||||
|  |       return '['; | ||||||
|  |     } | ||||||
|  |     *step = k + 1; | ||||||
|  |     return BRACKET; | ||||||
|  |   } | ||||||
|  |   if (pat[0] == '*') return STAR; | ||||||
|  |   if (pat[0] == '?') return QUESTION; | ||||||
|  | escaped: | ||||||
|  |   if (pat[0] >= 128U) { | ||||||
|  |     wchar_t wc; | ||||||
|  |     int k = mbtowc(&wc, pat, m); | ||||||
|  |     if (k < 0) { | ||||||
|  |       *step = 0; | ||||||
|  |       return UNMATCHABLE; | ||||||
|  |     } | ||||||
|  |     *step = k + esc; | ||||||
|  |     return wc; | ||||||
|  |   } | ||||||
|  |   return pat[0]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int FnmatchCaseFold(int k) { | ||||||
|  |   int c = towupper(k); | ||||||
|  |   return c == k ? towlower(k) : c; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int FnmatchBracket(const char *p, int k, int kfold) { | ||||||
|  |   wchar_t wc; | ||||||
|  |   int inv = 0; | ||||||
|  |   p++; | ||||||
|  |   if (*p == '^' || *p == '!') { | ||||||
|  |     inv = 1; | ||||||
|  |     p++; | ||||||
|  |   } | ||||||
|  |   if (*p == ']') { | ||||||
|  |     if (k == ']') return !inv; | ||||||
|  |     p++; | ||||||
|  |   } else if (*p == '-') { | ||||||
|  |     if (k == '-') return !inv; | ||||||
|  |     p++; | ||||||
|  |   } | ||||||
|  |   wc = p[-1]; | ||||||
|  |   for (; *p != ']'; p++) { | ||||||
|  |     if (p[0] == '-' && p[1] != ']') { | ||||||
|  |       wchar_t wc2; | ||||||
|  |       int l = mbtowc(&wc2, p + 1, 4); | ||||||
|  |       if (l < 0) return 0; | ||||||
|  |       if (wc <= wc2) | ||||||
|  |         if ((unsigned)k - wc <= wc2 - wc || (unsigned)kfold - wc <= wc2 - wc) | ||||||
|  |           return !inv; | ||||||
|  |       p += l - 1; | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     if (p[0] == '[' && (p[1] == ':' || p[1] == '.' || p[1] == '=')) { | ||||||
|  |       const char *p0 = p + 2; | ||||||
|  |       int z = p[1]; | ||||||
|  |       p += 3; | ||||||
|  |       while (p[-1] != z || p[0] != ']') p++; | ||||||
|  |       if (z == ':' && p - 1 - p0 < 16) { | ||||||
|  |         char buf[16]; | ||||||
|  |         memcpy(buf, p0, p - 1 - p0); | ||||||
|  |         buf[p - 1 - p0] = 0; | ||||||
|  |         if (iswctype(k, wctype(buf)) || iswctype(kfold, wctype(buf))) | ||||||
|  |           return !inv; | ||||||
|  |       } | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     if (*p < 128U) { | ||||||
|  |       wc = (unsigned char)*p; | ||||||
|  |     } else { | ||||||
|  |       int l = mbtowc(&wc, p, 4); | ||||||
|  |       if (l < 0) return 0; | ||||||
|  |       p += l - 1; | ||||||
|  |     } | ||||||
|  |     if (wc == k || wc == kfold) return !inv; | ||||||
|  |   } | ||||||
|  |   return inv; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int FnmatchPerform(const char *pat, size_t m, const char *str, size_t n, | ||||||
|  |                           int flags) { | ||||||
|  |   const char *p, *ptail, *endpat; | ||||||
|  |   const char *s, *stail, *endstr; | ||||||
|  |   size_t pinc, sinc, tailcnt = 0; | ||||||
|  |   int c, k, kfold; | ||||||
|  | 
 | ||||||
|  |   if (flags & FNM_PERIOD) { | ||||||
|  |     if (*str == '.' && *pat != '.') { | ||||||
|  |       return FNM_NOMATCH; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   for (;;) { | ||||||
|  |     switch ((c = FnmatchNextPattern(pat, m, &pinc, flags))) { | ||||||
|  |       case UNMATCHABLE: | ||||||
|  |         return FNM_NOMATCH; | ||||||
|  |       case STAR: | ||||||
|  |         pat++; | ||||||
|  |         m--; | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         k = FnmatchNextString(str, n, &sinc); | ||||||
|  |         if (k <= 0) return (c == END) ? 0 : FNM_NOMATCH; | ||||||
|  |         str += sinc; | ||||||
|  |         n -= sinc; | ||||||
|  |         kfold = flags & FNM_CASEFOLD ? FnmatchCaseFold(k) : k; | ||||||
|  |         if (c == BRACKET) { | ||||||
|  |           if (!FnmatchBracket(pat, k, kfold)) return FNM_NOMATCH; | ||||||
|  |         } else if (c != QUESTION && k != c && kfold != c) { | ||||||
|  |           return FNM_NOMATCH; | ||||||
|  |         } | ||||||
|  |         pat += pinc; | ||||||
|  |         m -= pinc; | ||||||
|  |         continue; | ||||||
|  |     } | ||||||
|  |     break; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /* Compute real pat length if it was initially unknown/-1 */ | ||||||
|  |   m = strnlen(pat, m); | ||||||
|  |   endpat = pat + m; | ||||||
|  | 
 | ||||||
|  |   /* Find the last * in pat and count chars needed after it */ | ||||||
|  |   for (p = ptail = pat; p < endpat; p += pinc) { | ||||||
|  |     switch (FnmatchNextPattern(p, endpat - p, &pinc, flags)) { | ||||||
|  |       case UNMATCHABLE: | ||||||
|  |         return FNM_NOMATCH; | ||||||
|  |       case STAR: | ||||||
|  |         tailcnt = 0; | ||||||
|  |         ptail = p + 1; | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         tailcnt++; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /* Past this point we need not check for UNMATCHABLE in pat,
 | ||||||
|  |    * because all of pat has already been parsed once. */ | ||||||
|  | 
 | ||||||
|  |   /* Compute real str length if it was initially unknown/-1 */ | ||||||
|  |   n = strnlen(str, n); | ||||||
|  |   endstr = str + n; | ||||||
|  |   if (n < tailcnt) { | ||||||
|  |     return FNM_NOMATCH; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /* Find the final tailcnt chars of str, accounting for UTF-8.
 | ||||||
|  |    * On illegal sequences we may get it wrong, but in that case | ||||||
|  |    * we necessarily have a matching failure anyway. */ | ||||||
|  |   for (s = endstr; s > str && tailcnt; tailcnt--) { | ||||||
|  |     if (s[-1] < 128U || MB_CUR_MAX == 1) { | ||||||
|  |       s--; | ||||||
|  |     } else { | ||||||
|  |       while ((unsigned char)*--s - 0x80U < 0x40 && s > str) | ||||||
|  |         ; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (tailcnt) return FNM_NOMATCH; | ||||||
|  |   stail = s; | ||||||
|  | 
 | ||||||
|  |   /* Check that the pat and str tails match */ | ||||||
|  |   p = ptail; | ||||||
|  |   for (;;) { | ||||||
|  |     c = FnmatchNextPattern(p, endpat - p, &pinc, flags); | ||||||
|  |     p += pinc; | ||||||
|  |     if ((k = FnmatchNextString(s, endstr - s, &sinc)) <= 0) { | ||||||
|  |       if (c != END) return FNM_NOMATCH; | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     s += sinc; | ||||||
|  |     kfold = flags & FNM_CASEFOLD ? FnmatchCaseFold(k) : k; | ||||||
|  |     if (c == BRACKET) { | ||||||
|  |       if (!FnmatchBracket(p - pinc, k, kfold)) return FNM_NOMATCH; | ||||||
|  |     } else if (c != QUESTION && k != c && kfold != c) { | ||||||
|  |       return FNM_NOMATCH; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /* We're all done with the tails now, so throw them out */ | ||||||
|  |   endstr = stail; | ||||||
|  |   endpat = ptail; | ||||||
|  | 
 | ||||||
|  |   /* Match pattern components until there are none left */ | ||||||
|  |   while (pat < endpat) { | ||||||
|  |     p = pat; | ||||||
|  |     s = str; | ||||||
|  |     for (;;) { | ||||||
|  |       c = FnmatchNextPattern(p, endpat - p, &pinc, flags); | ||||||
|  |       p += pinc; | ||||||
|  |       /* Encountering * completes/commits a component */ | ||||||
|  |       if (c == STAR) { | ||||||
|  |         pat = p; | ||||||
|  |         str = s; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       k = FnmatchNextString(s, endstr - s, &sinc); | ||||||
|  |       if (!k) return FNM_NOMATCH; | ||||||
|  |       kfold = flags & FNM_CASEFOLD ? FnmatchCaseFold(k) : k; | ||||||
|  |       if (c == BRACKET) { | ||||||
|  |         if (!FnmatchBracket(p - pinc, k, kfold)) break; | ||||||
|  |       } else if (c != QUESTION && k != c && kfold != c) { | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       s += sinc; | ||||||
|  |     } | ||||||
|  |     if (c == STAR) continue; | ||||||
|  |     /* If we failed, advance str, by 1 char if it's a valid
 | ||||||
|  |      * char, or past all invalid bytes otherwise. */ | ||||||
|  |     k = FnmatchNextString(str, endstr - str, &sinc); | ||||||
|  |     if (k > 0) { | ||||||
|  |       str += sinc; | ||||||
|  |     } else { | ||||||
|  |       str++; | ||||||
|  |       while (FnmatchNextString(str, endstr - str, &sinc) < 0) { | ||||||
|  |         str++; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Matches filename. | ||||||
|  |  * | ||||||
|  |  *   - `*` for wildcard | ||||||
|  |  *   - `?` for single character | ||||||
|  |  *   - `[abc]` to match character within set | ||||||
|  |  *   - `[!abc]` to match character not within set | ||||||
|  |  *   - `\*\?\[\]` for escaping above special syntax | ||||||
|  |  * | ||||||
|  |  * @see glob() | ||||||
|  |  */ | ||||||
|  | int fnmatch(const char *pat, const char *str, int flags) { | ||||||
|  |   const char *s, *p; | ||||||
|  |   size_t inc; | ||||||
|  |   int c; | ||||||
|  |   if (flags & FNM_PATHNAME) { | ||||||
|  |     for (;;) { | ||||||
|  |       for (s = str; *s && *s != '/'; s++) | ||||||
|  |         ; | ||||||
|  |       for (p = pat; | ||||||
|  |            (c = FnmatchNextPattern(p, -1, &inc, flags)) != END && c != '/'; | ||||||
|  |            p += inc) | ||||||
|  |         ; | ||||||
|  |       if (c != *s && (!*s || !(flags & FNM_LEADING_DIR))) return FNM_NOMATCH; | ||||||
|  |       if (FnmatchPerform(pat, p - pat, str, s - str, flags)) return FNM_NOMATCH; | ||||||
|  |       if (!c) return 0; | ||||||
|  |       str = s + 1; | ||||||
|  |       pat = p + inc; | ||||||
|  |     } | ||||||
|  |   } else if (flags & FNM_LEADING_DIR) { | ||||||
|  |     for (s = str; *s; s++) { | ||||||
|  |       if (*s != '/') continue; | ||||||
|  |       if (!FnmatchPerform(pat, -1, str, s - str, flags)) return 0; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return FnmatchPerform(pat, -1, str, -1, flags); | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								third_party/musl/fnmatch.h
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								third_party/musl/fnmatch.h
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,20 @@ | ||||||
|  | #ifndef COSMOPOLITAN_THIRD_PARTY_REGEX_FNMATCH_H_ | ||||||
|  | #define COSMOPOLITAN_THIRD_PARTY_REGEX_FNMATCH_H_ | ||||||
|  | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
|  | COSMOPOLITAN_C_START_ | ||||||
|  | 
 | ||||||
|  | #define FNM_PATHNAME    0x1 | ||||||
|  | #define FNM_NOESCAPE    0x2 | ||||||
|  | #define FNM_PERIOD      0x4 | ||||||
|  | #define FNM_LEADING_DIR 0x8 | ||||||
|  | #define FNM_CASEFOLD    0x10 | ||||||
|  | #define FNM_FILE_NAME   FNM_PATHNAME | ||||||
|  | 
 | ||||||
|  | #define FNM_NOMATCH 1 | ||||||
|  | #define FNM_NOSYS   (-1) | ||||||
|  | 
 | ||||||
|  | int fnmatch(const char *, const char *, int); | ||||||
|  | 
 | ||||||
|  | COSMOPOLITAN_C_END_ | ||||||
|  | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  | #endif /* COSMOPOLITAN_THIRD_PARTY_REGEX_FNMATCH_H_ */ | ||||||
							
								
								
									
										320
									
								
								third_party/musl/glob.c
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										320
									
								
								third_party/musl/glob.c
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,320 @@ | ||||||
|  | /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||||
|  | │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||||
|  | ╚──────────────────────────────────────────────────────────────────────────────╝ | ||||||
|  | │                                                                              │ | ||||||
|  | │  Musl Libc                                                                   │ | ||||||
|  | │  Copyright © 2005-2014 Rich Felker, et al.                                   │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  Permission is hereby granted, free of charge, to any person obtaining       │ | ||||||
|  | │  a copy of this software and associated documentation files (the             │ | ||||||
|  | │  "Software"), to deal in the Software without restriction, including         │ | ||||||
|  | │  without limitation the rights to use, copy, modify, merge, publish,         │ | ||||||
|  | │  distribute, sublicense, and/or sell copies of the Software, and to          │ | ||||||
|  | │  permit persons to whom the Software is furnished to do so, subject to       │ | ||||||
|  | │  the following conditions:                                                   │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  The above copyright notice and this permission notice shall be              │ | ||||||
|  | │  included in all copies or substantial portions of the Software.             │ | ||||||
|  | │                                                                              │ | ||||||
|  | │  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,             │ | ||||||
|  | │  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF          │ | ||||||
|  | │  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.      │ | ||||||
|  | │  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY        │ | ||||||
|  | │  CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,        │ | ||||||
|  | │  TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE           │ | ||||||
|  | │  SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                      │ | ||||||
|  | │                                                                              │ | ||||||
|  | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
|  | #include "libc/alg/alg.h" | ||||||
|  | #include "libc/calls/calls.h" | ||||||
|  | #include "libc/calls/struct/dirent.h" | ||||||
|  | #include "libc/calls/struct/stat.h" | ||||||
|  | #include "libc/errno.h" | ||||||
|  | #include "libc/mem/mem.h" | ||||||
|  | #include "libc/str/str.h" | ||||||
|  | #include "libc/sysv/consts/dt.h" | ||||||
|  | #include "third_party/musl/fnmatch.h" | ||||||
|  | #include "third_party/musl/glob.h" | ||||||
|  | 
 | ||||||
|  | asm(".ident\t\"\\n\\n\
 | ||||||
|  | Musl libc (MIT License)\\n\ | ||||||
|  | Copyright 2005-2014 Rich Felker, et. al.\""); | ||||||
|  | asm(".include \"libc/disclaimer.inc\""); | ||||||
|  | 
 | ||||||
|  | struct GlobList { | ||||||
|  |   struct GlobList *next; | ||||||
|  |   char name[]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int AppendGlob(struct GlobList **tail, const char *name, size_t len, | ||||||
|  |                       int mark) { | ||||||
|  |   struct GlobList *new; | ||||||
|  |   if ((new = malloc(sizeof(struct GlobList) + len + 2))) { | ||||||
|  |     (*tail)->next = new; | ||||||
|  |     new->next = NULL; | ||||||
|  |     memcpy(new->name, name, len + 1); | ||||||
|  |     if (mark && len && name[len - 1] != '/') { | ||||||
|  |       new->name[len] = '/'; | ||||||
|  |       new->name[len + 1] = 0; | ||||||
|  |     } | ||||||
|  |     *tail = new; | ||||||
|  |     return 0; | ||||||
|  |   } else { | ||||||
|  |     return -1; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int PerformGlob(char *buf, size_t pos, int type, char *pat, int flags, | ||||||
|  |                        int (*errfunc)(const char *path, int err), | ||||||
|  |                        struct GlobList **tail) { | ||||||
|  |   DIR *dir; | ||||||
|  |   size_t l; | ||||||
|  |   char *p, *p2; | ||||||
|  |   char saved_sep; | ||||||
|  |   ptrdiff_t i, j; | ||||||
|  |   struct stat st; | ||||||
|  |   struct dirent *de; | ||||||
|  |   int r, readerr, in_bracket, overflow, old_errno, fnm_flags; | ||||||
|  |   /* If GLOB_MARK is unused, we don't care about type. */ | ||||||
|  |   if (!type && !(flags & GLOB_MARK)) type = DT_REG; | ||||||
|  |   /* Special-case the remaining pattern being all slashes, in
 | ||||||
|  |    * which case we can use caller-passed type if it's a dir. */ | ||||||
|  |   if (*pat && type != DT_DIR) type = 0; | ||||||
|  |   while (pos + 1 < PATH_MAX && *pat == '/') { | ||||||
|  |     buf[pos++] = *pat++; | ||||||
|  |   } | ||||||
|  |   /* Consume maximal [escaped-]literal prefix of pattern, copying
 | ||||||
|  |    * and un-escaping it to the running buffer as we go. */ | ||||||
|  |   i = 0; | ||||||
|  |   j = 0; | ||||||
|  |   overflow = 0; | ||||||
|  |   in_bracket = 0; | ||||||
|  |   for (; pat[i] != '*' && pat[i] != '?' && (!in_bracket || pat[i] != ']'); | ||||||
|  |        i++) { | ||||||
|  |     if (!pat[i]) { | ||||||
|  |       if (overflow) return 0; | ||||||
|  |       pat += i; | ||||||
|  |       pos += j; | ||||||
|  |       i = j = 0; | ||||||
|  |       break; | ||||||
|  |     } else if (pat[i] == '[') { | ||||||
|  |       in_bracket = 1; | ||||||
|  |     } else if (pat[i] == '\\' && !(flags & GLOB_NOESCAPE)) { | ||||||
|  |       /* Backslashes inside a bracket are (at least by
 | ||||||
|  |        * our interpretation) non-special, so if next | ||||||
|  |        * char is ']' we have a complete expression. */ | ||||||
|  |       if (in_bracket && pat[i + 1] == ']') break; | ||||||
|  |       /* Unpaired final backslash never matches. */ | ||||||
|  |       if (!pat[i + 1]) return 0; | ||||||
|  |       i++; | ||||||
|  |     } | ||||||
|  |     if (pat[i] == '/') { | ||||||
|  |       if (overflow) return 0; | ||||||
|  |       in_bracket = 0; | ||||||
|  |       pat += i + 1; | ||||||
|  |       i = -1; | ||||||
|  |       pos += j + 1; | ||||||
|  |       j = -1; | ||||||
|  |     } | ||||||
|  |     /* Only store a character if it fits in the buffer, but if
 | ||||||
|  |      * a potential bracket expression is open, the overflow | ||||||
|  |      * must be remembered and handled later only if the bracket | ||||||
|  |      * is unterminated (and thereby a literal), so as not to | ||||||
|  |      * disallow long bracket expressions with short matches. */ | ||||||
|  |     if (pos + (j + 1) < PATH_MAX) { | ||||||
|  |       buf[pos + j++] = pat[i]; | ||||||
|  |     } else if (in_bracket) { | ||||||
|  |       overflow = 1; | ||||||
|  |     } else { | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  |     /* If we consume any new components, the caller-passed type
 | ||||||
|  |      * or dummy type from above is no longer valid. */ | ||||||
|  |     type = 0; | ||||||
|  |   } | ||||||
|  |   buf[pos] = 0; | ||||||
|  |   if (!*pat) { | ||||||
|  |     /* If we consumed any components above, or if GLOB_MARK is
 | ||||||
|  |      * requested and we don't yet know if the match is a dir, | ||||||
|  |      * we must call stat to confirm the file exists and/or | ||||||
|  |      * determine its type. */ | ||||||
|  |     if ((flags & GLOB_MARK) && type == DT_LNK) type = 0; | ||||||
|  |     if (!type && stat(buf, &st)) { | ||||||
|  |       if (errno != ENOENT && (errfunc(buf, errno) || (flags & GLOB_ERR))) { | ||||||
|  |         return GLOB_ABORTED; | ||||||
|  |       } | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  |     if (!type && S_ISDIR(st.st_mode)) type = DT_DIR; | ||||||
|  |     if (AppendGlob(tail, buf, pos, (flags & GLOB_MARK) && type == DT_DIR)) { | ||||||
|  |       return GLOB_NOSPACE; | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   p2 = strchr(pat, '/'); | ||||||
|  |   saved_sep = '/'; | ||||||
|  |   /* Check if the '/' was escaped and, if so, remove the escape char
 | ||||||
|  |    * so that it will not be unpaired when passed to fnmatch. */ | ||||||
|  |   if (p2 && !(flags & GLOB_NOESCAPE)) { | ||||||
|  |     for (p = p2; p > pat && p[-1] == '\\'; p--) | ||||||
|  |       ; | ||||||
|  |     if ((p2 - p) % 2) { | ||||||
|  |       p2--; | ||||||
|  |       saved_sep = '\\'; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   dir = opendir(pos ? buf : "."); | ||||||
|  |   if (!dir) { | ||||||
|  |     if (errfunc(buf, errno) || (flags & GLOB_ERR)) return GLOB_ABORTED; | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   old_errno = errno; | ||||||
|  |   while (errno = 0, de = readdir(dir)) { | ||||||
|  |     /* Quickly skip non-directories when there's pattern left. */ | ||||||
|  |     if (p2 && de->d_type && de->d_type != DT_DIR && de->d_type != DT_LNK) { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     l = strlen(de->d_name); | ||||||
|  |     if (l >= PATH_MAX - pos) continue; | ||||||
|  |     if (p2) *p2 = 0; | ||||||
|  |     fnm_flags = ((flags & GLOB_NOESCAPE) ? FNM_NOESCAPE : 0) | | ||||||
|  |                 ((!(flags & GLOB_PERIOD)) ? FNM_PERIOD : 0); | ||||||
|  |     if (fnmatch(pat, de->d_name, fnm_flags)) continue; | ||||||
|  |     /* With GLOB_PERIOD don't allow matching . or .. unless fnmatch()
 | ||||||
|  |      * would match them with FNM_PERIOD rules in effect. */ | ||||||
|  |     if (p2 && (flags & GLOB_PERIOD) && de->d_name[0] == '.' && | ||||||
|  |         (!de->d_name[1] || de->d_name[1] == '.' && !de->d_name[2]) && | ||||||
|  |         fnmatch(pat, de->d_name, fnm_flags | FNM_PERIOD)) { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     memcpy(buf + pos, de->d_name, l + 1); | ||||||
|  |     if (p2) *p2 = saved_sep; | ||||||
|  |     r = PerformGlob(buf, pos + l, de->d_type, p2 ? p2 : "", flags, errfunc, | ||||||
|  |                     tail); | ||||||
|  |     if (r) { | ||||||
|  |       closedir(dir); | ||||||
|  |       return r; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   readerr = errno; | ||||||
|  |   if (p2) *p2 = saved_sep; | ||||||
|  |   closedir(dir); | ||||||
|  |   if (readerr && (errfunc(buf, errno) || (flags & GLOB_ERR))) { | ||||||
|  |     return GLOB_ABORTED; | ||||||
|  |   } | ||||||
|  |   errno = old_errno; | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int IgnoreGlobError(const char *path, int err) { | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void FreeGlobList(struct GlobList *head) { | ||||||
|  |   struct GlobList *match, *next; | ||||||
|  |   for (match = head->next; match; match = next) { | ||||||
|  |     next = match->next; | ||||||
|  |     free(match); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int GlobPredicate(const void *a, const void *b) { | ||||||
|  |   return strcmp(*(const char **)a, *(const char **)b); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Finds pathnames matching pattern. | ||||||
|  |  * | ||||||
|  |  * For example: | ||||||
|  |  * | ||||||
|  |  *   glob_t g = {.gl_offs = 2}; | ||||||
|  |  *   glob("*.*", GLOB_DOOFFS, NULL, &g); | ||||||
|  |  *   glob("../.*", GLOB_DOOFFS | GLOB_APPEND, NULL, &g); | ||||||
|  |  *   g.gl_pathv[0] = "ls"; | ||||||
|  |  *   g.gl_pathv[1] = "-l"; | ||||||
|  |  *   execvp("ls", &g.gl_pathv[0]); | ||||||
|  |  *   globfree(g); | ||||||
|  |  * | ||||||
|  |  * @param pat can have star wildcard see fnmatch() | ||||||
|  |  * @param g will receive matching entries and needs globfree() | ||||||
|  |  * @return 0 on success or GLOB_NOMATCH, GLOB_NOSPACE on OOM, or | ||||||
|  |  *     GLOB_ABORTED on read error | ||||||
|  |  */ | ||||||
|  | int glob(const char *pat, int flags, int errfunc(const char *path, int err), | ||||||
|  |          glob_t *g) { | ||||||
|  |   int error = 0; | ||||||
|  |   size_t cnt, i; | ||||||
|  |   char *p, **pathv, buf[PATH_MAX]; | ||||||
|  |   struct GlobList head = {.next = NULL}, *tail = &head; | ||||||
|  |   size_t offs = (flags & GLOB_DOOFFS) ? g->gl_offs : 0; | ||||||
|  |   if (!errfunc) errfunc = IgnoreGlobError; | ||||||
|  |   if (!(flags & GLOB_APPEND)) { | ||||||
|  |     g->gl_offs = offs; | ||||||
|  |     g->gl_pathc = 0; | ||||||
|  |     g->gl_pathv = NULL; | ||||||
|  |   } | ||||||
|  |   if (*pat) { | ||||||
|  |     char *p = strdup(pat); | ||||||
|  |     if (!p) return GLOB_NOSPACE; | ||||||
|  |     buf[0] = 0; | ||||||
|  |     error = PerformGlob(buf, 0, 0, p, flags, errfunc, &tail); | ||||||
|  |     free(p); | ||||||
|  |   } | ||||||
|  |   if (error == GLOB_NOSPACE) { | ||||||
|  |     FreeGlobList(&head); | ||||||
|  |     return error; | ||||||
|  |   } | ||||||
|  |   for (cnt = 0, tail = head.next; tail; tail = tail->next, cnt++) | ||||||
|  |     ; | ||||||
|  |   if (!cnt) { | ||||||
|  |     if (flags & GLOB_NOCHECK) { | ||||||
|  |       tail = &head; | ||||||
|  |       if (AppendGlob(&tail, pat, strlen(pat), 0)) { | ||||||
|  |         return GLOB_NOSPACE; | ||||||
|  |       } | ||||||
|  |       cnt++; | ||||||
|  |     } else | ||||||
|  |       return GLOB_NOMATCH; | ||||||
|  |   } | ||||||
|  |   if (flags & GLOB_APPEND) { | ||||||
|  |     pathv = | ||||||
|  |         realloc(g->gl_pathv, (offs + g->gl_pathc + cnt + 1) * sizeof(char *)); | ||||||
|  |     if (!pathv) { | ||||||
|  |       FreeGlobList(&head); | ||||||
|  |       return GLOB_NOSPACE; | ||||||
|  |     } | ||||||
|  |     g->gl_pathv = pathv; | ||||||
|  |     offs += g->gl_pathc; | ||||||
|  |   } else { | ||||||
|  |     g->gl_pathv = malloc((offs + cnt + 1) * sizeof(char *)); | ||||||
|  |     if (!g->gl_pathv) { | ||||||
|  |       FreeGlobList(&head); | ||||||
|  |       return GLOB_NOSPACE; | ||||||
|  |     } | ||||||
|  |     for (i = 0; i < offs; i++) { | ||||||
|  |       g->gl_pathv[i] = NULL; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   for (i = 0, tail = head.next; i < cnt; tail = tail->next, i++) { | ||||||
|  |     g->gl_pathv[offs + i] = tail->name; | ||||||
|  |   } | ||||||
|  |   g->gl_pathv[offs + i] = NULL; | ||||||
|  |   g->gl_pathc += cnt; | ||||||
|  |   if (!(flags & GLOB_NOSORT)) { | ||||||
|  |     qsort(g->gl_pathv + offs, cnt, sizeof(char *), GlobPredicate); | ||||||
|  |   } | ||||||
|  |   return error; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Frees entries allocated by glob(). | ||||||
|  |  */ | ||||||
|  | void globfree(glob_t *g) { | ||||||
|  |   size_t i; | ||||||
|  |   for (i = 0; i < g->gl_pathc; i++) { | ||||||
|  |     free(g->gl_pathv[g->gl_offs + i] - offsetof(struct GlobList, name)); | ||||||
|  |   } | ||||||
|  |   free(g->gl_pathv); | ||||||
|  |   g->gl_pathc = 0; | ||||||
|  |   g->gl_pathv = NULL; | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								third_party/musl/glob.h
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								third_party/musl/glob.h
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | ||||||
|  | #ifndef COSMOPOLITAN_THIRD_PARTY_MUSL_GLOB_H_ | ||||||
|  | #define COSMOPOLITAN_THIRD_PARTY_MUSL_GLOB_H_ | ||||||
|  | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
|  | COSMOPOLITAN_C_START_ | ||||||
|  | 
 | ||||||
|  | #define GLOB_ERR      0x01 | ||||||
|  | #define GLOB_MARK     0x02 | ||||||
|  | #define GLOB_NOSORT   0x04 | ||||||
|  | #define GLOB_DOOFFS   0x08 /* reserves null slots at start of gl_pathv */ | ||||||
|  | #define GLOB_NOCHECK  0x10 /* just yield pattern if GLOB_NOMATCH happens */ | ||||||
|  | #define GLOB_APPEND   0x20 /* enables us to call glob() multiple times */ | ||||||
|  | #define GLOB_NOESCAPE 0x40 /* don't allow things like \*\?\[\] escaping */ | ||||||
|  | #define GLOB_PERIOD   0x80 | ||||||
|  | 
 | ||||||
|  | #define GLOB_TILDE       0x1000 | ||||||
|  | #define GLOB_TILDE_CHECK 0x4000 | ||||||
|  | 
 | ||||||
|  | #define GLOB_NOSPACE 1 | ||||||
|  | #define GLOB_ABORTED 2 | ||||||
|  | #define GLOB_NOMATCH 3 | ||||||
|  | #define GLOB_NOSYS   4 | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |   size_t gl_pathc; | ||||||
|  |   char **gl_pathv; | ||||||
|  |   size_t gl_offs; | ||||||
|  |   int __dummy1; | ||||||
|  |   void *__dummy2[5]; | ||||||
|  | } glob_t; | ||||||
|  | 
 | ||||||
|  | int glob(const char *, int, int (*)(const char *, int), glob_t *); | ||||||
|  | void globfree(glob_t *); | ||||||
|  | 
 | ||||||
|  | COSMOPOLITAN_C_END_ | ||||||
|  | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  | #endif /* COSMOPOLITAN_THIRD_PARTY_MUSL_GLOB_H_ */ | ||||||
							
								
								
									
										2
									
								
								third_party/musl/musl.mk
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/musl/musl.mk
									
										
									
									
										vendored
									
									
								
							|  | @ -18,7 +18,9 @@ THIRD_PARTY_MUSL_A_OBJS =				\ | ||||||
| 	$(THIRD_PARTY_MUSL_A_SRCS:%.c=o/$(MODE)/%.o) | 	$(THIRD_PARTY_MUSL_A_SRCS:%.c=o/$(MODE)/%.o) | ||||||
| 
 | 
 | ||||||
| THIRD_PARTY_MUSL_A_DIRECTDEPS =				\
 | THIRD_PARTY_MUSL_A_DIRECTDEPS =				\
 | ||||||
|  | 	LIBC_ALG					\
 | ||||||
| 	LIBC_CALLS					\
 | 	LIBC_CALLS					\
 | ||||||
|  | 	LIBC_CALLS_HEFTY				\
 | ||||||
| 	LIBC_MEM					\
 | 	LIBC_MEM					\
 | ||||||
| 	LIBC_NEXGEN32E					\
 | 	LIBC_NEXGEN32E					\
 | ||||||
| 	LIBC_STDIO					\
 | 	LIBC_STDIO					\
 | ||||||
|  |  | ||||||
							
								
								
									
										294
									
								
								third_party/regex/glob.c.todo
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										294
									
								
								third_party/regex/glob.c.todo
									
										
									
									
										vendored
									
									
								
							|  | @ -1,294 +0,0 @@ | ||||||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ |  | ||||||
| │vi: set net ft=c ts=8 sts=2 sw=2 fenc=utf-8                                :vi│ |  | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ |  | ||||||
| /* clang-format off */ |  | ||||||
| 
 |  | ||||||
| /* |  | ||||||
|   Musl Libc |  | ||||||
|   Copyright © 2005-2014 Rich Felker, et al. |  | ||||||
| 
 |  | ||||||
|   Permission is hereby granted, free of charge, to any person obtaining |  | ||||||
|   a copy of this software and associated documentation files (the |  | ||||||
|   "Software"), to deal in the Software without restriction, including |  | ||||||
|   without limitation the rights to use, copy, modify, merge, publish, |  | ||||||
|   distribute, sublicense, and/or sell copies of the Software, and to |  | ||||||
|   permit persons to whom the Software is furnished to do so, subject to |  | ||||||
|   the following conditions: |  | ||||||
| 
 |  | ||||||
|   The above copyright notice and this permission notice shall be |  | ||||||
|   included in all copies or substantial portions of the Software. |  | ||||||
| 
 |  | ||||||
|   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |  | ||||||
|   EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |  | ||||||
|   MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. |  | ||||||
|   IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY |  | ||||||
|   CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, |  | ||||||
|   TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE |  | ||||||
|   SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |  | ||||||
| */ |  | ||||||
| 
 |  | ||||||
| #define _BSD_SOURCE |  | ||||||
| #include <dirent.h> |  | ||||||
| #include <errno.h> |  | ||||||
| #include <fnmatch.h> |  | ||||||
| #include <glob.h> |  | ||||||
| #include <limits.h> |  | ||||||
| #include <stddef.h> |  | ||||||
| #include <stdlib.h> |  | ||||||
| #include <string.h> |  | ||||||
| #include <sys/stat.h> |  | ||||||
| 
 |  | ||||||
| asm(".ident\t\"\\n\\n\ |  | ||||||
| Musl Libc » glob (MIT License)\\n\ |  | ||||||
| Copyright 2005-2014 Rich Felker\""); |  | ||||||
| 
 |  | ||||||
| struct match |  | ||||||
| { |  | ||||||
| 	struct match *next; |  | ||||||
| 	char name[]; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| static int append(struct match **tail, const char *name, size_t len, int mark) |  | ||||||
| { |  | ||||||
| 	struct match *new = malloc(sizeof(struct match) + len + 2); |  | ||||||
| 	if (!new) return -1; |  | ||||||
| 	(*tail)->next = new; |  | ||||||
| 	new->next = NULL; |  | ||||||
| 	memcpy(new->name, name, len+1); |  | ||||||
| 	if (mark && len && name[len-1]!='/') { |  | ||||||
| 		new->name[len] = '/'; |  | ||||||
| 		new->name[len+1] = 0; |  | ||||||
| 	} |  | ||||||
| 	*tail = new; |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int do_glob(char *buf, size_t pos, int type, char *pat, int flags, int (*errfunc)(const char *path, int err), struct match **tail) |  | ||||||
| { |  | ||||||
| 	/* If GLOB_MARK is unused, we don't care about type. */ |  | ||||||
| 	if (!type && !(flags & GLOB_MARK)) type = DT_REG; |  | ||||||
| 
 |  | ||||||
| 	/* Special-case the remaining pattern being all slashes, in |  | ||||||
| 	 * which case we can use caller-passed type if it's a dir. */ |  | ||||||
| 	if (*pat && type!=DT_DIR) type = 0; |  | ||||||
| 	while (pos+1 < PATH_MAX && *pat=='/') buf[pos++] = *pat++; |  | ||||||
| 
 |  | ||||||
| 	/* Consume maximal [escaped-]literal prefix of pattern, copying |  | ||||||
| 	 * and un-escaping it to the running buffer as we go. */ |  | ||||||
| 	ptrdiff_t i=0, j=0; |  | ||||||
| 	int in_bracket = 0, overflow = 0; |  | ||||||
| 	for (; pat[i]!='*' && pat[i]!='?' && (!in_bracket || pat[i]!=']'); i++) { |  | ||||||
| 		if (!pat[i]) { |  | ||||||
| 			if (overflow) return 0; |  | ||||||
| 			pat += i; |  | ||||||
| 			pos += j; |  | ||||||
| 			i = j = 0; |  | ||||||
| 			break; |  | ||||||
| 		} else if (pat[i] == '[') { |  | ||||||
| 			in_bracket = 1; |  | ||||||
| 		} else if (pat[i] == '\\' && !(flags & GLOB_NOESCAPE)) { |  | ||||||
| 			/* Backslashes inside a bracket are (at least by |  | ||||||
| 			 * our interpretation) non-special, so if next |  | ||||||
| 			 * char is ']' we have a complete expression. */ |  | ||||||
| 			if (in_bracket && pat[i+1]==']') break; |  | ||||||
| 			/* Unpaired final backslash never matches. */ |  | ||||||
| 			if (!pat[i+1]) return 0; |  | ||||||
| 			i++; |  | ||||||
| 		} |  | ||||||
| 		if (pat[i] == '/') { |  | ||||||
| 			if (overflow) return 0; |  | ||||||
| 			in_bracket = 0; |  | ||||||
| 			pat += i+1; |  | ||||||
| 			i = -1; |  | ||||||
| 			pos += j+1; |  | ||||||
| 			j = -1; |  | ||||||
| 		} |  | ||||||
| 		/* Only store a character if it fits in the buffer, but if |  | ||||||
| 		 * a potential bracket expression is open, the overflow |  | ||||||
| 		 * must be remembered and handled later only if the bracket |  | ||||||
| 		 * is unterminated (and thereby a literal), so as not to |  | ||||||
| 		 * disallow long bracket expressions with short matches. */ |  | ||||||
| 		if (pos+(j+1) < PATH_MAX) { |  | ||||||
| 			buf[pos+j++] = pat[i]; |  | ||||||
| 		} else if (in_bracket) { |  | ||||||
| 			overflow = 1; |  | ||||||
| 		} else { |  | ||||||
| 			return 0; |  | ||||||
| 		} |  | ||||||
| 		/* If we consume any new components, the caller-passed type |  | ||||||
| 		 * or dummy type from above is no longer valid. */ |  | ||||||
| 		type = 0; |  | ||||||
| 	} |  | ||||||
| 	buf[pos] = 0; |  | ||||||
| 	if (!*pat) { |  | ||||||
| 		/* If we consumed any components above, or if GLOB_MARK is |  | ||||||
| 		 * requested and we don't yet know if the match is a dir, |  | ||||||
| 		 * we must call stat to confirm the file exists and/or |  | ||||||
| 		 * determine its type. */ |  | ||||||
| 		struct stat st; |  | ||||||
| 		if ((flags & GLOB_MARK) && type==DT_LNK) type = 0; |  | ||||||
| 		if (!type && stat(buf, &st)) { |  | ||||||
| 			if (errno!=ENOENT && (errfunc(buf, errno) || (flags & GLOB_ERR))) |  | ||||||
| 				return GLOB_ABORTED; |  | ||||||
| 			return 0; |  | ||||||
| 		} |  | ||||||
| 		if (!type && S_ISDIR(st.st_mode)) type = DT_DIR; |  | ||||||
| 		if (append(tail, buf, pos, (flags & GLOB_MARK) && type==DT_DIR)) |  | ||||||
| 			return GLOB_NOSPACE; |  | ||||||
| 		return 0; |  | ||||||
| 	} |  | ||||||
| 	char *p2 = strchr(pat, '/'), saved_sep = '/'; |  | ||||||
| 	/* Check if the '/' was escaped and, if so, remove the escape char |  | ||||||
| 	 * so that it will not be unpaired when passed to fnmatch. */ |  | ||||||
| 	if (p2 && !(flags & GLOB_NOESCAPE)) { |  | ||||||
| 		char *p; |  | ||||||
| 		for (p=p2; p>pat && p[-1]=='\\'; p--); |  | ||||||
| 		if ((p2-p)%2) { |  | ||||||
| 			p2--; |  | ||||||
| 			saved_sep = '\\'; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	DIR *dir = opendir(pos ? buf : "."); |  | ||||||
| 	if (!dir) { |  | ||||||
| 		if (errfunc(buf, errno) || (flags & GLOB_ERR)) |  | ||||||
| 			return GLOB_ABORTED; |  | ||||||
| 		return 0; |  | ||||||
| 	} |  | ||||||
| 	int old_errno = errno; |  | ||||||
| 	struct dirent *de; |  | ||||||
| 	while (errno=0, de=readdir(dir)) { |  | ||||||
| 		/* Quickly skip non-directories when there's pattern left. */ |  | ||||||
| 		if (p2 && de->d_type && de->d_type!=DT_DIR && de->d_type!=DT_LNK) |  | ||||||
| 			continue; |  | ||||||
| 
 |  | ||||||
| 		size_t l = strlen(de->d_name); |  | ||||||
| 		if (l >= PATH_MAX-pos) continue; |  | ||||||
| 
 |  | ||||||
| 		if (p2) *p2 = 0; |  | ||||||
| 
 |  | ||||||
| 		int fnm_flags= ((flags & GLOB_NOESCAPE) ? FNM_NOESCAPE : 0) |  | ||||||
| 			| ((!(flags & GLOB_PERIOD)) ? FNM_PERIOD : 0); |  | ||||||
| 
 |  | ||||||
| 		if (fnmatch(pat, de->d_name, fnm_flags)) |  | ||||||
| 			continue; |  | ||||||
| 
 |  | ||||||
| 		/* With GLOB_PERIOD, don't allow matching . or .. unless |  | ||||||
| 		 * fnmatch would match them with FNM_PERIOD rules in effect. */ |  | ||||||
| 		if (p2 && (flags & GLOB_PERIOD) && de->d_name[0]=='.' |  | ||||||
| 		    && (!de->d_name[1] || de->d_name[1]=='.' && !de->d_name[2]) |  | ||||||
| 		    && fnmatch(pat, de->d_name, fnm_flags | FNM_PERIOD)) |  | ||||||
| 			continue; |  | ||||||
| 
 |  | ||||||
| 		memcpy(buf+pos, de->d_name, l+1); |  | ||||||
| 		if (p2) *p2 = saved_sep; |  | ||||||
| 		int r = do_glob(buf, pos+l, de->d_type, p2 ? p2 : "", flags, errfunc, tail); |  | ||||||
| 		if (r) { |  | ||||||
| 			closedir(dir); |  | ||||||
| 			return r; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	int readerr = errno; |  | ||||||
| 	if (p2) *p2 = saved_sep; |  | ||||||
| 	closedir(dir); |  | ||||||
| 	if (readerr && (errfunc(buf, errno) || (flags & GLOB_ERR))) |  | ||||||
| 		return GLOB_ABORTED; |  | ||||||
| 	errno = old_errno; |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int ignore_err(const char *path, int err) |  | ||||||
| { |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void freelist(struct match *head) |  | ||||||
| { |  | ||||||
| 	struct match *match, *next; |  | ||||||
| 	for (match=head->next; match; match=next) { |  | ||||||
| 		next = match->next; |  | ||||||
| 		free(match); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int sort(const void *a, const void *b) |  | ||||||
| { |  | ||||||
| 	return strcmp(*(const char **)a, *(const char **)b); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int glob(const char *restrict pat, int flags, int (*errfunc)(const char *path, int err), glob_t *restrict g) |  | ||||||
| { |  | ||||||
| 	struct match head = { .next = NULL }, *tail = &head; |  | ||||||
| 	size_t cnt, i; |  | ||||||
| 	size_t offs = (flags & GLOB_DOOFFS) ? g->gl_offs : 0; |  | ||||||
| 	int error = 0; |  | ||||||
| 	char buf[PATH_MAX]; |  | ||||||
| 	 |  | ||||||
| 	if (!errfunc) errfunc = ignore_err; |  | ||||||
| 
 |  | ||||||
| 	if (!(flags & GLOB_APPEND)) { |  | ||||||
| 		g->gl_offs = offs; |  | ||||||
| 		g->gl_pathc = 0; |  | ||||||
| 		g->gl_pathv = NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (*pat) { |  | ||||||
| 		char *p = strdup(pat); |  | ||||||
| 		if (!p) return GLOB_NOSPACE; |  | ||||||
| 		buf[0] = 0; |  | ||||||
| 		error = do_glob(buf, 0, 0, p, flags, errfunc, &tail); |  | ||||||
| 		free(p); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (error == GLOB_NOSPACE) { |  | ||||||
| 		freelist(&head); |  | ||||||
| 		return error; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	for (cnt=0, tail=head.next; tail; tail=tail->next, cnt++); |  | ||||||
| 	if (!cnt) { |  | ||||||
| 		if (flags & GLOB_NOCHECK) { |  | ||||||
| 			tail = &head; |  | ||||||
| 			if (append(&tail, pat, strlen(pat), 0)) |  | ||||||
| 				return GLOB_NOSPACE; |  | ||||||
| 			cnt++; |  | ||||||
| 		} else |  | ||||||
| 			return GLOB_NOMATCH; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (flags & GLOB_APPEND) { |  | ||||||
| 		char **pathv = realloc(g->gl_pathv, (offs + g->gl_pathc + cnt + 1) * sizeof(char *)); |  | ||||||
| 		if (!pathv) { |  | ||||||
| 			freelist(&head); |  | ||||||
| 			return GLOB_NOSPACE; |  | ||||||
| 		} |  | ||||||
| 		g->gl_pathv = pathv; |  | ||||||
| 		offs += g->gl_pathc; |  | ||||||
| 	} else { |  | ||||||
| 		g->gl_pathv = malloc((offs + cnt + 1) * sizeof(char *)); |  | ||||||
| 		if (!g->gl_pathv) { |  | ||||||
| 			freelist(&head); |  | ||||||
| 			return GLOB_NOSPACE; |  | ||||||
| 		} |  | ||||||
| 		for (i=0; i<offs; i++) |  | ||||||
| 			g->gl_pathv[i] = NULL; |  | ||||||
| 	} |  | ||||||
| 	for (i=0, tail=head.next; i<cnt; tail=tail->next, i++) |  | ||||||
| 		g->gl_pathv[offs + i] = tail->name; |  | ||||||
| 	g->gl_pathv[offs + i] = NULL; |  | ||||||
| 	g->gl_pathc += cnt; |  | ||||||
| 
 |  | ||||||
| 	if (!(flags & GLOB_NOSORT)) |  | ||||||
| 		qsort(g->gl_pathv+offs, cnt, sizeof(char *), sort); |  | ||||||
| 	 |  | ||||||
| 	return error; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void globfree(glob_t *g) |  | ||||||
| { |  | ||||||
| 	size_t i; |  | ||||||
| 	for (i=0; i<g->gl_pathc; i++) |  | ||||||
| 		free(g->gl_pathv[g->gl_offs + i] - offsetof(struct match, name)); |  | ||||||
| 	free(g->gl_pathv); |  | ||||||
| 	g->gl_pathc = 0; |  | ||||||
| 	g->gl_pathv = NULL; |  | ||||||
| } |  | ||||||
							
								
								
									
										2
									
								
								third_party/xed/x86ild.greg.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/xed/x86ild.greg.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1229,7 +1229,7 @@ privileged static void xed_decode_instruction_length( | ||||||
|  * |  * | ||||||
|  * @return d->decoded_length is byte length, or 1 w/ error code |  * @return d->decoded_length is byte length, or 1 w/ error code | ||||||
|  * @note binary footprint increases ~4kb if this is used |  * @note binary footprint increases ~4kb if this is used | ||||||
|  * @see cf. biggest thing in tensorflow, gdb, clang, etc. binaries |  * @see biggest code in gdb/clang/tensorflow binaries | ||||||
|  */ |  */ | ||||||
| privileged enum XedError xed_instruction_length_decode( | privileged enum XedError xed_instruction_length_decode( | ||||||
|     struct XedDecodedInst *xedd, const uint8_t *itext, const size_t bytes) { |     struct XedDecodedInst *xedd, const uint8_t *itext, const size_t bytes) { | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								third_party/xed/x86tab.S
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								third_party/xed/x86tab.S
									
										
									
									
										vendored
									
									
								
							|  | @ -1,25 +1,25 @@ | ||||||
| /*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8     -*-│
 | /*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8     -*-│
 | ||||||
| │vi: set et ft=asm ts=8 sw=8 fenc=utf-8                                     :vi│ | │vi: set et ft=asm ts=8 sw=8 fenc=utf-8                                     :vi│ | ||||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||||
|  | │ Copyright 2018 Intel Corporation                                             │ | ||||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||||
| │                                                                              │ | │                                                                              │ | ||||||
| │ This program is free software; you can redistribute it and/or modify         │
 | │ Licensed under the Apache License, Version 2.0 (the "License");              │
 | ||||||
| │ it under the terms of the GNU General Public License as published by         │ | │ you may not use this file except in compliance with the License.             │ | ||||||
| │ the Free Software Foundation; version 2 of the License.                      │
 | │ You may obtain a copy of the License at                                      │ | ||||||
| │                                                                              │ | │                                                                              │ | ||||||
| │ This program is distributed in the hope that it will be useful, but          │ | │     http://www.apache.org/licenses/LICENSE-2.0                               │ | ||||||
| │ WITHOUT ANY WARRANTY; without even the implied warranty of                   │
 |  | ||||||
| │ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU             │ |  | ||||||
| │ General Public License for more details.                                     │ |  | ||||||
| │                                                                              │ | │                                                                              │ | ||||||
| │ You should have received a copy of the GNU General Public License            │ | │ Unless required by applicable law or agreed to in writing, software          │ | ||||||
| │ along with this program; if not, write to the Free Software                  │
 | │ distributed under the License is distributed on an "AS IS" BASIS,            │ | ||||||
| │ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA                │ | │ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.     │ | ||||||
| │ 02110-1301 USA                                                               │ | │ See the License for the specific language governing permissions and          │ | ||||||
|  | │ limitations under the License.                                               │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/macros.h" | #include "libc/macros.h" | ||||||
| 
 | 
 | ||||||
| /	Phash tables for instruction length decoding. | /	Phash tables for instruction length decoding. | ||||||
|  | /	@see	build/rle.py for more context here
 | ||||||
| 
 | 
 | ||||||
| 	.initbss 300,_init_x86tab | 	.initbss 300,_init_x86tab | ||||||
| xed_prefix_table_bit: | xed_prefix_table_bit: | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								third_party/xed/xed.mk
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/xed/xed.mk
									
										
									
									
										vendored
									
									
								
							|  | @ -7,7 +7,7 @@ | ||||||
| #
 | #
 | ||||||
| # DESCRIPTION
 | # DESCRIPTION
 | ||||||
| #
 | #
 | ||||||
| #   See test/libc/x86ild_test.c for more information.
 | #   See test/libc/xed/x86ild_test.c for more information.
 | ||||||
| 
 | 
 | ||||||
| PKGS += THIRD_PARTY_XED | PKGS += THIRD_PARTY_XED | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										31
									
								
								tool/build/summy.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								tool/build/summy.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,31 @@ | ||||||
|  | /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||||
|  | │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||||
|  | ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||||
|  | │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ This program is free software; you can redistribute it and/or modify         │ | ||||||
|  | │ it under the terms of the GNU General Public License as published by         │ | ||||||
|  | │ the Free Software Foundation; version 2 of the License.                      │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ This program is distributed in the hope that it will be useful, but          │ | ||||||
|  | │ WITHOUT ANY WARRANTY; without even the implied warranty of                   │ | ||||||
|  | │ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU             │ | ||||||
|  | │ General Public License for more details.                                     │ | ||||||
|  | │                                                                              │ | ||||||
|  | │ You should have received a copy of the GNU General Public License            │ | ||||||
|  | │ along with this program; if not, write to the Free Software                  │ | ||||||
|  | │ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||||
|  | │ 02110-1301 USA                                                               │ | ||||||
|  | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
|  | #include "libc/stdio/stdio.h" | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * @fileoverview Sums per-line integers from stdin. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | int main(int argc, char *argv[]) { | ||||||
|  |   long x, sum = 0; | ||||||
|  |   while (scanf("%ld", &x) > 0) sum += x; | ||||||
|  |   printf("%,ld\n", sum); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | @ -62,6 +62,7 @@ | ||||||
| 
 | 
 | ||||||
| char *symbol_; | char *symbol_; | ||||||
| char *outpath_; | char *outpath_; | ||||||
|  | char *yoink_; | ||||||
| 
 | 
 | ||||||
| const size_t kMinCompressSize = 32; | const size_t kMinCompressSize = 32; | ||||||
| const char kNoCompressExts[][8] = {".gz",  ".xz",  ".jpg",  ".png", | const char kNoCompressExts[][8] = {".gz",  ".xz",  ".jpg",  ".png", | ||||||
|  | @ -70,13 +71,14 @@ const char kNoCompressExts[][8] = {".gz",  ".xz",  ".jpg",  ".png", | ||||||
| 
 | 
 | ||||||
| noreturn void PrintUsage(int rc, FILE *f) { | noreturn void PrintUsage(int rc, FILE *f) { | ||||||
|   fprintf(f, "%s%s%s\n", "Usage: ", program_invocation_name, |   fprintf(f, "%s%s%s\n", "Usage: ", program_invocation_name, | ||||||
|           " [-o FILE] [-s SYMBOL] [FILE...]\n"); |           " [-o FILE] [-s SYMBOL] [-y YOINK] [FILE...]\n"); | ||||||
|   exit(rc); |   exit(rc); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void GetOpts(int *argc, char ***argv) { | void GetOpts(int *argc, char ***argv) { | ||||||
|   int opt; |   int opt; | ||||||
|   while ((opt = getopt(*argc, *argv, "?ho:s:")) != -1) { |   yoink_ = "__zip_start"; | ||||||
|  |   while ((opt = getopt(*argc, *argv, "?ho:s:y:")) != -1) { | ||||||
|     switch (opt) { |     switch (opt) { | ||||||
|       case 'o': |       case 'o': | ||||||
|         outpath_ = optarg; |         outpath_ = optarg; | ||||||
|  | @ -84,6 +86,9 @@ void GetOpts(int *argc, char ***argv) { | ||||||
|       case 's': |       case 's': | ||||||
|         symbol_ = optarg; |         symbol_ = optarg; | ||||||
|         break; |         break; | ||||||
|  |       case 'y': | ||||||
|  |         yoink_ = optarg; | ||||||
|  |         break; | ||||||
|       case '?': |       case '?': | ||||||
|       case 'h': |       case 'h': | ||||||
|         PrintUsage(EXIT_SUCCESS, stdout); |         PrintUsage(EXIT_SUCCESS, stdout); | ||||||
|  | @ -273,8 +278,7 @@ void PullEndOfCentralDirectoryIntoLinkage(struct ElfWriter *elf) { | ||||||
|   elfwriter_align(elf, 1, 0); |   elfwriter_align(elf, 1, 0); | ||||||
|   elfwriter_startsection(elf, ".yoink", SHT_PROGBITS, |   elfwriter_startsection(elf, ".yoink", SHT_PROGBITS, | ||||||
|                          SHF_ALLOC | SHF_EXECINSTR); |                          SHF_ALLOC | SHF_EXECINSTR); | ||||||
|   elfwriter_yoink(elf, "__zip_start"); |   elfwriter_yoink(elf, yoink_); | ||||||
|   elfwriter_yoink(elf, "__zip_end"); |  | ||||||
|   elfwriter_finishsection(elf); |   elfwriter_finishsection(elf); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -337,4 +337,10 @@ | ||||||
|   (add-hook 'asm-mode-hook 'cosmo-asm-supplemental-hook) |   (add-hook 'asm-mode-hook 'cosmo-asm-supplemental-hook) | ||||||
|   (setq asm-font-lock-keywords cosmo-asm-font-lock-keywords)) |   (setq asm-font-lock-keywords cosmo-asm-font-lock-keywords)) | ||||||
| 
 | 
 | ||||||
|  | ;; Make -*-unix-assembly-*- mode line work correctly. | ||||||
|  | ;; TODO(jart): Would be nice to use GitHub's name instead of changing asm-mode. | ||||||
|  | (defun unix-assembly-mode () | ||||||
|  |   (interactive) | ||||||
|  |   (asm-mode)) | ||||||
|  | 
 | ||||||
| (provide 'cosmo-asm-mode) | (provide 'cosmo-asm-mode) | ||||||
|  |  | ||||||
|  | @ -6,7 +6,7 @@ sudo apt build-dep emacs | ||||||
| cd "$HOME" | cd "$HOME" | ||||||
| export CFLAGS="-O2" | export CFLAGS="-O2" | ||||||
| rm -rf emacs-26.3 | rm -rf emacs-26.3 | ||||||
| wget http://mirrors.kernel.org/gnu/emacs/emacs-26.3.tar.gz | wget https://mirrors.kernel.org/gnu/emacs/emacs-26.3.tar.gz | ||||||
| tar xf emacs-26.3.tar.gz | tar xf emacs-26.3.tar.gz | ||||||
| cd emacs-26.3 | cd emacs-26.3 | ||||||
| ./configure | ./configure | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue