src/buffer.c

Tue, 07 Feb 2023 20:08:45 +0100

author
Mike Becker <universe@uap-core.de>
date
Tue, 07 Feb 2023 20:08:45 +0100
changeset 650
77021e06b1a8
parent 567
f90a7cfe2480
child 673
60fb6aec157d
permissions
-rw-r--r--

fix code not compiling under windows+mingw

universe@483 1 /*
universe@483 2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
universe@483 3 *
universe@483 4 * Copyright 2021 Mike Becker, Olaf Wintermann All rights reserved.
universe@483 5 *
universe@483 6 * Redistribution and use in source and binary forms, with or without
universe@483 7 * modification, are permitted provided that the following conditions are met:
universe@483 8 *
universe@483 9 * 1. Redistributions of source code must retain the above copyright
universe@483 10 * notice, this list of conditions and the following disclaimer.
universe@483 11 *
universe@483 12 * 2. Redistributions in binary form must reproduce the above copyright
universe@483 13 * notice, this list of conditions and the following disclaimer in the
universe@483 14 * documentation and/or other materials provided with the distribution.
universe@483 15 *
universe@483 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
universe@483 17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
universe@483 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
universe@483 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
universe@483 20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
universe@483 21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
universe@483 22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
universe@483 23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
universe@483 24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
universe@483 25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
universe@483 26 * POSSIBILITY OF SUCH DAMAGE.
universe@483 27 */
universe@483 28
universe@483 29 #include "cx/buffer.h"
universe@483 30 #include "cx/utils.h"
universe@483 31
universe@530 32 #include <stdio.h>
universe@483 33 #include <string.h>
universe@483 34
universe@501 35 int cxBufferInit(
universe@501 36 CxBuffer *buffer,
universe@483 37 void *space,
universe@483 38 size_t capacity,
universe@529 39 CxAllocator const *allocator,
universe@483 40 int flags
universe@483 41 ) {
universe@501 42 buffer->allocator = allocator;
universe@501 43 buffer->flags = flags;
universe@501 44 if (!space) {
universe@501 45 buffer->bytes = cxMalloc(allocator, capacity);
universe@501 46 if (buffer->bytes == NULL) {
universe@501 47 return 1;
universe@483 48 }
universe@501 49 buffer->flags |= CX_BUFFER_FREE_CONTENTS;
universe@501 50 } else {
universe@501 51 buffer->bytes = space;
universe@501 52 }
universe@501 53 buffer->capacity = capacity;
universe@501 54 buffer->size = 0;
universe@539 55 buffer->pos = 0;
universe@483 56
universe@539 57 buffer->flush_func = NULL;
universe@541 58 buffer->flush_target = NULL;
universe@539 59 buffer->flush_blkmax = 0;
universe@539 60 buffer->flush_blksize = 4096;
universe@539 61 buffer->flush_threshold = SIZE_MAX;
universe@483 62
universe@501 63 return 0;
universe@483 64 }
universe@483 65
universe@500 66 void cxBufferDestroy(CxBuffer *buffer) {
universe@483 67 if ((buffer->flags & CX_BUFFER_FREE_CONTENTS) == CX_BUFFER_FREE_CONTENTS) {
universe@501 68 cxFree(buffer->allocator, buffer->bytes);
universe@483 69 }
universe@483 70 }
universe@483 71
universe@483 72 int cxBufferSeek(
universe@500 73 CxBuffer *buffer,
universe@483 74 off_t offset,
universe@483 75 int whence
universe@483 76 ) {
universe@483 77 size_t npos;
universe@483 78 switch (whence) {
universe@483 79 case SEEK_CUR:
universe@483 80 npos = buffer->pos;
universe@483 81 break;
universe@483 82 case SEEK_END:
universe@483 83 npos = buffer->size;
universe@483 84 break;
universe@483 85 case SEEK_SET:
universe@483 86 npos = 0;
universe@483 87 break;
universe@483 88 default:
universe@483 89 return -1;
universe@483 90 }
universe@483 91
universe@483 92 size_t opos = npos;
universe@483 93 npos += offset;
universe@483 94
universe@483 95 if ((offset > 0 && npos < opos) || (offset < 0 && npos > opos)) {
universe@483 96 return -1;
universe@483 97 }
universe@483 98
universe@483 99 if (npos >= buffer->size) {
universe@483 100 return -1;
universe@483 101 } else {
universe@483 102 buffer->pos = npos;
universe@483 103 return 0;
universe@483 104 }
universe@483 105
universe@483 106 }
universe@483 107
universe@529 108 void cxBufferClear(CxBuffer *buffer) {
universe@529 109 memset(buffer->bytes, 0, buffer->size);
universe@529 110 buffer->size = 0;
universe@529 111 buffer->pos = 0;
universe@529 112 }
universe@529 113
universe@529 114 int cxBufferEof(CxBuffer const *buffer) {
universe@483 115 return buffer->pos >= buffer->size;
universe@483 116 }
universe@483 117
universe@483 118 int cxBufferMinimumCapacity(
universe@500 119 CxBuffer *buffer,
universe@532 120 size_t newcap
universe@483 121 ) {
universe@532 122 if (newcap <= buffer->capacity) {
universe@532 123 return 0;
universe@483 124 }
universe@483 125
universe@536 126 if (cxReallocate(buffer->allocator,
universe@536 127 (void **) &buffer->bytes, newcap) == 0) {
universe@483 128 buffer->capacity = newcap;
universe@533 129 return 0;
universe@483 130 } else {
universe@483 131 return -1;
universe@483 132 }
universe@483 133 }
universe@483 134
universe@544 135 /**
universe@544 136 * Helps flushing data to the flush target of a buffer.
universe@544 137 *
universe@544 138 * @param buffer the buffer containing the config
universe@544 139 * @param space the data to flush
universe@544 140 * @param size the element size
universe@544 141 * @param nitems the number of items
universe@544 142 * @return the number of items flushed
universe@544 143 */
universe@544 144 static size_t cx_buffer_write_flush_helper(
universe@544 145 CxBuffer *buffer,
universe@544 146 unsigned char const *space,
universe@544 147 size_t size,
universe@544 148 size_t nitems
universe@544 149 ) {
universe@544 150 size_t pos = 0;
universe@544 151 size_t remaining = nitems;
universe@544 152 size_t max_items = buffer->flush_blksize / size;
universe@544 153 while (remaining > 0) {
universe@544 154 size_t items = remaining > max_items ? max_items : remaining;
universe@544 155 size_t flushed = buffer->flush_func(
universe@544 156 space + pos,
universe@544 157 size, items,
universe@544 158 buffer->flush_target);
universe@544 159 if (flushed > 0) {
universe@544 160 pos += (flushed * size);
universe@544 161 remaining -= flushed;
universe@544 162 } else {
universe@544 163 // if no bytes can be flushed out anymore, we give up
universe@544 164 break;
universe@544 165 }
universe@544 166 }
universe@544 167 return nitems - remaining;
universe@544 168 }
universe@544 169
universe@483 170 size_t cxBufferWrite(
universe@489 171 void const *ptr,
universe@483 172 size_t size,
universe@483 173 size_t nitems,
universe@500 174 CxBuffer *buffer
universe@483 175 ) {
universe@543 176 // optimize for easy case
universe@543 177 if (size == 1 && (buffer->capacity - buffer->pos) >= nitems) {
universe@543 178 memcpy(buffer->bytes + buffer->pos, ptr, nitems);
universe@543 179 buffer->pos += nitems;
universe@543 180 if (buffer->pos > buffer->size) {
universe@543 181 buffer->size = buffer->pos;
universe@543 182 }
universe@543 183 return nitems;
universe@543 184 }
universe@543 185
universe@483 186 size_t len;
universe@544 187 size_t nitems_out = nitems;
universe@483 188 if (cx_szmul(size, nitems, &len)) {
universe@483 189 return 0;
universe@483 190 }
universe@483 191 size_t required = buffer->pos + len;
universe@483 192 if (buffer->pos > required) {
universe@483 193 return 0;
universe@483 194 }
universe@483 195
universe@539 196 bool perform_flush = false;
universe@483 197 if (required > buffer->capacity) {
universe@539 198 if ((buffer->flags & CX_BUFFER_AUTO_EXTEND) == CX_BUFFER_AUTO_EXTEND && required) {
universe@539 199 if (buffer->flush_blkmax > 0 && required > buffer->flush_threshold) {
universe@539 200 perform_flush = true;
universe@539 201 } else {
universe@539 202 if (cxBufferMinimumCapacity(buffer, required)) {
universe@539 203 return 0;
universe@539 204 }
universe@483 205 }
universe@483 206 } else {
universe@539 207 if (buffer->flush_blkmax > 0) {
universe@539 208 perform_flush = true;
universe@539 209 } else {
universe@539 210 // truncate data to be written, if we can neither extend nor flush
universe@539 211 len = buffer->capacity - buffer->pos;
universe@539 212 if (size > 1) {
universe@539 213 len -= len % size;
universe@539 214 }
universe@544 215 nitems_out = len / size;
universe@483 216 }
universe@483 217 }
universe@483 218 }
universe@483 219
universe@483 220 if (len == 0) {
universe@483 221 return len;
universe@483 222 }
universe@483 223
universe@539 224 if (perform_flush) {
universe@544 225 size_t flush_max;
universe@544 226 if (cx_szmul(buffer->flush_blkmax, buffer->flush_blksize, &flush_max)) {
universe@544 227 return 0;
universe@544 228 }
universe@544 229 size_t flush_pos = buffer->flush_func == NULL || buffer->flush_target == NULL
universe@544 230 ? buffer->pos
universe@544 231 : cx_buffer_write_flush_helper(buffer, buffer->bytes, 1, buffer->pos);
universe@544 232 if (flush_pos == buffer->pos) {
universe@544 233 // entire buffer has been flushed, we can reset
universe@544 234 buffer->size = buffer->pos = 0;
universe@544 235
universe@544 236 size_t items_flush; // how many items can also be directly flushed
universe@544 237 size_t items_keep; // how many items have to be written to the buffer
universe@544 238
universe@544 239 items_flush = flush_max >= required ? nitems : (flush_max - flush_pos) / size;
universe@544 240 if (items_flush > 0) {
universe@544 241 items_flush = cx_buffer_write_flush_helper(buffer, ptr, size, items_flush / size);
universe@544 242 // in case we could not flush everything, keep the rest
universe@544 243 }
universe@544 244 items_keep = nitems - items_flush;
universe@544 245 if (items_keep > 0) {
universe@544 246 // try again with the remaining stuff
universe@544 247 unsigned char const *new_ptr = ptr;
universe@544 248 new_ptr += items_flush * size;
universe@567 249 // report the directly flushed items as written plus the remaining stuff
universe@567 250 return items_flush + cxBufferWrite(new_ptr, size, items_keep, buffer);
universe@544 251 } else {
universe@544 252 // all items have been flushed - report them as written
universe@544 253 return nitems;
universe@544 254 }
universe@544 255 } else if (flush_pos == 0) {
universe@544 256 // nothing could be flushed at all, we immediately give up without writing any data
universe@544 257 return 0;
universe@544 258 } else {
universe@567 259 // we were partially successful, we shift left and try again
universe@544 260 cxBufferShiftLeft(buffer, flush_pos);
universe@544 261 return cxBufferWrite(ptr, size, nitems, buffer);
universe@544 262 }
universe@539 263 } else {
universe@539 264 memcpy(buffer->bytes + buffer->pos, ptr, len);
universe@539 265 buffer->pos += len;
universe@539 266 if (buffer->pos > buffer->size) {
universe@539 267 buffer->size = buffer->pos;
universe@539 268 }
universe@544 269 return nitems_out;
universe@483 270 }
universe@483 271
universe@483 272 }
universe@483 273
universe@538 274 int cxBufferPut(
universe@538 275 CxBuffer *buffer,
universe@538 276 int c
universe@538 277 ) {
universe@538 278 c &= 0xFF;
universe@538 279 unsigned char const ch = c;
universe@538 280 if (cxBufferWrite(&ch, 1, 1, buffer) == 1) {
universe@538 281 return c;
universe@538 282 } else {
universe@538 283 return EOF;
universe@538 284 }
universe@538 285 }
universe@538 286
universe@538 287 size_t cxBufferPutString(
universe@538 288 CxBuffer *buffer,
universe@538 289 const char *str
universe@538 290 ) {
universe@538 291 return cxBufferWrite(str, 1, strlen(str), buffer);
universe@538 292 }
universe@538 293
universe@483 294 size_t cxBufferRead(
universe@483 295 void *ptr,
universe@483 296 size_t size,
universe@483 297 size_t nitems,
universe@500 298 CxBuffer *buffer
universe@483 299 ) {
universe@483 300 size_t len;
universe@483 301 if (cx_szmul(size, nitems, &len)) {
universe@483 302 return 0;
universe@483 303 }
universe@483 304 if (buffer->pos + len > buffer->size) {
universe@483 305 len = buffer->size - buffer->pos;
universe@483 306 if (size > 1) len -= len % size;
universe@483 307 }
universe@483 308
universe@483 309 if (len <= 0) {
universe@483 310 return len;
universe@483 311 }
universe@483 312
universe@483 313 memcpy(ptr, buffer->bytes + buffer->pos, len);
universe@483 314 buffer->pos += len;
universe@483 315
universe@483 316 return len / size;
universe@483 317 }
universe@483 318
universe@500 319 int cxBufferGet(CxBuffer *buffer) {
universe@483 320 if (cxBufferEof(buffer)) {
universe@483 321 return EOF;
universe@483 322 } else {
universe@483 323 int c = buffer->bytes[buffer->pos];
universe@483 324 buffer->pos++;
universe@483 325 return c;
universe@483 326 }
universe@483 327 }
universe@483 328
universe@483 329 int cxBufferShiftLeft(
universe@500 330 CxBuffer *buffer,
universe@483 331 size_t shift
universe@483 332 ) {
universe@483 333 if (shift >= buffer->size) {
universe@483 334 buffer->pos = buffer->size = 0;
universe@483 335 } else {
universe@483 336 memmove(buffer->bytes, buffer->bytes + shift, buffer->size - shift);
universe@483 337 buffer->size -= shift;
universe@483 338
universe@483 339 if (buffer->pos >= shift) {
universe@483 340 buffer->pos -= shift;
universe@483 341 } else {
universe@483 342 buffer->pos = 0;
universe@483 343 }
universe@483 344 }
universe@483 345 return 0;
universe@483 346 }
universe@483 347
universe@483 348 int cxBufferShiftRight(
universe@500 349 CxBuffer *buffer,
universe@483 350 size_t shift
universe@483 351 ) {
universe@483 352 size_t req_capacity = buffer->size + shift;
universe@483 353 size_t movebytes;
universe@483 354
universe@483 355 // auto extend buffer, if required and enabled
universe@483 356 if (buffer->capacity < req_capacity) {
universe@483 357 if ((buffer->flags & CX_BUFFER_AUTO_EXTEND) == CX_BUFFER_AUTO_EXTEND) {
universe@483 358 if (cxBufferMinimumCapacity(buffer, req_capacity)) {
universe@483 359 return 1;
universe@483 360 }
universe@483 361 movebytes = buffer->size;
universe@483 362 } else {
universe@483 363 movebytes = buffer->capacity - shift;
universe@483 364 }
universe@483 365 } else {
universe@483 366 movebytes = buffer->size;
universe@483 367 }
universe@483 368
universe@483 369 memmove(buffer->bytes + shift, buffer->bytes, movebytes);
universe@483 370 buffer->size = shift + movebytes;
universe@483 371
universe@483 372 buffer->pos += shift;
universe@483 373 if (buffer->pos > buffer->size) {
universe@483 374 buffer->pos = buffer->size;
universe@483 375 }
universe@483 376
universe@483 377 return 0;
universe@483 378 }
universe@483 379
universe@483 380 int cxBufferShift(
universe@500 381 CxBuffer *buffer,
universe@483 382 off_t shift
universe@483 383 ) {
universe@483 384 if (shift < 0) {
universe@483 385 return cxBufferShiftLeft(buffer, (size_t) (-shift));
universe@483 386 } else if (shift > 0) {
universe@483 387 return cxBufferShiftRight(buffer, (size_t) shift);
universe@483 388 } else {
universe@483 389 return 0;
universe@483 390 }
universe@483 391 }

mercurial