mirror of
https://github.com/python/cpython.git
synced 2025-10-24 07:26:11 +00:00

Most mutable data is protected by a striped lock that is keyed on the referenced object's address. The weakref's hash is protected using the weakref's per-object lock. Note that this only affects free-threaded builds. Apart from some minor refactoring, the added code is all either gated by `ifdef`s or is a no-op (e.g. `Py_BEGIN_CRITICAL_SECTION`).
615 lines
15 KiB
C
615 lines
15 KiB
C
#ifndef Py_BUILD_CORE_BUILTIN
|
|
# define Py_BUILD_CORE_MODULE 1
|
|
#endif
|
|
|
|
#include "blob.h"
|
|
#include "util.h"
|
|
|
|
#define clinic_state() (pysqlite_get_state_by_type(Py_TYPE(self)))
|
|
#include "clinic/blob.c.h"
|
|
#undef clinic_state
|
|
|
|
/*[clinic input]
|
|
module _sqlite3
|
|
class _sqlite3.Blob "pysqlite_Blob *" "clinic_state()->BlobType"
|
|
[clinic start generated code]*/
|
|
/*[clinic end generated code: output=da39a3ee5e6b4b0d input=908d3e16a45f8da7]*/
|
|
|
|
static void
|
|
close_blob(pysqlite_Blob *self)
|
|
{
|
|
if (self->blob) {
|
|
sqlite3_blob *blob = self->blob;
|
|
self->blob = NULL;
|
|
|
|
Py_BEGIN_ALLOW_THREADS
|
|
sqlite3_blob_close(blob);
|
|
Py_END_ALLOW_THREADS
|
|
}
|
|
}
|
|
|
|
static int
|
|
blob_traverse(pysqlite_Blob *self, visitproc visit, void *arg)
|
|
{
|
|
Py_VISIT(Py_TYPE(self));
|
|
Py_VISIT(self->connection);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
blob_clear(pysqlite_Blob *self)
|
|
{
|
|
Py_CLEAR(self->connection);
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
blob_dealloc(pysqlite_Blob *self)
|
|
{
|
|
PyTypeObject *tp = Py_TYPE(self);
|
|
PyObject_GC_UnTrack(self);
|
|
|
|
close_blob(self);
|
|
|
|
if (self->in_weakreflist != NULL) {
|
|
PyObject_ClearWeakRefs((PyObject*)self);
|
|
}
|
|
tp->tp_clear((PyObject *)self);
|
|
tp->tp_free(self);
|
|
Py_DECREF(tp);
|
|
}
|
|
|
|
// Return 1 if the blob object is usable, 0 if not.
|
|
static int
|
|
check_blob(pysqlite_Blob *self)
|
|
{
|
|
if (!pysqlite_check_connection(self->connection) ||
|
|
!pysqlite_check_thread(self->connection)) {
|
|
return 0;
|
|
}
|
|
if (self->blob == NULL) {
|
|
pysqlite_state *state = self->connection->state;
|
|
PyErr_SetString(state->ProgrammingError,
|
|
"Cannot operate on a closed blob.");
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.close as blob_close
|
|
|
|
Close the blob.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_close_impl(pysqlite_Blob *self)
|
|
/*[clinic end generated code: output=848accc20a138d1b input=7bc178a402a40bd8]*/
|
|
{
|
|
if (!pysqlite_check_connection(self->connection) ||
|
|
!pysqlite_check_thread(self->connection))
|
|
{
|
|
return NULL;
|
|
}
|
|
close_blob(self);
|
|
Py_RETURN_NONE;
|
|
};
|
|
|
|
void
|
|
pysqlite_close_all_blobs(pysqlite_Connection *self)
|
|
{
|
|
for (int i = 0; i < PyList_GET_SIZE(self->blobs); i++) {
|
|
PyObject *weakref = PyList_GET_ITEM(self->blobs, i);
|
|
PyObject *blob;
|
|
if (!PyWeakref_GetRef(weakref, &blob)) {
|
|
continue;
|
|
}
|
|
close_blob((pysqlite_Blob *)blob);
|
|
Py_DECREF(blob);
|
|
}
|
|
}
|
|
|
|
static void
|
|
blob_seterror(pysqlite_Blob *self, int rc)
|
|
{
|
|
assert(self->connection != NULL);
|
|
_pysqlite_seterror(self->connection->state, self->connection->db);
|
|
}
|
|
|
|
static PyObject *
|
|
read_single(pysqlite_Blob *self, Py_ssize_t offset)
|
|
{
|
|
unsigned char buf = 0;
|
|
int rc;
|
|
Py_BEGIN_ALLOW_THREADS
|
|
rc = sqlite3_blob_read(self->blob, (void *)&buf, 1, (int)offset);
|
|
Py_END_ALLOW_THREADS
|
|
|
|
if (rc != SQLITE_OK) {
|
|
blob_seterror(self, rc);
|
|
return NULL;
|
|
}
|
|
return PyLong_FromUnsignedLong((unsigned long)buf);
|
|
}
|
|
|
|
static PyObject *
|
|
read_multiple(pysqlite_Blob *self, Py_ssize_t length, Py_ssize_t offset)
|
|
{
|
|
assert(length <= sqlite3_blob_bytes(self->blob));
|
|
assert(offset < sqlite3_blob_bytes(self->blob));
|
|
|
|
PyObject *buffer = PyBytes_FromStringAndSize(NULL, length);
|
|
if (buffer == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
char *raw_buffer = PyBytes_AS_STRING(buffer);
|
|
int rc;
|
|
Py_BEGIN_ALLOW_THREADS
|
|
rc = sqlite3_blob_read(self->blob, raw_buffer, (int)length, (int)offset);
|
|
Py_END_ALLOW_THREADS
|
|
|
|
if (rc != SQLITE_OK) {
|
|
Py_DECREF(buffer);
|
|
blob_seterror(self, rc);
|
|
return NULL;
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.read as blob_read
|
|
|
|
length: int = -1
|
|
Read length in bytes.
|
|
/
|
|
|
|
Read data at the current offset position.
|
|
|
|
If the end of the blob is reached, the data up to end of file will be returned.
|
|
When length is not specified, or is negative, Blob.read() will read until the
|
|
end of the blob.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_read_impl(pysqlite_Blob *self, int length)
|
|
/*[clinic end generated code: output=1fc99b2541360dde input=f2e4aa4378837250]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
|
|
/* Make sure we never read past "EOB". Also read the rest of the blob if a
|
|
* negative length is specified. */
|
|
int blob_len = sqlite3_blob_bytes(self->blob);
|
|
int max_read_len = blob_len - self->offset;
|
|
if (length < 0 || length > max_read_len) {
|
|
length = max_read_len;
|
|
}
|
|
|
|
assert(length >= 0);
|
|
if (length == 0) {
|
|
return PyBytes_FromStringAndSize(NULL, 0);
|
|
}
|
|
|
|
PyObject *buffer = read_multiple(self, length, self->offset);
|
|
if (buffer == NULL) {
|
|
return NULL;
|
|
}
|
|
self->offset += length;
|
|
return buffer;
|
|
};
|
|
|
|
static int
|
|
inner_write(pysqlite_Blob *self, const void *buf, Py_ssize_t len,
|
|
Py_ssize_t offset)
|
|
{
|
|
Py_ssize_t blob_len = sqlite3_blob_bytes(self->blob);
|
|
Py_ssize_t remaining_len = blob_len - offset;
|
|
if (len > remaining_len) {
|
|
PyErr_SetString(PyExc_ValueError, "data longer than blob length");
|
|
return -1;
|
|
}
|
|
|
|
assert(offset <= blob_len);
|
|
int rc;
|
|
Py_BEGIN_ALLOW_THREADS
|
|
rc = sqlite3_blob_write(self->blob, buf, (int)len, (int)offset);
|
|
Py_END_ALLOW_THREADS
|
|
|
|
if (rc != SQLITE_OK) {
|
|
blob_seterror(self, rc);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.write as blob_write
|
|
|
|
data: Py_buffer
|
|
/
|
|
|
|
Write data at the current offset.
|
|
|
|
This function cannot change the blob length. Writing beyond the end of the
|
|
blob will result in an exception being raised.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_write_impl(pysqlite_Blob *self, Py_buffer *data)
|
|
/*[clinic end generated code: output=b34cf22601b570b2 input=a84712f24a028e6d]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
|
|
int rc = inner_write(self, data->buf, data->len, self->offset);
|
|
if (rc < 0) {
|
|
return NULL;
|
|
}
|
|
self->offset += (int)data->len;
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.seek as blob_seek
|
|
|
|
offset: int
|
|
origin: int = 0
|
|
/
|
|
|
|
Set the current access position to offset.
|
|
|
|
The origin argument defaults to os.SEEK_SET (absolute blob positioning).
|
|
Other values for origin are os.SEEK_CUR (seek relative to the current position)
|
|
and os.SEEK_END (seek relative to the blob's end).
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_seek_impl(pysqlite_Blob *self, int offset, int origin)
|
|
/*[clinic end generated code: output=854c5a0e208547a5 input=5da9a07e55fe6bb6]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
|
|
int blob_len = sqlite3_blob_bytes(self->blob);
|
|
switch (origin) {
|
|
case SEEK_SET:
|
|
break;
|
|
case SEEK_CUR:
|
|
if (offset > INT_MAX - self->offset) {
|
|
goto overflow;
|
|
}
|
|
offset += self->offset;
|
|
break;
|
|
case SEEK_END:
|
|
if (offset > INT_MAX - blob_len) {
|
|
goto overflow;
|
|
}
|
|
offset += blob_len;
|
|
break;
|
|
default:
|
|
PyErr_SetString(PyExc_ValueError,
|
|
"'origin' should be os.SEEK_SET, os.SEEK_CUR, or "
|
|
"os.SEEK_END");
|
|
return NULL;
|
|
}
|
|
|
|
if (offset < 0 || offset > blob_len) {
|
|
PyErr_SetString(PyExc_ValueError, "offset out of blob range");
|
|
return NULL;
|
|
}
|
|
|
|
self->offset = offset;
|
|
Py_RETURN_NONE;
|
|
|
|
overflow:
|
|
PyErr_SetString(PyExc_OverflowError, "seek offset results in overflow");
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.tell as blob_tell
|
|
|
|
Return the current access position for the blob.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_tell_impl(pysqlite_Blob *self)
|
|
/*[clinic end generated code: output=3d3ba484a90b3a99 input=7e34057aa303612c]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
return PyLong_FromLong(self->offset);
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.__enter__ as blob_enter
|
|
|
|
Blob context manager enter.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_enter_impl(pysqlite_Blob *self)
|
|
/*[clinic end generated code: output=4fd32484b071a6cd input=fe4842c3c582d5a7]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
return Py_NewRef(self);
|
|
}
|
|
|
|
|
|
/*[clinic input]
|
|
_sqlite3.Blob.__exit__ as blob_exit
|
|
|
|
type: object
|
|
val: object
|
|
tb: object
|
|
/
|
|
|
|
Blob context manager exit.
|
|
[clinic start generated code]*/
|
|
|
|
static PyObject *
|
|
blob_exit_impl(pysqlite_Blob *self, PyObject *type, PyObject *val,
|
|
PyObject *tb)
|
|
/*[clinic end generated code: output=fc86ceeb2b68c7b2 input=575d9ecea205f35f]*/
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
close_blob(self);
|
|
Py_RETURN_FALSE;
|
|
}
|
|
|
|
static Py_ssize_t
|
|
blob_length(pysqlite_Blob *self)
|
|
{
|
|
if (!check_blob(self)) {
|
|
return -1;
|
|
}
|
|
return sqlite3_blob_bytes(self->blob);
|
|
};
|
|
|
|
static Py_ssize_t
|
|
get_subscript_index(pysqlite_Blob *self, PyObject *item)
|
|
{
|
|
Py_ssize_t i = PyNumber_AsSsize_t(item, PyExc_IndexError);
|
|
if (i == -1 && PyErr_Occurred()) {
|
|
return -1;
|
|
}
|
|
int blob_len = sqlite3_blob_bytes(self->blob);
|
|
if (i < 0) {
|
|
i += blob_len;
|
|
}
|
|
if (i < 0 || i >= blob_len) {
|
|
PyErr_SetString(PyExc_IndexError, "Blob index out of range");
|
|
return -1;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
static PyObject *
|
|
subscript_index(pysqlite_Blob *self, PyObject *item)
|
|
{
|
|
Py_ssize_t i = get_subscript_index(self, item);
|
|
if (i < 0) {
|
|
return NULL;
|
|
}
|
|
return read_single(self, i);
|
|
}
|
|
|
|
static int
|
|
get_slice_info(pysqlite_Blob *self, PyObject *item, Py_ssize_t *start,
|
|
Py_ssize_t *stop, Py_ssize_t *step, Py_ssize_t *slicelen)
|
|
{
|
|
if (PySlice_Unpack(item, start, stop, step) < 0) {
|
|
return -1;
|
|
}
|
|
int len = sqlite3_blob_bytes(self->blob);
|
|
*slicelen = PySlice_AdjustIndices(len, start, stop, *step);
|
|
return 0;
|
|
}
|
|
|
|
static PyObject *
|
|
subscript_slice(pysqlite_Blob *self, PyObject *item)
|
|
{
|
|
Py_ssize_t start, stop, step, len;
|
|
if (get_slice_info(self, item, &start, &stop, &step, &len) < 0) {
|
|
return NULL;
|
|
}
|
|
|
|
if (step == 1) {
|
|
return read_multiple(self, len, start);
|
|
}
|
|
PyObject *blob = read_multiple(self, stop - start, start);
|
|
if (blob == NULL) {
|
|
return NULL;
|
|
}
|
|
PyObject *result = PyBytes_FromStringAndSize(NULL, len);
|
|
if (result != NULL) {
|
|
char *blob_buf = PyBytes_AS_STRING(blob);
|
|
char *res_buf = PyBytes_AS_STRING(result);
|
|
for (Py_ssize_t i = 0, j = 0; i < len; i++, j += step) {
|
|
res_buf[i] = blob_buf[j];
|
|
}
|
|
Py_DECREF(blob);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static PyObject *
|
|
blob_subscript(pysqlite_Blob *self, PyObject *item)
|
|
{
|
|
if (!check_blob(self)) {
|
|
return NULL;
|
|
}
|
|
|
|
if (PyIndex_Check(item)) {
|
|
return subscript_index(self, item);
|
|
}
|
|
if (PySlice_Check(item)) {
|
|
return subscript_slice(self, item);
|
|
}
|
|
|
|
PyErr_SetString(PyExc_TypeError, "Blob indices must be integers");
|
|
return NULL;
|
|
}
|
|
|
|
static int
|
|
ass_subscript_index(pysqlite_Blob *self, PyObject *item, PyObject *value)
|
|
{
|
|
if (value == NULL) {
|
|
PyErr_SetString(PyExc_TypeError,
|
|
"Blob doesn't support item deletion");
|
|
return -1;
|
|
}
|
|
if (!PyLong_Check(value)) {
|
|
PyErr_Format(PyExc_TypeError,
|
|
"'%s' object cannot be interpreted as an integer",
|
|
Py_TYPE(value)->tp_name);
|
|
return -1;
|
|
}
|
|
Py_ssize_t i = get_subscript_index(self, item);
|
|
if (i < 0) {
|
|
return -1;
|
|
}
|
|
|
|
long val = PyLong_AsLong(value);
|
|
if (val == -1 && PyErr_Occurred()) {
|
|
PyErr_Clear();
|
|
val = -1;
|
|
}
|
|
if (val < 0 || val > 255) {
|
|
PyErr_SetString(PyExc_ValueError, "byte must be in range(0, 256)");
|
|
return -1;
|
|
}
|
|
// Downcast to avoid endianness problems.
|
|
unsigned char byte = (unsigned char)val;
|
|
return inner_write(self, (const void *)&byte, 1, i);
|
|
}
|
|
|
|
static int
|
|
ass_subscript_slice(pysqlite_Blob *self, PyObject *item, PyObject *value)
|
|
{
|
|
if (value == NULL) {
|
|
PyErr_SetString(PyExc_TypeError,
|
|
"Blob doesn't support slice deletion");
|
|
return -1;
|
|
}
|
|
|
|
Py_ssize_t start, stop, step, len;
|
|
if (get_slice_info(self, item, &start, &stop, &step, &len) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
if (len == 0) {
|
|
return 0;
|
|
}
|
|
|
|
Py_buffer vbuf;
|
|
if (PyObject_GetBuffer(value, &vbuf, PyBUF_SIMPLE) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
int rc = -1;
|
|
if (vbuf.len != len) {
|
|
PyErr_SetString(PyExc_IndexError,
|
|
"Blob slice assignment is wrong size");
|
|
}
|
|
else if (step == 1) {
|
|
rc = inner_write(self, vbuf.buf, len, start);
|
|
}
|
|
else {
|
|
PyObject *blob_bytes = read_multiple(self, stop - start, start);
|
|
if (blob_bytes != NULL) {
|
|
char *blob_buf = PyBytes_AS_STRING(blob_bytes);
|
|
for (Py_ssize_t i = 0, j = 0; i < len; i++, j += step) {
|
|
blob_buf[j] = ((char *)vbuf.buf)[i];
|
|
}
|
|
rc = inner_write(self, blob_buf, stop - start, start);
|
|
Py_DECREF(blob_bytes);
|
|
}
|
|
}
|
|
PyBuffer_Release(&vbuf);
|
|
return rc;
|
|
}
|
|
|
|
static int
|
|
blob_ass_subscript(pysqlite_Blob *self, PyObject *item, PyObject *value)
|
|
{
|
|
if (!check_blob(self)) {
|
|
return -1;
|
|
}
|
|
|
|
if (PyIndex_Check(item)) {
|
|
return ass_subscript_index(self, item, value);
|
|
}
|
|
if (PySlice_Check(item)) {
|
|
return ass_subscript_slice(self, item, value);
|
|
}
|
|
|
|
PyErr_SetString(PyExc_TypeError, "Blob indices must be integers");
|
|
return -1;
|
|
}
|
|
|
|
|
|
static PyMethodDef blob_methods[] = {
|
|
BLOB_CLOSE_METHODDEF
|
|
BLOB_ENTER_METHODDEF
|
|
BLOB_EXIT_METHODDEF
|
|
BLOB_READ_METHODDEF
|
|
BLOB_SEEK_METHODDEF
|
|
BLOB_TELL_METHODDEF
|
|
BLOB_WRITE_METHODDEF
|
|
{NULL, NULL}
|
|
};
|
|
|
|
static struct PyMemberDef blob_members[] = {
|
|
{"__weaklistoffset__", Py_T_PYSSIZET, offsetof(pysqlite_Blob, in_weakreflist), Py_READONLY},
|
|
{NULL},
|
|
};
|
|
|
|
static PyType_Slot blob_slots[] = {
|
|
{Py_tp_dealloc, blob_dealloc},
|
|
{Py_tp_traverse, blob_traverse},
|
|
{Py_tp_clear, blob_clear},
|
|
{Py_tp_methods, blob_methods},
|
|
{Py_tp_members, blob_members},
|
|
|
|
// Mapping protocol
|
|
{Py_mp_length, blob_length},
|
|
{Py_mp_subscript, blob_subscript},
|
|
{Py_mp_ass_subscript, blob_ass_subscript},
|
|
{0, NULL},
|
|
};
|
|
|
|
static PyType_Spec blob_spec = {
|
|
.name = MODULE_NAME ".Blob",
|
|
.basicsize = sizeof(pysqlite_Blob),
|
|
.flags = (Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
|
|
Py_TPFLAGS_IMMUTABLETYPE | Py_TPFLAGS_DISALLOW_INSTANTIATION),
|
|
.slots = blob_slots,
|
|
};
|
|
|
|
int
|
|
pysqlite_blob_setup_types(PyObject *mod)
|
|
{
|
|
PyObject *type = PyType_FromModuleAndSpec(mod, &blob_spec, NULL);
|
|
if (type == NULL) {
|
|
return -1;
|
|
}
|
|
pysqlite_state *state = pysqlite_get_state(mod);
|
|
state->BlobType = (PyTypeObject *)type;
|
|
return 0;
|
|
}
|