mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 11:10:58 +00:00 
			
		
		
		
	Improve performance of bitscanning intrinsics
This change helps spectre more intelligently plan execution, by working around false output dependencies, impacting ops like popcnt bsr and bsf
This commit is contained in:
		
							parent
							
								
									29cb53881e
								
							
						
					
					
						commit
						87d7010495
					
				
					 13 changed files with 100 additions and 224 deletions
				
			
		|  | @ -9,13 +9,13 @@ unsigned long popcnt(unsigned long) pureconst; | ||||||
| #if defined(__GNUC__) && !defined(__STRICT_ANSI__) | #if defined(__GNUC__) && !defined(__STRICT_ANSI__) | ||||||
| #define popcnt(X)                                                \ | #define popcnt(X)                                                \ | ||||||
|   (__builtin_constant_p(X) ? __builtin_popcountll(X) : ({        \ |   (__builtin_constant_p(X) ? __builtin_popcountll(X) : ({        \ | ||||||
|     unsigned long Res, Pop = (X);                         \ |     unsigned long PoP = (X);                                     \ | ||||||
|     if (X86_HAVE(POPCNT)) {                                      \ |     if (X86_HAVE(POPCNT)) {                                      \ | ||||||
|       asm("popcnt\t%1,%0" : "=r"(Res) : "r"(Pop) : "cc"); \ |       asm("popcnt\t%0,%0" : "+r"(PoP) : /* no inputs */ : "cc"); \ | ||||||
|     } else {                                                     \ |     } else {                                                     \ | ||||||
|       Res = (popcnt)(Pop);                                \ |       PoP = (popcnt)(PoP);                                       \ | ||||||
|     }                                                            \ |     }                                                            \ | ||||||
|     Res;                                                  \ |     PoP;                                                         \ | ||||||
|   })) |   })) | ||||||
| #endif /* GNUC && !ANSI */ | #endif /* GNUC && !ANSI */ | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -22,10 +22,20 @@ int bsfl(long); | ||||||
| int bsfll(long long); | int bsfll(long long); | ||||||
| int bsfmax(uintmax_t); | int bsfmax(uintmax_t); | ||||||
| 
 | 
 | ||||||
| #ifdef __GNUC__ | #if defined(__GNUC__) && !defined(__STRICT_ANSI__) | ||||||
| #define bsf(u)   __builtin_ctz(u) | #define bsf(u)                                                  \ | ||||||
| #define bsfl(u)  __builtin_ctzl(u) |   ({                                                            \ | ||||||
| #define bsfll(u) __builtin_ctzll(u) |     unsigned BiTs;                                              \ | ||||||
|  |     asm("bsf\t%0,%0" : "=r"(BiTs) : "0"((unsigned)(u)) : "cc"); \ | ||||||
|  |     BiTs;                                                       \ | ||||||
|  |   }) | ||||||
|  | #define bsfl(u)                                                      \ | ||||||
|  |   ({                                                                 \ | ||||||
|  |     unsigned long BiTs;                                              \ | ||||||
|  |     asm("bsf\t%0,%0" : "=r"(BiTs) : "0"((unsigned long)(u)) : "cc"); \ | ||||||
|  |     (unsigned)BiTs;                                                  \ | ||||||
|  |   }) | ||||||
|  | #define bsfll(u) bsfl(u) | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
|  |  | ||||||
|  | @ -22,10 +22,20 @@ int bsrl(long); | ||||||
| int bsrll(long long); | int bsrll(long long); | ||||||
| int bsrmax(uintmax_t); | int bsrmax(uintmax_t); | ||||||
| 
 | 
 | ||||||
| #ifdef __GNUC__ | #if defined(__GNUC__) && defined(__x86_64__) && !defined(__STRICT_ANSI__) | ||||||
| #define bsr(u)   ((sizeof(int) * 8 - 1) ^ __builtin_clz(u)) | #define bsr(u)                                                  \ | ||||||
| #define bsrl(u)  ((sizeof(long) * 8 - 1) ^ __builtin_clzl(u)) |   ({                                                            \ | ||||||
| #define bsrll(u) ((sizeof(long long) * 8 - 1) ^ __builtin_clzll(u)) |     unsigned BiTs;                                              \ | ||||||
|  |     asm("bsr\t%0,%0" : "=r"(BiTs) : "0"((unsigned)(u)) : "cc"); \ | ||||||
|  |     BiTs;                                                       \ | ||||||
|  |   }) | ||||||
|  | #define bsrl(u)                                                      \ | ||||||
|  |   ({                                                                 \ | ||||||
|  |     unsigned long BiTs;                                              \ | ||||||
|  |     asm("bsr\t%0,%0" : "=r"(BiTs) : "0"((unsigned long)(u)) : "cc"); \ | ||||||
|  |     (unsigned)BiTs;                                                  \ | ||||||
|  |   }) | ||||||
|  | #define bsrll(u) bsrl(u) | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| COSMOPOLITAN_C_END_ | COSMOPOLITAN_C_END_ | ||||||
|  |  | ||||||
|  | @ -1,45 +0,0 @@ | ||||||
| /*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8     -*-│
 |  | ||||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ |  | ||||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ |  | ||||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ Permission to use, copy, modify, and/or distribute this software for         │ |  | ||||||
| │ any purpose with or without fee is hereby granted, provided that the         │ |  | ||||||
| │ above copyright notice and this permission notice appear in all copies.      │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │ |  | ||||||
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │ |  | ||||||
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │ |  | ||||||
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL         │ |  | ||||||
| │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR        │ |  | ||||||
| │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER               │ |  | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ |  | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ |  | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ |  | ||||||
| #include "libc/macros.internal.h" |  | ||||||
| 
 |  | ||||||
| //	Finds leading bits in 𝑥. |  | ||||||
| // |  | ||||||
| //	      uint32 𝑥   bsf(𝑥) tzcnt(𝑥)   ffs(𝑥)   bsr(𝑥) lzcnt(𝑥) |  | ||||||
| //	    0x00000000      wut       32        0      wut       32 |  | ||||||
| //	    0x00000001        0        0        1        0       31 |  | ||||||
| //	    0x80000001        0        0        1       31        0 |  | ||||||
| //	    0x80000000       31       31       32       31        0 |  | ||||||
| //	    0x00000010        4        4        5        4       27 |  | ||||||
| //	    0x08000010        4        4        5       27        4 |  | ||||||
| //	    0x08000000       27       27       28       27        4 |  | ||||||
| //	    0xffffffff        0        0        1       31        0 |  | ||||||
| // |  | ||||||
| //	@param	edi is 32-bit unsigned 𝑥 value
 |  | ||||||
| //	@return	eax number in range [0,32) or 32 if 𝑥 is 0
 |  | ||||||
| //	@see	also treasure trove of nearly identical functions
 |  | ||||||
| lzcnt:	.leafprologue
 |  | ||||||
| 	.profilable |  | ||||||
| 	mov	$31,%eax |  | ||||||
| 	mov	$-1,%edx |  | ||||||
| 	bsr	%edi,%ecx |  | ||||||
| 	cmovz	%edx,%ecx |  | ||||||
| 	sub	%ecx,%eax |  | ||||||
| 	.leafepilogue |  | ||||||
| 	.endfn	lzcnt,globl |  | ||||||
| 	.source	__FILE__
 |  | ||||||
|  | @ -1,25 +0,0 @@ | ||||||
| #ifndef COSMOPOLITAN_LIBC_NEXGEN32E_LZCNT_H_ |  | ||||||
| #define COSMOPOLITAN_LIBC_NEXGEN32E_LZCNT_H_ |  | ||||||
| #include "libc/nexgen32e/x86feature.h" |  | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) |  | ||||||
| COSMOPOLITAN_C_START_ |  | ||||||
| 
 |  | ||||||
| unsigned lzcnt(unsigned) libcesque pureconst; |  | ||||||
| unsigned long lzcntl(unsigned long) libcesque pureconst; |  | ||||||
| 
 |  | ||||||
| #define lzcnt(X)                                           \ |  | ||||||
|   ({                                                       \ |  | ||||||
|     typeof(X) Res;                                         \ |  | ||||||
|     if (X86_HAVE(ABM)) {                                   \ |  | ||||||
|       /* amd piledriver+ (a.k.a. bdver1) c. 2011 */        \ |  | ||||||
|       /* intel haswell+ c. 2013 */                         \ |  | ||||||
|       asm("lzcnt\t%1,%0" : "=r,r"(Res) : "r,m"(X) : "cc"); \ |  | ||||||
|     } else {                                               \ |  | ||||||
|       Res = (lzcnt)(X);                                    \ |  | ||||||
|     }                                                      \ |  | ||||||
|     Res;                                                   \ |  | ||||||
|   }) |  | ||||||
| 
 |  | ||||||
| COSMOPOLITAN_C_END_ |  | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ |  | ||||||
| #endif /* COSMOPOLITAN_LIBC_NEXGEN32E_LZCNT_H_ */ |  | ||||||
|  | @ -1,45 +0,0 @@ | ||||||
| /*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8     -*-│
 |  | ||||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ |  | ||||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ |  | ||||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ Permission to use, copy, modify, and/or distribute this software for         │ |  | ||||||
| │ any purpose with or without fee is hereby granted, provided that the         │ |  | ||||||
| │ above copyright notice and this permission notice appear in all copies.      │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │ |  | ||||||
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │ |  | ||||||
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │ |  | ||||||
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL         │ |  | ||||||
| │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR        │ |  | ||||||
| │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER               │ |  | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ |  | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ |  | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ |  | ||||||
| #include "libc/macros.internal.h" |  | ||||||
| 
 |  | ||||||
| //	Finds leading bits in 𝑥. |  | ||||||
| // |  | ||||||
| //	      uint32 𝑥   bsf(𝑥) tzcnt(𝑥)   ffs(𝑥)   bsr(𝑥) lzcnt(𝑥) |  | ||||||
| //	    0x00000000      wut       32        0      wut       32 |  | ||||||
| //	    0x00000001        0        0        1        0       31 |  | ||||||
| //	    0x80000001        0        0        1       31        0 |  | ||||||
| //	    0x80000000       31       31       32       31        0 |  | ||||||
| //	    0x00000010        4        4        5        4       27 |  | ||||||
| //	    0x08000010        4        4        5       27        4 |  | ||||||
| //	    0x08000000       27       27       28       27        4 |  | ||||||
| //	    0xffffffff        0        0        1       31        0 |  | ||||||
| // |  | ||||||
| //	@param	rdi is 64-bit unsigned 𝑥 value
 |  | ||||||
| //	@return	rax number in range [0,64) or 64 if 𝑥 is 0
 |  | ||||||
| //	@see	also treasure trove of nearly identical functions
 |  | ||||||
| lzcntl:	.leafprologue
 |  | ||||||
| 	.profilable |  | ||||||
| 	mov	$31,%eax |  | ||||||
| 	mov	$-1,%edx |  | ||||||
| 	bsr	%rdi,%rcx |  | ||||||
| 	cmovz	%edx,%ecx |  | ||||||
| 	sub	%ecx,%eax |  | ||||||
| 	.endfn	lzcntl,globl |  | ||||||
| 	.alias	lzcntl,lzcntll |  | ||||||
| 	.source	__FILE__
 |  | ||||||
|  | @ -1,25 +0,0 @@ | ||||||
| #ifndef COSMOPOLITAN_LIBC_NEXGEN32E_TZCNT_H_ |  | ||||||
| #define COSMOPOLITAN_LIBC_NEXGEN32E_TZCNT_H_ |  | ||||||
| #include "libc/nexgen32e/x86feature.h" |  | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) |  | ||||||
| COSMOPOLITAN_C_START_ |  | ||||||
| 
 |  | ||||||
| unsigned tzcnt(unsigned) libcesque pureconst; |  | ||||||
| unsigned long tzcntl(unsigned long) libcesque pureconst; |  | ||||||
| 
 |  | ||||||
| #define tzcnt(X)                                           \ |  | ||||||
|   ({                                                       \ |  | ||||||
|     typeof(X) Res;                                         \ |  | ||||||
|     if (X86_HAVE(BMI)) {                                   \ |  | ||||||
|       /* amd piledriver+ (a.k.a. bdver1) c. 2011 */        \ |  | ||||||
|       /* intel haswell+ c. 2013 */                         \ |  | ||||||
|       asm("tzcnt\t%1,%0" : "=r,r"(Res) : "r,m"(X) : "cc"); \ |  | ||||||
|     } else {                                               \ |  | ||||||
|       Res = (tzcnt)(X);                                    \ |  | ||||||
|     }                                                      \ |  | ||||||
|     Res;                                                   \ |  | ||||||
|   }) |  | ||||||
| 
 |  | ||||||
| COSMOPOLITAN_C_END_ |  | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ |  | ||||||
| #endif /* COSMOPOLITAN_LIBC_NEXGEN32E_TZCNT_H_ */ |  | ||||||
|  | @ -1,44 +0,0 @@ | ||||||
| /*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8     -*-│
 |  | ||||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ |  | ||||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ |  | ||||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ Permission to use, copy, modify, and/or distribute this software for         │ |  | ||||||
| │ any purpose with or without fee is hereby granted, provided that the         │ |  | ||||||
| │ above copyright notice and this permission notice appear in all copies.      │ |  | ||||||
| │                                                                              │ |  | ||||||
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │ |  | ||||||
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │ |  | ||||||
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │ |  | ||||||
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL         │ |  | ||||||
| │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR        │ |  | ||||||
| │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER               │ |  | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ |  | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ |  | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ |  | ||||||
| #include "libc/macros.internal.h" |  | ||||||
| 
 |  | ||||||
| //	Finds lowest set bit in 𝑥. |  | ||||||
| // |  | ||||||
| //	      uint32 𝑥   bsf(𝑥) tzcnt(𝑥)   ffs(𝑥)   bsr(𝑥) lzcnt(𝑥) |  | ||||||
| //	    0x00000000      wut       32        0      wut       32 |  | ||||||
| //	    0x00000001        0        0        1        0       31 |  | ||||||
| //	    0x80000001        0        0        1       31        0 |  | ||||||
| //	    0x80000000       31       31       32       31        0 |  | ||||||
| //	    0x00000010        4        4        5        4       27 |  | ||||||
| //	    0x08000010        4        4        5       27        4 |  | ||||||
| //	    0x08000000       27       27       28       27        4 |  | ||||||
| //	    0xffffffff        0        0        1       31        0 |  | ||||||
| // |  | ||||||
| //	@param	rdi is 64-bit unsigned 𝑥 value
 |  | ||||||
| //	@return	rax number in range [0,64) or 64 if 𝑥 is 0
 |  | ||||||
| //	@see	also treasure trove of nearly identical functions
 |  | ||||||
| tzcntl:	.leafprologue
 |  | ||||||
| 	.profilable |  | ||||||
| 	mov	$64,%esi |  | ||||||
| 	bsf	%rdi,%rax |  | ||||||
| 	cmovz	%esi,%eax |  | ||||||
| 	.leafepilogue |  | ||||||
| 	.endfn	tzcntl,globl |  | ||||||
| 	.alias	tzcntl,tzcntll |  | ||||||
| 	.source	__FILE__
 |  | ||||||
|  | @ -18,26 +18,16 @@ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/macros.internal.h" | #include "libc/macros.internal.h" | ||||||
| .source	__FILE__
 | .source	__FILE__
 | ||||||
|  | .rodata | ||||||
| 
 | 
 | ||||||
| //	Finds lowest set bit in 𝑥. | //	Nontrivial NUL-terminated string test vector. | ||||||
| // | 	.align	1
 | ||||||
| //	      uint32 𝑥   bsf(𝑥) tzcnt(𝑥)   ffs(𝑥)   bsr(𝑥) lzcnt(𝑥) | kBlocktronics: | ||||||
| //	    0x00000000      wut       32        0      wut       32 | 0:	.incbin	"libc/testlib/blocktronics.txt" | ||||||
| //	    0x00000001        0        0        1        0       31 | 1:	.byte	0 | ||||||
| //	    0x80000001        0        0        1       31        0 | 	.endobj	kBlocktronics,globl | ||||||
| //	    0x80000000       31       31       32       31        0 | 
 | ||||||
| //	    0x00000010        4        4        5        4       27 | 	.align	8
 | ||||||
| //	    0x08000010        4        4        5       27        4 | kBlocktronicsSize: | ||||||
| //	    0x08000000       27       27       28       27        4 | 	.quad	1b-0b | ||||||
| //	    0xffffffff        0        0        1       31        0 | 	.endobj	kBlocktronicsSize,globl | ||||||
| // |  | ||||||
| //	@param	edi is 32-bit unsigned 𝑥 value
 |  | ||||||
| //	@return	eax number in range [0,32) or 32 if 𝑥 is 0
 |  | ||||||
| //	@see	also treasure trove of nearly identical functions
 |  | ||||||
| tzcnt:	.leafprologue
 |  | ||||||
| 	.profilable |  | ||||||
| 	mov	$32,%esi |  | ||||||
| 	bsf	%edi,%eax |  | ||||||
| 	cmovz	%esi,%eax |  | ||||||
| 	.leafepilogue |  | ||||||
| 	.endfn	tzcnt,globl |  | ||||||
							
								
								
									
										12
									
								
								libc/testlib/blocktronics.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								libc/testlib/blocktronics.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,12 @@ | ||||||
|  | #ifndef COSMOPOLITAN_LIBC_TESTLIB_BLOCKTRONICS_H_ | ||||||
|  | #define COSMOPOLITAN_LIBC_TESTLIB_BLOCKTRONICS_H_ | ||||||
|  | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
|  | COSMOPOLITAN_C_START_ | ||||||
|  | 
 | ||||||
|  | extern size_t kBlocktronicsSize; | ||||||
|  | extern char kBlocktronics[]; | ||||||
|  | extern uint8_t kBlocktronicsZip[]; | ||||||
|  | 
 | ||||||
|  | COSMOPOLITAN_C_END_ | ||||||
|  | #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||||
|  | #endif /* COSMOPOLITAN_LIBC_TESTLIB_BLOCKTRONICS_H_ */ | ||||||
							
								
								
									
										32
									
								
								libc/testlib/blocktronics.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								libc/testlib/blocktronics.txt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,32 @@ | ||||||
|  |             ░▄██▒▄█ ▐██ ░░░     ▀▀████▒▀█▄ | ||||||
|  |            ▐███▓██░ ██▌            ▀████▄■█▄ | ||||||
|  |           ▐█▓███▀█░██▀   ░          ░▀█████▓▄ | ||||||
|  |          ▐█▓██▀▄█▒██▀  ▄▄░  ▄▄▄ ░░░   ░▀████▒▄ | ||||||
|  |         ▐████▀▄█■█▀      ▀▀             ░█████░ | ||||||
|  |        ▐█▓█▀████▀          ░             ▐▓███▒ | ||||||
|  |        █░███▀▀     ░░░    ▄█       ░░░    █████ | ||||||
|  |       ▐█▓█░▀▀  ░░▄█▄▄▄▄▄  ▀▄ ▌▄▄▄░▄▄▄▄▄   ▐████░ | ||||||
|  |       ▐███▌   ▄▀█████████▄ ▌▐▄████████▄   ▐▓███░ | ||||||
|  |       ▐███░░░▀▄█▀▄▄████▄▀░  ▐████████▒ ▀   ░███░ | ||||||
|  |       ░████░ ▓▀ ▄███████▀▌   ▀▄■████▀▀█▀   ██▀█ | ||||||
|  |        ▓███░ ░▄▀▀░░░ ▀  ░░▌   ▄▀▀▄░░▀░▄▀▄ ▐██▀▄ | ||||||
|  |        ░███░  ▄▓▓▄▄░▀▀█▀█ ▌░░  ▀█▀█▀▀     ▐██▀ | ||||||
|  |      █▀▄▐██   ▀░░   ▄▀ ▐ █    ▀ ▄▄▄░     ░▀▄█▄▀█ | ||||||
|  |      ▌▄  █▓ ▒      ░  █▄█▄▀▄▄▄███▄▀▄ ░░   ░ ▀  █▌ | ||||||
|  |       █▌▄░▌      ░░░▄▀█▀███████▄▀▄▀▄▀▀▄▄▄  █▀█░▐ | ||||||
|  |        ██▄     ░░░▄█▄▀██▄█■██████▄█▄█▄■▀█░  ▐░▐ | ||||||
|  |         ▀██░   ░▄██████████████████▄█▄█ ░█ ░ ▄▀ | ||||||
|  |         ▀▓█▄▓░░  ▒█▀█████████████████████▒ ██▀ | ||||||
|  |          ▀███ ▓▒   ██████████████▀▀▀▀█▄▀ ░▄█▒ | ||||||
|  |           ▀███ ▀█▄▀▄█████▀▀  ▓▓▓▄░   ▐  ░▄██ | ||||||
|  |             ▀██ ▄███████▄████████▀░░   ░▄██ | ||||||
|  |       ▄██▀▀▄ █▄▀▄██▒▒███████████▀▀▀▄░ ░███░ | ||||||
|  |     ▄██▀▄▄░░▀▐▄████▄ █████▀▄░░█▀▄▀░░  ▄██░ | ||||||
|  |     █████▄▄▄███▀░█▌██▄▀▀█████▄▄░░░▄▄███▀██▄ ▄▀▀▀▄▄ | ||||||
|  |      ▀██████▀■▄█▄▄ ░▀███████████████▓▓░░▄██▀▄████▄▄▀▄ | ||||||
|  |      | ||||||
|  |     █▀█ █  █▀█ █▀█ █▄▀ ▐▀█▀▌█▀█ █▀█ █▄ █ ▀█▀ █▀█ █▀▀ | ||||||
|  |     █▀▄ █  █ █ █   █ ▀▄  █  █▀▄ █ █ █ ▀█  █  █   ▀▀█ | ||||||
|  |     █▄█ █▄▌█▄█ █▄█ █  █  █  █ █ █▄█ █  █ ▄█▄ █▄█ █▄█ | ||||||
|  |      | ||||||
|  |     THERE WILL BE BLOCKS               march 01 2017 | ||||||
|  | @ -13,11 +13,13 @@ LIBC_TESTLIB_A = o/$(MODE)/libc/testlib/testlib.a | ||||||
| LIBC_TESTLIB_A_CHECKS = $(LIBC_TESTLIB_A).pkg | LIBC_TESTLIB_A_CHECKS = $(LIBC_TESTLIB_A).pkg | ||||||
| 
 | 
 | ||||||
| LIBC_TESTLIB_A_ASSETS =						\
 | LIBC_TESTLIB_A_ASSETS =						\
 | ||||||
| 	libc/testlib/hyperion.txt | 	libc/testlib/hyperion.txt				\
 | ||||||
|  | 	libc/testlib/blocktronics.txt | ||||||
| 
 | 
 | ||||||
| LIBC_TESTLIB_A_HDRS =						\
 | LIBC_TESTLIB_A_HDRS =						\
 | ||||||
| 	libc/testlib/bench.h					\
 | 	libc/testlib/bench.h					\
 | ||||||
| 	libc/testlib/ezbench.h					\
 | 	libc/testlib/ezbench.h					\
 | ||||||
|  | 	libc/testlib/blocktronics.h				\
 | ||||||
| 	libc/testlib/hyperion.h					\
 | 	libc/testlib/hyperion.h					\
 | ||||||
| 	libc/testlib/testlib.h | 	libc/testlib/testlib.h | ||||||
| 
 | 
 | ||||||
|  | @ -25,6 +27,7 @@ LIBC_TESTLIB_A_SRCS_S =						\ | ||||||
| 	libc/testlib/bench.S					\
 | 	libc/testlib/bench.S					\
 | ||||||
| 	libc/testlib/combo.S					\
 | 	libc/testlib/combo.S					\
 | ||||||
| 	libc/testlib/fixture.S					\
 | 	libc/testlib/fixture.S					\
 | ||||||
|  | 	libc/testlib/blocktronics.S				\
 | ||||||
| 	libc/testlib/hyperion.S					\
 | 	libc/testlib/hyperion.S					\
 | ||||||
| 	libc/testlib/testcase.S					\
 | 	libc/testlib/testcase.S					\
 | ||||||
| 	libc/testlib/thrashcodecache.S				\
 | 	libc/testlib/thrashcodecache.S				\
 | ||||||
|  |  | ||||||
|  | @ -16,6 +16,7 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
|  | #include "libc/testlib/blocktronics.h" | ||||||
| #include "libc/testlib/ezbench.h" | #include "libc/testlib/ezbench.h" | ||||||
| #include "libc/testlib/hyperion.h" | #include "libc/testlib/hyperion.h" | ||||||
| #include "libc/testlib/testlib.h" | #include "libc/testlib/testlib.h" | ||||||
|  | @ -48,4 +49,6 @@ BENCH(HasControlCodes, bench) { | ||||||
|   EZBENCH2("HasControlCodes small", donothing, HasControlCodes("hello", -1, 0)); |   EZBENCH2("HasControlCodes small", donothing, HasControlCodes("hello", -1, 0)); | ||||||
|   EZBENCH2("HasControlCodes big", donothing, |   EZBENCH2("HasControlCodes big", donothing, | ||||||
|            HasControlCodes(kHyperion, kHyperionSize, kControlC1)); |            HasControlCodes(kHyperion, kHyperionSize, kControlC1)); | ||||||
|  |   EZBENCH2("HasControlCodes unicode", donothing, | ||||||
|  |            HasControlCodes(kBlocktronics, kBlocktronicsSize, kControlC1)); | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue