blob: 750284cc6fd433085148fa01c85d96478e3cac93 [file]
#ifndef STRINGWRITER_EXTRA_OPS_H
#define STRINGWRITER_EXTRA_OPS_H
#ifdef MYPYC_EXPERIMENTAL
#include <stdbool.h>
#include <stdint.h>
#include <Python.h>
#include "mypyc_util.h"
#include "strings/librt_strings.h"
static inline CPyTagged
CPyStringWriter_Len(PyObject *obj) {
return (CPyTagged)((StringWriterObject *)obj)->len << 1;
}
static inline bool
CPyStringWriter_EnsureSize(StringWriterObject *data, Py_ssize_t n) {
if (likely(data->capacity - data->len >= n)) {
return true;
} else {
return LibRTStrings_grow_string_buffer(data, n);
}
}
static inline char
CPyStringWriter_Append(PyObject *obj, int32_t value) {
StringWriterObject *self = (StringWriterObject *)obj;
char kind = self->kind;
// Fast path: kind 1 (ASCII/Latin-1) with character < 256
if (kind == 1 && (uint32_t)value < 256) {
// Store length in local variable to enable additional optimizations
Py_ssize_t len = self->len;
if (!CPyStringWriter_EnsureSize(self, 1))
return CPY_NONE_ERROR;
self->buf[len] = (char)value;
self->len = len + 1;
return CPY_NONE;
}
// Slow path: handles kind switching and other cases
return LibRTStrings_string_append_slow_path(self, value);
}
// If index is negative, convert to non-negative index (no range checking)
static inline int64_t CPyStringWriter_AdjustIndex(PyObject *obj, int64_t index) {
if (index < 0) {
return index + ((StringWriterObject *)obj)->len;
}
return index;
}
static inline bool CPyStringWriter_RangeCheck(PyObject *obj, int64_t index) {
return index >= 0 && index < ((StringWriterObject *)obj)->len;
}
static inline int32_t CPyStringWriter_GetItem(PyObject *obj, int64_t index) {
StringWriterObject *self = (StringWriterObject *)obj;
char kind = self->kind;
char *buf = self->buf;
if (kind == 1) {
return (uint8_t)buf[index];
} else if (kind == 2) {
uint16_t val;
memcpy(&val, buf + index * 2, 2);
return (int32_t)val;
} else {
uint32_t val;
memcpy(&val, buf + index * 4, 4);
return (int32_t)val;
}
}
#endif // MYPYC_EXPERIMENTAL
#endif