Path: blob/main/contrib/jemalloc/src/buf_writer.c
103823 views
#include "jemalloc/internal/jemalloc_preamble.h"1#include "jemalloc/internal/jemalloc_internal_includes.h"23#include "jemalloc/internal/buf_writer.h"4#include "jemalloc/internal/malloc_io.h"56static void *7buf_writer_allocate_internal_buf(tsdn_t *tsdn, size_t buf_len) {8#ifdef JEMALLOC_JET9if (buf_len > SC_LARGE_MAXCLASS) {10return NULL;11}12#else13assert(buf_len <= SC_LARGE_MAXCLASS);14#endif15return iallocztm(tsdn, buf_len, sz_size2index(buf_len), false, NULL,16true, arena_get(tsdn, 0, false), true);17}1819static void20buf_writer_free_internal_buf(tsdn_t *tsdn, void *buf) {21if (buf != NULL) {22idalloctm(tsdn, buf, NULL, NULL, true, true);23}24}2526static void27buf_writer_assert(buf_writer_t *buf_writer) {28assert(buf_writer != NULL);29assert(buf_writer->write_cb != NULL);30if (buf_writer->buf != NULL) {31assert(buf_writer->buf_size > 0);32} else {33assert(buf_writer->buf_size == 0);34assert(buf_writer->internal_buf);35}36assert(buf_writer->buf_end <= buf_writer->buf_size);37}3839bool40buf_writer_init(tsdn_t *tsdn, buf_writer_t *buf_writer, write_cb_t *write_cb,41void *cbopaque, char *buf, size_t buf_len) {42if (write_cb != NULL) {43buf_writer->write_cb = write_cb;44} else {45buf_writer->write_cb = je_malloc_message != NULL ?46je_malloc_message : wrtmessage;47}48buf_writer->cbopaque = cbopaque;49assert(buf_len >= 2);50if (buf != NULL) {51buf_writer->buf = buf;52buf_writer->internal_buf = false;53} else {54buf_writer->buf = buf_writer_allocate_internal_buf(tsdn,55buf_len);56buf_writer->internal_buf = true;57}58if (buf_writer->buf != NULL) {59buf_writer->buf_size = buf_len - 1; /* Allowing for '\0'. */60} else {61buf_writer->buf_size = 0;62}63buf_writer->buf_end = 0;64buf_writer_assert(buf_writer);65return buf_writer->buf == NULL;66}6768void69buf_writer_flush(buf_writer_t *buf_writer) {70buf_writer_assert(buf_writer);71if (buf_writer->buf == NULL) {72return;73}74buf_writer->buf[buf_writer->buf_end] = '\0';75buf_writer->write_cb(buf_writer->cbopaque, buf_writer->buf);76buf_writer->buf_end = 0;77buf_writer_assert(buf_writer);78}7980void81buf_writer_cb(void *buf_writer_arg, const char *s) {82buf_writer_t *buf_writer = (buf_writer_t *)buf_writer_arg;83buf_writer_assert(buf_writer);84if (buf_writer->buf == NULL) {85buf_writer->write_cb(buf_writer->cbopaque, s);86return;87}88size_t i, slen, n;89for (i = 0, slen = strlen(s); i < slen; i += n) {90if (buf_writer->buf_end == buf_writer->buf_size) {91buf_writer_flush(buf_writer);92}93size_t s_remain = slen - i;94size_t buf_remain = buf_writer->buf_size - buf_writer->buf_end;95n = s_remain < buf_remain ? s_remain : buf_remain;96memcpy(buf_writer->buf + buf_writer->buf_end, s + i, n);97buf_writer->buf_end += n;98buf_writer_assert(buf_writer);99}100assert(i == slen);101}102103void104buf_writer_terminate(tsdn_t *tsdn, buf_writer_t *buf_writer) {105buf_writer_assert(buf_writer);106buf_writer_flush(buf_writer);107if (buf_writer->internal_buf) {108buf_writer_free_internal_buf(tsdn, buf_writer->buf);109}110}111112void113buf_writer_pipe(buf_writer_t *buf_writer, read_cb_t *read_cb,114void *read_cbopaque) {115/*116* A tiny local buffer in case the buffered writer failed to allocate117* at init.118*/119static char backup_buf[16];120static buf_writer_t backup_buf_writer;121122buf_writer_assert(buf_writer);123assert(read_cb != NULL);124if (buf_writer->buf == NULL) {125buf_writer_init(TSDN_NULL, &backup_buf_writer,126buf_writer->write_cb, buf_writer->cbopaque, backup_buf,127sizeof(backup_buf));128buf_writer = &backup_buf_writer;129}130assert(buf_writer->buf != NULL);131ssize_t nread = 0;132do {133buf_writer->buf_end += nread;134buf_writer_assert(buf_writer);135if (buf_writer->buf_end == buf_writer->buf_size) {136buf_writer_flush(buf_writer);137}138nread = read_cb(read_cbopaque,139buf_writer->buf + buf_writer->buf_end,140buf_writer->buf_size - buf_writer->buf_end);141} while (nread > 0);142buf_writer_flush(buf_writer);143}144145146