/* mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution Copyright (C)2013-2023 The MMGen Project This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #define PY_SSIZE_T_CLEAN #include #include static PyObject * pubkey_gen(PyObject *self, PyObject *args) { const unsigned char * privkey; const int klen; const int compressed; if (!PyArg_ParseTuple(args, "y#I", &privkey, &klen, &compressed)) { PyErr_SetString(PyExc_ValueError, "Unable to parse extension mod arguments"); return NULL; } if (klen != 32) { PyErr_SetString(PyExc_ValueError, "Private key length not 32 bytes"); return NULL; } secp256k1_pubkey pubkey; size_t pubkeyclen = compressed == 1 ? 33 : 65; unsigned char pubkeyc[pubkeyclen]; static secp256k1_context *ctx = NULL; if (ctx == NULL) { /* puts ("Initializing context"); */ ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); } if (ctx == NULL) { PyErr_SetString(PyExc_RuntimeError, "Context initialization failed"); return NULL; } if (secp256k1_ec_pubkey_create(ctx, &pubkey, privkey) != 1) { PyErr_SetString(PyExc_RuntimeError, "Public key creation failed"); return NULL; } if (secp256k1_ec_pubkey_serialize(ctx, pubkeyc, &pubkeyclen, &pubkey, compressed == 1 ? SECP256K1_EC_COMPRESSED : SECP256K1_EC_UNCOMPRESSED) != 1) { PyErr_SetString(PyExc_RuntimeError, "Public key serialization failed"); return NULL; } return Py_BuildValue("y#", pubkeyc,pubkeyclen); } /* https://docs.python.org/3/howto/cporting.html */ struct module_state { PyObject *error; }; #define GETSTATE(m) ((struct module_state*)PyModule_GetState(m)) static PyMethodDef secp256k1_methods[] = { { "pubkey_gen", pubkey_gen, METH_VARARGS, "Generate a serialized pubkey from privkey bytes" }, {NULL, NULL} }; static int secp256k1_traverse(PyObject *m, visitproc visit, void *arg) { Py_VISIT(GETSTATE(m)->error); return 0; } static int secp256k1_clear(PyObject *m) { Py_CLEAR(GETSTATE(m)->error); return 0; } static struct PyModuleDef moduledef = { PyModuleDef_HEAD_INIT, "secp256k1", NULL, sizeof(struct module_state), secp256k1_methods, NULL, secp256k1_traverse, secp256k1_clear, NULL }; #define INITERROR return NULL PyMODINIT_FUNC PyInit_secp256k1(void) { PyObject *module = PyModule_Create(&moduledef); if (module == NULL) INITERROR; struct module_state *st = GETSTATE(module); st->error = PyErr_NewException("secp256k1.Error", NULL, NULL); if (st->error == NULL) { Py_DECREF(module); INITERROR; } return module; }