mirror of
https://github.com/status-im/c-kzg-4844.git
synced 2025-01-09 17:55:47 +00:00
69f6155d75
* Convert argument types to bytes * Update java bindings * Update python bindings * Update node.js bindings * Update c# bindings * Fix java binding compile issues * Fix incorrect memcpy in nodejs bindings * Fix bug (called the wrong func) * Fix issues with java bindings * Fix issues with node.js bindings * Remove unnecessary wrapped funcs for c# * Rename struct member to bytes * Use goto out for callocs * Fix nit * Make un-exported funcs static * Fix python bindings * Check commitment length in python bindings * Update python error message * Steal good ideas from #37 * Fix tests.py which didn't get copied over * Convert remaining a[] to *a * Add missing Py_DECREF * Bytes only rust (#1) * Make interface bytes only * Fix benches * Avoid newtypes for kzg types * Fix benches again * Make fields private * tidy * Address review comments * Fix one small thing in rust bindings * Use ckzg types where possible * Remove null terminator from domain bytes in rust * Update rust binding docs * Use BYTES_PER_* where applicable * Add extra check for calloc Co-authored-by: Pawan Dhananjay <pawandhananjay@gmail.com>
141 lines
4.8 KiB
C
141 lines
4.8 KiB
C
#define PY_SSIZE_T_CLEAN
|
|
#include <Python.h>
|
|
#include "c_kzg_4844.h"
|
|
|
|
static void free_KZGSettings(PyObject *c) {
|
|
KZGSettings *s = PyCapsule_GetPointer(c, "KZGSettings");
|
|
free_trusted_setup(s);
|
|
free(s);
|
|
}
|
|
|
|
static PyObject* load_trusted_setup_wrap(PyObject *self, PyObject *args) {
|
|
PyObject *f;
|
|
|
|
if (!PyArg_ParseTuple(args, "U", &f))
|
|
return PyErr_Format(PyExc_ValueError, "expected a string");
|
|
|
|
KZGSettings *s = (KZGSettings*)malloc(sizeof(KZGSettings));
|
|
|
|
if (s == NULL) return PyErr_NoMemory();
|
|
|
|
if (load_trusted_setup_file(s, fopen(PyUnicode_AsUTF8(f), "r")) != C_KZG_OK) {
|
|
free(s);
|
|
return PyErr_Format(PyExc_RuntimeError, "error loading trusted setup");
|
|
}
|
|
|
|
return PyCapsule_New(s, "KZGSettings", free_KZGSettings);
|
|
}
|
|
|
|
static PyObject* blob_to_kzg_commitment_wrap(PyObject *self, PyObject *args) {
|
|
PyObject *b;
|
|
PyObject *s;
|
|
|
|
if (!PyArg_UnpackTuple(args, "blob_to_kzg_commitment_wrap", 2, 2, &b, &s) ||
|
|
!PyBytes_Check(b) ||
|
|
!PyCapsule_IsValid(s, "KZGSettings"))
|
|
return PyErr_Format(PyExc_ValueError, "expected bytes and trusted setup");
|
|
|
|
if (PyBytes_Size(b) != BYTES_PER_BLOB)
|
|
return PyErr_Format(PyExc_ValueError, "expected blobs to be BYTES_PER_BLOB bytes");
|
|
|
|
PyObject *out = PyBytes_FromStringAndSize(NULL, BYTES_PER_COMMITMENT);
|
|
if (out == NULL) return PyErr_NoMemory();
|
|
|
|
Blob *blob = (Blob *)PyBytes_AsString(b);
|
|
KZGCommitment *k = (KZGCommitment *)PyBytes_AsString(out);
|
|
if (blob_to_kzg_commitment(k, blob, PyCapsule_GetPointer(s, "KZGSettings")) != C_KZG_OK) {
|
|
Py_DECREF(out);
|
|
return PyErr_Format(PyExc_RuntimeError, "blob_to_kzg_commitment failed");
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
static PyObject* compute_aggregate_kzg_proof_wrap(PyObject *self, PyObject *args) {
|
|
PyObject *b, *s;
|
|
|
|
if (!PyArg_UnpackTuple(args, "compute_aggregate_kzg_proof", 2, 2, &b, &s) ||
|
|
!PyBytes_Check(b) ||
|
|
!PyCapsule_IsValid(s, "KZGSettings"))
|
|
return PyErr_Format(PyExc_ValueError, "expected bytes, trusted setup");
|
|
|
|
Py_ssize_t n = PyBytes_Size(b);
|
|
if (n % BYTES_PER_BLOB != 0)
|
|
return PyErr_Format(PyExc_ValueError, "expected blobs to be a multiple of BYTES_PER_BLOB bytes");
|
|
n = n / BYTES_PER_BLOB;
|
|
|
|
PyObject *out = PyBytes_FromStringAndSize(NULL, BYTES_PER_PROOF);
|
|
if (out == NULL) return PyErr_NoMemory();
|
|
|
|
Blob *blobs = (Blob *)PyBytes_AsString(b);
|
|
KZGProof *k = (KZGProof *)PyBytes_AsString(out);
|
|
if (compute_aggregate_kzg_proof(k, blobs, n,
|
|
PyCapsule_GetPointer(s, "KZGSettings")) != C_KZG_OK) {
|
|
Py_DECREF(out);
|
|
return PyErr_Format(PyExc_RuntimeError, "compute_aggregate_kzg_proof failed");
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
static PyObject* verify_aggregate_kzg_proof_wrap(PyObject *self, PyObject *args) {
|
|
PyObject *b, *c, *p, *s;
|
|
|
|
if (!PyArg_UnpackTuple(args, "verify_aggregate_kzg_proof", 4, 4, &b, &c, &p, &s) ||
|
|
!PyBytes_Check(b) ||
|
|
!PyBytes_Check(c) ||
|
|
!PyBytes_Check(p) ||
|
|
!PyCapsule_IsValid(s, "KZGSettings"))
|
|
return PyErr_Format(PyExc_ValueError,
|
|
"expected bytes, bytes, bytes, trusted setup");
|
|
|
|
if (PyBytes_Size(p) != BYTES_PER_PROOF)
|
|
return PyErr_Format(PyExc_ValueError, "expected proof to be BYTES_PER_PROOF bytes");
|
|
|
|
Py_ssize_t n = PyBytes_Size(b);
|
|
if (n % BYTES_PER_BLOB != 0)
|
|
return PyErr_Format(PyExc_ValueError, "expected blobs to be a multiple of BYTES_PER_BLOB bytes");
|
|
n = n / BYTES_PER_BLOB;
|
|
|
|
Py_ssize_t m = PyBytes_Size(c);
|
|
if (m % BYTES_PER_COMMITMENT != 0)
|
|
return PyErr_Format(PyExc_ValueError, "expected commitments to be a multiple of BYTES_PER_COMMITMENT bytes");
|
|
m = m / BYTES_PER_COMMITMENT;
|
|
|
|
if (m != n)
|
|
return PyErr_Format(PyExc_ValueError, "expected same number of commitments as polynomials");
|
|
|
|
const Blob* blobs = (Blob *)PyBytes_AsString(b);
|
|
const KZGProof *proof = (KZGProof *)PyBytes_AsString(p);
|
|
const KZGCommitment *commitments = (KZGCommitment *)PyBytes_AsString(c);
|
|
|
|
bool out;
|
|
if (verify_aggregate_kzg_proof(&out,
|
|
blobs, commitments, n, proof,
|
|
PyCapsule_GetPointer(s, "KZGSettings")) != C_KZG_OK) {
|
|
return PyErr_Format(PyExc_RuntimeError, "verify_aggregate_kzg_proof failed");
|
|
}
|
|
|
|
if (out) Py_RETURN_TRUE; else Py_RETURN_FALSE;
|
|
}
|
|
|
|
static PyMethodDef ckzgmethods[] = {
|
|
{"load_trusted_setup", load_trusted_setup_wrap, METH_VARARGS, "Load trusted setup from file path"},
|
|
{"blob_to_kzg_commitment", blob_to_kzg_commitment_wrap, METH_VARARGS, "Create a commitment from a blob"},
|
|
{"compute_aggregate_kzg_proof", compute_aggregate_kzg_proof_wrap, METH_VARARGS, "Compute aggregate KZG proof"},
|
|
{"verify_aggregate_kzg_proof", verify_aggregate_kzg_proof_wrap, METH_VARARGS, "Verify aggregate KZG proof"},
|
|
{NULL, NULL, 0, NULL}
|
|
};
|
|
|
|
static struct PyModuleDef ckzg = {
|
|
PyModuleDef_HEAD_INIT,
|
|
"ckzg",
|
|
NULL,
|
|
-1,
|
|
ckzgmethods
|
|
};
|
|
|
|
PyMODINIT_FUNC PyInit_ckzg(void) {
|
|
return PyModule_Create(&ckzg);
|
|
}
|