| /* bytes to hex implementation */ |
| |
| #include "Python.h" |
| |
| #include "pystrhex.h" |
| |
| static PyObject *_Py_strhex_impl(const char* argbuf, const Py_ssize_t arglen, |
| int return_bytes) |
| { |
| PyObject *retval; |
| Py_UCS1* retbuf; |
| Py_ssize_t i, j; |
| |
| assert(arglen >= 0); |
| if (arglen > PY_SSIZE_T_MAX / 2) |
| return PyErr_NoMemory(); |
| |
| if (return_bytes) { |
| /* If _PyBytes_FromSize() were public we could avoid malloc+copy. */ |
| retbuf = (Py_UCS1*) PyMem_Malloc(arglen*2); |
| if (!retbuf) |
| return PyErr_NoMemory(); |
| retval = NULL; /* silence a compiler warning, assigned later. */ |
| } else { |
| retval = PyUnicode_New(arglen*2, 127); |
| if (!retval) |
| return NULL; |
| retbuf = PyUnicode_1BYTE_DATA(retval); |
| } |
| |
| /* make hex version of string, taken from shamodule.c */ |
| for (i=j=0; i < arglen; i++) { |
| unsigned char c; |
| c = (argbuf[i] >> 4) & 0xf; |
| retbuf[j++] = Py_hexdigits[c]; |
| c = argbuf[i] & 0xf; |
| retbuf[j++] = Py_hexdigits[c]; |
| } |
| |
| if (return_bytes) { |
| retval = PyBytes_FromStringAndSize((const char *)retbuf, arglen*2); |
| PyMem_Free(retbuf); |
| } |
| #ifdef Py_DEBUG |
| else { |
| assert(_PyUnicode_CheckConsistency(retval, 1)); |
| } |
| #endif |
| |
| return retval; |
| } |
| |
| PyObject * _Py_strhex(const char* argbuf, const Py_ssize_t arglen) |
| { |
| return _Py_strhex_impl(argbuf, arglen, 0); |
| } |
| |
| /* Same as above but returns a bytes() instead of str() to avoid the |
| * need to decode the str() when bytes are needed. */ |
| PyObject * _Py_strhex_bytes(const char* argbuf, const Py_ssize_t arglen) |
| { |
| return _Py_strhex_impl(argbuf, arglen, 1); |
| } |