mirror of
				https://github.com/janet-lang/janet
				synced 2025-10-31 07:33:01 +00:00 
			
		
		
		
	Major refactor and restructure. Add CMake for anticipated windows
support.
This commit is contained in:
		
							
								
								
									
										160
									
								
								src/core/wrap.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								src/core/wrap.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,160 @@ | ||||
| /* | ||||
| * Copyright (c) 2017 Calvin Rose | ||||
| * | ||||
| * 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 <dst/dst.h> | ||||
|  | ||||
| #ifdef DST_NANBOX | ||||
|  | ||||
| void *dst_nanbox_to_pointer(Dst x) { | ||||
|     /* We need to do this shift to keep the higher bits of the pointer | ||||
|      * the same as bit 47 as required by the x86 architecture. We may save | ||||
|      * an instruction if we do x.u64 & DST_NANBOX_POINTERBITS, but this 0s | ||||
|      * the high bits, and may make the pointer non-canocial on x86. If we switch | ||||
|      * to 47 bit pointers (which is what userspace uses on Windows, we can use | ||||
|      * the single mask rather than two shifts. */ | ||||
| #if defined (DST_NANBOX_47) || defined (DST_32) | ||||
|     x.i64 &= DST_NANBOX_POINTERBITS; | ||||
| #else | ||||
|     x.i64 = (x.i64 << 16) >> 16; | ||||
| #endif | ||||
|     return x.pointer; | ||||
| } | ||||
|  | ||||
| Dst dst_nanbox_from_pointer(void *p, uint64_t tagmask) { | ||||
|     Dst ret; | ||||
|     ret.pointer = p; | ||||
| #if defined (DST_NANBOX_47) || defined (DST_32) | ||||
| #else | ||||
|     ret.u64 &= DST_NANBOX_POINTERBITS; | ||||
| #endif | ||||
|     ret.u64 |= tagmask; | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| Dst dst_nanbox_from_cpointer(const void *p, uint64_t tagmask) { | ||||
|     Dst ret; | ||||
|     ret.cpointer = p; | ||||
| #if defined (DST_NANBOX_47) || defined (DST_32) | ||||
| #else | ||||
|     ret.u64 &= DST_NANBOX_POINTERBITS; | ||||
| #endif | ||||
|     ret.u64 |= tagmask; | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| Dst dst_nanbox_from_double(double d) { | ||||
|     Dst ret; | ||||
|     ret.real = d; | ||||
|     /* Normalize NaNs */ | ||||
|     if (d != d) | ||||
|         ret.u64 = dst_nanbox_tag(DST_REAL); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| Dst dst_nanbox_from_bits(uint64_t bits) { | ||||
|     Dst ret; | ||||
|     ret.u64 = bits; | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| void *dst_nanbox_memalloc_empty(int32_t count) { | ||||
|     int32_t i; | ||||
|     void *mem = malloc(count * sizeof(DstKV)); | ||||
|     DstKV *mmem = (DstKV *)mem; | ||||
|     for (i = 0; i < count; i++) { | ||||
|         DstKV *kv = mmem + i; | ||||
|         kv->key = dst_wrap_nil(); | ||||
|         kv->value = dst_wrap_nil(); | ||||
|     } | ||||
|     return mem; | ||||
| } | ||||
|  | ||||
| void dst_nanbox_memempty(DstKV *mem, int32_t count) { | ||||
|     int32_t i; | ||||
|     for (i = 0; i < count; i++) { | ||||
|         mem[i].key = dst_wrap_nil(); | ||||
|         mem[i].value = dst_wrap_nil(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #else | ||||
|  | ||||
| /* Wrapper functions wrap a data type that is used from C into a | ||||
|  * dst value, which can then be used in dst internal functions. Use | ||||
|  * these functions sparingly, as these function will let the programmer | ||||
|  * leak memory, where as the stack based API ensures that all values can | ||||
|  * be collected by the garbage collector. */ | ||||
|  | ||||
| Dst dst_wrap_nil() { | ||||
|     Dst y; | ||||
|     y.type = DST_NIL; | ||||
|     y.as.u64 = 0; | ||||
|     return y; | ||||
| } | ||||
|  | ||||
| Dst dst_wrap_true() { | ||||
|     Dst y; | ||||
|     y.type = DST_TRUE; | ||||
|     y.as.u64 = 0; | ||||
|     return y; | ||||
| } | ||||
|  | ||||
| Dst dst_wrap_false() { | ||||
|     Dst y; | ||||
|     y.type = DST_FALSE; | ||||
|     y.as.u64 = 0; | ||||
|     return y; | ||||
| } | ||||
|  | ||||
| Dst dst_wrap_boolean(int x) { | ||||
|     Dst y; | ||||
|     y.type = x ? DST_TRUE : DST_FALSE; | ||||
|     y.as.u64 = 0; | ||||
|     return y; | ||||
| } | ||||
|  | ||||
| #define DST_WRAP_DEFINE(NAME, TYPE, DTYPE, UM)\ | ||||
| Dst dst_wrap_##NAME(TYPE x) {\ | ||||
|     Dst y;\ | ||||
|     y.type = DTYPE;\ | ||||
|     y.as.u64 = 0; /* zero other bits in case of 32 bit integer */ \ | ||||
|     y.as.UM = x;\ | ||||
|     return y;\ | ||||
| } | ||||
|  | ||||
| DST_WRAP_DEFINE(real, double, DST_REAL, real) | ||||
| DST_WRAP_DEFINE(integer, int32_t, DST_INTEGER, integer) | ||||
| DST_WRAP_DEFINE(string, const uint8_t *, DST_STRING, cpointer) | ||||
| DST_WRAP_DEFINE(symbol, const uint8_t *, DST_SYMBOL, cpointer) | ||||
| DST_WRAP_DEFINE(array, DstArray *, DST_ARRAY, pointer) | ||||
| DST_WRAP_DEFINE(tuple, const Dst *, DST_TUPLE, cpointer) | ||||
| DST_WRAP_DEFINE(struct, const DstKV *, DST_STRUCT, cpointer) | ||||
| DST_WRAP_DEFINE(fiber, DstFiber *, DST_FIBER, pointer) | ||||
| DST_WRAP_DEFINE(buffer, DstBuffer *, DST_BUFFER, pointer) | ||||
| DST_WRAP_DEFINE(function, DstFunction *, DST_FUNCTION, pointer) | ||||
| DST_WRAP_DEFINE(cfunction, DstCFunction, DST_CFUNCTION, pointer) | ||||
| DST_WRAP_DEFINE(table, DstTable *, DST_TABLE, pointer) | ||||
| DST_WRAP_DEFINE(abstract, void *, DST_ABSTRACT, pointer) | ||||
|  | ||||
| #undef DST_WRAP_DEFINE | ||||
|  | ||||
| #endif | ||||
		Reference in New Issue
	
	Block a user
	 bakpakin
					bakpakin