mirror of
				https://github.com/janet-lang/janet
				synced 2025-10-24 20:27:41 +00:00 
			
		
		
		
	Add buffer/bit functions and buffer/blit. Expose janet_gethalfrange
in the C api for less duplicated range checking code.
This commit is contained in:
		| @@ -55,7 +55,8 @@ void janet_buffer_ensure(JanetBuffer *buffer, int32_t capacity, int32_t growth) | |||||||
|     uint8_t *new_data; |     uint8_t *new_data; | ||||||
|     uint8_t *old = buffer->data; |     uint8_t *old = buffer->data; | ||||||
|     if (capacity <= buffer->capacity) return; |     if (capacity <= buffer->capacity) return; | ||||||
|     capacity *= growth; |     int64_t big_capacity = capacity * growth; | ||||||
|  |     capacity = big_capacity > INT32_MAX ? INT32_MAX : big_capacity; | ||||||
|     new_data = realloc(old, capacity * sizeof(uint8_t)); |     new_data = realloc(old, capacity * sizeof(uint8_t)); | ||||||
|     if (NULL == new_data) { |     if (NULL == new_data) { | ||||||
|         JANET_OUT_OF_MEMORY; |         JANET_OUT_OF_MEMORY; | ||||||
| @@ -161,6 +162,19 @@ static Janet cfun_new(int32_t argc, Janet *argv) { | |||||||
|     return janet_wrap_buffer(buffer); |     return janet_wrap_buffer(buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | static Janet cfun_new_filled(int32_t argc, Janet *argv) { | ||||||
|  |     janet_arity(argc, 1, 2); | ||||||
|  |     int32_t count = janet_getinteger(argv, 0); | ||||||
|  |     int32_t byte = 0; | ||||||
|  |     if (argc == 2) { | ||||||
|  |         byte = janet_getinteger(argv, 1) & 0xFF; | ||||||
|  |     } | ||||||
|  |     JanetBuffer *buffer = janet_buffer(count); | ||||||
|  |     memset(buffer->data, byte, count); | ||||||
|  |     buffer->count = count; | ||||||
|  |     return janet_wrap_buffer(buffer); | ||||||
|  | } | ||||||
|  |  | ||||||
| static Janet cfun_u8(int32_t argc, Janet *argv) { | static Janet cfun_u8(int32_t argc, Janet *argv) { | ||||||
|     int32_t i; |     int32_t i; | ||||||
|     janet_arity(argc, 1, -1); |     janet_arity(argc, 1, -1); | ||||||
| @@ -225,12 +239,93 @@ static Janet cfun_slice(int32_t argc, Janet *argv) { | |||||||
|     return janet_wrap_buffer(buffer); |     return janet_wrap_buffer(buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | static void bitloc(int32_t argc, Janet *argv, JanetBuffer **b, int32_t *index, int *bit) { | ||||||
|  |     janet_fixarity(argc, 2); | ||||||
|  |     JanetBuffer *buffer = janet_getbuffer(argv, 0); | ||||||
|  |     double x = janet_getnumber(argv, 1); | ||||||
|  |     int64_t bitindex = (int64_t) x; | ||||||
|  |     int64_t byteindex = bitindex >> 3; | ||||||
|  |     int which_bit = bitindex & 7; | ||||||
|  |     if (bitindex != x || bitindex < 0 || byteindex >= buffer->count) | ||||||
|  |         janet_panicf("invalid bit index %v", argv[1]); | ||||||
|  |     *b = buffer; | ||||||
|  |     *index = byteindex; | ||||||
|  |     *bit = which_bit; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static Janet cfun_bitset(int32_t argc, Janet *argv) { | ||||||
|  |     int bit; | ||||||
|  |     int32_t index; | ||||||
|  |     JanetBuffer *buffer; | ||||||
|  |     bitloc(argc, argv, &buffer, &index, &bit); | ||||||
|  |     buffer->data[index] |= 1 << bit; | ||||||
|  |     return argv[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static Janet cfun_bitclear(int32_t argc, Janet *argv) { | ||||||
|  |     int bit; | ||||||
|  |     int32_t index; | ||||||
|  |     JanetBuffer *buffer; | ||||||
|  |     bitloc(argc, argv, &buffer, &index, &bit); | ||||||
|  |     buffer->data[index] &= ~(1 << bit); | ||||||
|  |     return argv[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static Janet cfun_bitget(int32_t argc, Janet *argv) { | ||||||
|  |     int bit; | ||||||
|  |     int32_t index; | ||||||
|  |     JanetBuffer *buffer; | ||||||
|  |     bitloc(argc, argv, &buffer, &index, &bit); | ||||||
|  |     return janet_wrap_boolean(buffer->data[index] & (1 << bit)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static Janet cfun_bittoggle(int32_t argc, Janet *argv) { | ||||||
|  |     int bit; | ||||||
|  |     int32_t index; | ||||||
|  |     JanetBuffer *buffer; | ||||||
|  |     bitloc(argc, argv, &buffer, &index, &bit); | ||||||
|  |     buffer->data[index] ^= (1 << bit); | ||||||
|  |     return argv[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static Janet cfun_blit(int32_t argc, Janet *argv) { | ||||||
|  |     janet_arity(argc, 2, 5); | ||||||
|  |     JanetBuffer *dest = janet_getbuffer(argv, 0); | ||||||
|  |     JanetByteView src = janet_getbytes(argv, 1); | ||||||
|  |     int32_t offset_dest = 0; | ||||||
|  |     int32_t offset_src = 0; | ||||||
|  |     if (argc > 2) | ||||||
|  |         offset_dest = janet_gethalfrange(argv, 2, dest->count, "dest-start"); | ||||||
|  |     if (argc > 3) | ||||||
|  |         offset_src = janet_gethalfrange(argv, 3, src.len, "src-start"); | ||||||
|  |     int32_t length_src; | ||||||
|  |     if (argc > 4) { | ||||||
|  |         int32_t src_end = janet_gethalfrange(argv, 4, src.len, "src-end"); | ||||||
|  |         length_src = src_end - offset_src; | ||||||
|  |         if (length_src < 0) length_src = 0; | ||||||
|  |     } else { | ||||||
|  |         length_src = src.len - offset_src; | ||||||
|  |     } | ||||||
|  |     int64_t last = ((int64_t) offset_dest - offset_src) + length_src; | ||||||
|  |     if (last > INT32_MAX) | ||||||
|  |         janet_panic("buffer blit out of range"); | ||||||
|  |     janet_buffer_ensure(dest, last, 2); | ||||||
|  |     if (last > dest->count) dest->count = last; | ||||||
|  |     memcpy(dest->data + offset_dest, src.bytes + offset_src, length_src); | ||||||
|  |     return argv[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
| static const JanetReg cfuns[] = { | static const JanetReg cfuns[] = { | ||||||
|     {"buffer/new", cfun_new, |     {"buffer/new", cfun_new, | ||||||
|         JDOC("(buffer/new capacity)\n\n" |         JDOC("(buffer/new capacity)\n\n" | ||||||
|                 "Creates a new, empty buffer with enough memory for capacity bytes. " |                 "Creates a new, empty buffer with enough memory for capacity bytes. " | ||||||
|                 "Returns a new buffer.") |                 "Returns a new buffer.") | ||||||
|     }, |     }, | ||||||
|  |     {"buffer/new-filled", cfun_new_filled, | ||||||
|  |         JDOC("(buffer/new-filled count [, byte=0])\n\n" | ||||||
|  |                 "Creates a new buffer of length count filled with byte. " | ||||||
|  |                 "Returns the new buffer.") | ||||||
|  |     }, | ||||||
|     {"buffer/push-byte", cfun_u8, |     {"buffer/push-byte", cfun_u8, | ||||||
|         JDOC("(buffer/push-byte buffer x)\n\n" |         JDOC("(buffer/push-byte buffer x)\n\n" | ||||||
|                 "Append a byte to a buffer. Will expand the buffer as necessary. " |                 "Append a byte to a buffer. Will expand the buffer as necessary. " | ||||||
| @@ -264,6 +359,28 @@ static const JanetReg cfuns[] = { | |||||||
|                 "end of the array. By default, start is 0 and end is the length of the buffer. " |                 "end of the array. By default, start is 0 and end is the length of the buffer. " | ||||||
|                 "Returns a new buffer.") |                 "Returns a new buffer.") | ||||||
|     }, |     }, | ||||||
|  |     {"buffer/bit-set", cfun_bitset, | ||||||
|  |         JDOC("(buffer/bit-set buffer index)\n\n" | ||||||
|  |                 "Sets the bit at the given bit-index. Returns the buffer.") | ||||||
|  |     }, | ||||||
|  |     {"buffer/bit-clear", cfun_bitclear, | ||||||
|  |         JDOC("(buffer/bit-clear buffer index)\n\n" | ||||||
|  |                 "Clears the bit at the given bit-index. Returns the buffer.") | ||||||
|  |     }, | ||||||
|  |     {"buffer/bit", cfun_bitget, | ||||||
|  |         JDOC("(buffer/bit buffer index)\n\n" | ||||||
|  |                 "Gets the bit at the given bit-index. Returns true if the bit is set, false if not.") | ||||||
|  |     }, | ||||||
|  |     {"buffer/bit-toggle", cfun_bittoggle, | ||||||
|  |         JDOC("(buffer/bit-toggle buffer index)\n\n" | ||||||
|  |                 "Toggles the bit at the given bit index in buffer. Returns the buffer.") | ||||||
|  |     }, | ||||||
|  |     {"buffer/blit", cfun_blit, | ||||||
|  |         JDOC("(buffer/blit dest src [, dest-start=0 [, src-start=0 [, src-end=-1]]])\n\n" | ||||||
|  |                 "Insert the contents of src into dest. Can optionally take indices that " | ||||||
|  |                 "indicate which part of src to copy into which part of dest. Indices can be " | ||||||
|  |                 "negative to index from the end of src or dest. Returns dest.") | ||||||
|  |     }, | ||||||
|     {NULL, NULL, NULL} |     {NULL, NULL, NULL} | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -110,6 +110,14 @@ int64_t janet_getinteger64(const Janet *argv, int32_t n) { | |||||||
|     return (int64_t) janet_unwrap_number(x); |     return (int64_t) janet_unwrap_number(x); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int32_t janet_gethalfrange(const Janet *argv, int32_t n, int32_t length, const char *which) { | ||||||
|  |     int32_t raw = janet_getinteger(argv, n); | ||||||
|  |     if (raw < 0) raw += length + 1; | ||||||
|  |     if (raw < 0 || raw > length) | ||||||
|  |         janet_panicf("%s index %d out of range [0,%d]", which, raw, length); | ||||||
|  |     return raw; | ||||||
|  | } | ||||||
|  |  | ||||||
| JanetView janet_getindexed(const Janet *argv, int32_t n) { | JanetView janet_getindexed(const Janet *argv, int32_t n) { | ||||||
|     Janet x = argv[n]; |     Janet x = argv[n]; | ||||||
|     JanetView view; |     JanetView view; | ||||||
| @@ -157,32 +165,13 @@ JanetRange janet_getslice(int32_t argc, const Janet *argv) { | |||||||
|         range.start = 0; |         range.start = 0; | ||||||
|         range.end = length; |         range.end = length; | ||||||
|     } else if (argc == 2) { |     } else if (argc == 2) { | ||||||
|         range.start = janet_getinteger(argv, 1); |         range.start = janet_gethalfrange(argv, 1, length, "start"); | ||||||
|         range.end = length; |         range.end = length; | ||||||
|         if (range.start < 0) { |  | ||||||
|             range.start += length + 1; |  | ||||||
|         } |  | ||||||
|         if (range.start < 0 || range.start > length) { |  | ||||||
|             janet_panicf("slice start: index %d out of range [0,%d]", range.start, length); |  | ||||||
|         } |  | ||||||
|     } else { |     } else { | ||||||
|         range.start = janet_getinteger(argv, 1); |         range.start = janet_gethalfrange(argv, 1, length, "start"); | ||||||
|         range.end = janet_getinteger(argv, 2); |         range.end = janet_gethalfrange(argv, 2, length, "end"); | ||||||
|         if (range.start < 0) { |         if (range.end < range.start) | ||||||
|             range.start += length + 1; |  | ||||||
|         } |  | ||||||
|         if (range.end < 0) { |  | ||||||
|             range.end += length + 1; |  | ||||||
|         } |  | ||||||
|         if (range.start < 0 || range.start > length) { |  | ||||||
|             janet_panicf("slice start: index %d out of range [0,%d]", range.start, length); |  | ||||||
|         } |  | ||||||
|         if (range.end < 0 || range.end > length) { |  | ||||||
|             janet_panicf("slice end: index %d out of range [0,%d]", range.end, length); |  | ||||||
|         } |  | ||||||
|         if (range.end < range.start) { |  | ||||||
|             range.end = range.start; |             range.end = range.start; | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|     return range; |     return range; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1162,6 +1162,7 @@ JANET_API JanetByteView janet_getbytes(const Janet *argv, int32_t n); | |||||||
| JANET_API JanetDictView janet_getdictionary(const Janet *argv, int32_t n); | JANET_API JanetDictView janet_getdictionary(const Janet *argv, int32_t n); | ||||||
| JANET_API void *janet_getabstract(const Janet *argv, int32_t n, const JanetAbstractType *at); | JANET_API void *janet_getabstract(const Janet *argv, int32_t n, const JanetAbstractType *at); | ||||||
| JANET_API JanetRange janet_getslice(int32_t argc, const Janet *argv); | JANET_API JanetRange janet_getslice(int32_t argc, const Janet *argv); | ||||||
|  | JANET_API int32_t janet_gethalfrange(const Janet *argv, int32_t n, int32_t length, const char *which); | ||||||
|  |  | ||||||
| /***** END SECTION MAIN *****/ | /***** END SECTION MAIN *****/ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -133,4 +133,24 @@ | |||||||
| (assert-error "bxor check types" (bxor 1 ())) | (assert-error "bxor check types" (bxor 1 ())) | ||||||
| (assert-error "bnot check types" (bnot ())) | (assert-error "bnot check types" (bnot ())) | ||||||
|  |  | ||||||
|  | # Buffer blitting | ||||||
|  |  | ||||||
|  | (def b (buffer/new-filled 100)) | ||||||
|  | (buffer/bit-set b 100) | ||||||
|  | (buffer/bit-clear b 100) | ||||||
|  | (assert (zero? (sum b)) "buffer bit set and clear") | ||||||
|  | (buffer/bit-toggle b 101) | ||||||
|  | (assert (= 32 (sum b)) "buffer bit set and clear") | ||||||
|  |  | ||||||
|  | (def b2 @"hello world") | ||||||
|  |  | ||||||
|  | (buffer/blit b2 "joyto ") | ||||||
|  | (assert (= (string b2) "joyto world") "buffer/blit 1") | ||||||
|  |  | ||||||
|  | (buffer/blit b2 "joyto" 6) | ||||||
|  | (assert (= (string b2) "joyto joyto") "buffer/blit 2") | ||||||
|  |  | ||||||
|  | (buffer/blit b2 "abcdefg" 5 6) | ||||||
|  | (assert (= (string b2) "joytogjoyto") "buffer/blit 3") | ||||||
|  |  | ||||||
| (end-suite) | (end-suite) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Calvin Rose
					Calvin Rose