1 /*
2 * Written in 2013 by Dmitry Chestnykh <dmitry@codingrobots.com>
3 * Modified for CPython by Christian Heimes <christian@python.org>
4 *
5 * To the extent possible under law, the author have dedicated all
6 * copyright and related and neighboring rights to this software to
7 * the public domain worldwide. This software is distributed without
8 * any warranty. http://creativecommons.org/publicdomain/zero/1.0/
9 */
10
11 /* WARNING: autogenerated file!
12 *
13 * The blake2s_impl.c is autogenerated from blake2s_impl.c.
14 */
15
16 #include "Python.h"
17 #include "pystrhex.h"
18 #include "pythread.h"
19
20 #include "../hashlib.h"
21 #include "blake2ns.h"
22
23 #define HAVE_BLAKE2S 1
24 #define BLAKE2_LOCAL_INLINE(type) Py_LOCAL_INLINE(type)
25
26 #include "impl/blake2.h"
27 #include "impl/blake2-impl.h" /* for secure_zero_memory() and store48() */
28
29 /* pure SSE2 implementation is very slow, so only use the more optimized SSSE3+
30 * https://bugs.python.org/issue31834 */
31 #if defined(__SSSE3__) || defined(__SSE4_1__) || defined(__AVX__) || defined(__XOP__)
32 #include "impl/blake2s.c"
33 #else
34 #include "impl/blake2s-ref.c"
35 #endif
36
37
38 extern PyTypeObject PyBlake2_BLAKE2sType;
39
40 typedef struct {
41 PyObject_HEAD
42 blake2s_param param;
43 blake2s_state state;
44 PyThread_type_lock lock;
45 } BLAKE2sObject;
46
47 #include "clinic/blake2s_impl.c.h"
48
49 /*[clinic input]
50 module _blake2
51 class _blake2.blake2s "BLAKE2sObject *" "&PyBlake2_BLAKE2sType"
52 [clinic start generated code]*/
53 /*[clinic end generated code: output=da39a3ee5e6b4b0d input=4b79d7ffe07286ce]*/
54
55
56 static BLAKE2sObject *
new_BLAKE2sObject(PyTypeObject * type)57 new_BLAKE2sObject(PyTypeObject *type)
58 {
59 BLAKE2sObject *self;
60 self = (BLAKE2sObject *)type->tp_alloc(type, 0);
61 if (self != NULL) {
62 self->lock = NULL;
63 }
64 return self;
65 }
66
67 /*[clinic input]
68 @classmethod
69 _blake2.blake2s.__new__ as py_blake2s_new
70 data: object(c_default="NULL") = b''
71 /
72 *
73 digest_size: int(c_default="BLAKE2S_OUTBYTES") = _blake2.blake2s.MAX_DIGEST_SIZE
74 key: Py_buffer(c_default="NULL", py_default="b''") = None
75 salt: Py_buffer(c_default="NULL", py_default="b''") = None
76 person: Py_buffer(c_default="NULL", py_default="b''") = None
77 fanout: int = 1
78 depth: int = 1
79 leaf_size as leaf_size_obj: object(c_default="NULL") = 0
80 node_offset as node_offset_obj: object(c_default="NULL") = 0
81 node_depth: int = 0
82 inner_size: int = 0
83 last_node: bool = False
84
85 Return a new BLAKE2s hash object.
86 [clinic start generated code]*/
87
88 static PyObject *
py_blake2s_new_impl(PyTypeObject * type,PyObject * data,int digest_size,Py_buffer * key,Py_buffer * salt,Py_buffer * person,int fanout,int depth,PyObject * leaf_size_obj,PyObject * node_offset_obj,int node_depth,int inner_size,int last_node)89 py_blake2s_new_impl(PyTypeObject *type, PyObject *data, int digest_size,
90 Py_buffer *key, Py_buffer *salt, Py_buffer *person,
91 int fanout, int depth, PyObject *leaf_size_obj,
92 PyObject *node_offset_obj, int node_depth,
93 int inner_size, int last_node)
94 /*[clinic end generated code: output=fe060b258a8cbfc6 input=3abfaabe7f5f62cc]*/
95 {
96 BLAKE2sObject *self = NULL;
97 Py_buffer buf;
98
99 unsigned long leaf_size = 0;
100 unsigned long long node_offset = 0;
101
102 self = new_BLAKE2sObject(type);
103 if (self == NULL) {
104 goto error;
105 }
106
107 /* Zero parameter block. */
108 memset(&self->param, 0, sizeof(self->param));
109
110 /* Set digest size. */
111 if (digest_size <= 0 || digest_size > BLAKE2S_OUTBYTES) {
112 PyErr_Format(PyExc_ValueError,
113 "digest_size must be between 1 and %d bytes",
114 BLAKE2S_OUTBYTES);
115 goto error;
116 }
117 self->param.digest_length = digest_size;
118
119 /* Set salt parameter. */
120 if ((salt->obj != NULL) && salt->len) {
121 if (salt->len > BLAKE2S_SALTBYTES) {
122 PyErr_Format(PyExc_ValueError,
123 "maximum salt length is %d bytes",
124 BLAKE2S_SALTBYTES);
125 goto error;
126 }
127 memcpy(self->param.salt, salt->buf, salt->len);
128 }
129
130 /* Set personalization parameter. */
131 if ((person->obj != NULL) && person->len) {
132 if (person->len > BLAKE2S_PERSONALBYTES) {
133 PyErr_Format(PyExc_ValueError,
134 "maximum person length is %d bytes",
135 BLAKE2S_PERSONALBYTES);
136 goto error;
137 }
138 memcpy(self->param.personal, person->buf, person->len);
139 }
140
141 /* Set tree parameters. */
142 if (fanout < 0 || fanout > 255) {
143 PyErr_SetString(PyExc_ValueError,
144 "fanout must be between 0 and 255");
145 goto error;
146 }
147 self->param.fanout = (uint8_t)fanout;
148
149 if (depth <= 0 || depth > 255) {
150 PyErr_SetString(PyExc_ValueError,
151 "depth must be between 1 and 255");
152 goto error;
153 }
154 self->param.depth = (uint8_t)depth;
155
156 if (leaf_size_obj != NULL) {
157 leaf_size = PyLong_AsUnsignedLong(leaf_size_obj);
158 if (leaf_size == (unsigned long) -1 && PyErr_Occurred()) {
159 goto error;
160 }
161 if (leaf_size > 0xFFFFFFFFU) {
162 PyErr_SetString(PyExc_OverflowError, "leaf_size is too large");
163 goto error;
164 }
165 }
166 // NB: Simple assignment here would be incorrect on big endian platforms.
167 store32(&(self->param.leaf_length), leaf_size);
168
169 if (node_offset_obj != NULL) {
170 node_offset = PyLong_AsUnsignedLongLong(node_offset_obj);
171 if (node_offset == (unsigned long long) -1 && PyErr_Occurred()) {
172 goto error;
173 }
174 }
175 #ifdef HAVE_BLAKE2S
176 if (node_offset > 0xFFFFFFFFFFFFULL) {
177 /* maximum 2**48 - 1 */
178 PyErr_SetString(PyExc_OverflowError, "node_offset is too large");
179 goto error;
180 }
181 store48(&(self->param.node_offset), node_offset);
182 #else
183 // NB: Simple assignment here would be incorrect on big endian platforms.
184 store64(&(self->param.node_offset), node_offset);
185 #endif
186
187 if (node_depth < 0 || node_depth > 255) {
188 PyErr_SetString(PyExc_ValueError,
189 "node_depth must be between 0 and 255");
190 goto error;
191 }
192 self->param.node_depth = node_depth;
193
194 if (inner_size < 0 || inner_size > BLAKE2S_OUTBYTES) {
195 PyErr_Format(PyExc_ValueError,
196 "inner_size must be between 0 and is %d",
197 BLAKE2S_OUTBYTES);
198 goto error;
199 }
200 self->param.inner_length = inner_size;
201
202 /* Set key length. */
203 if ((key->obj != NULL) && key->len) {
204 if (key->len > BLAKE2S_KEYBYTES) {
205 PyErr_Format(PyExc_ValueError,
206 "maximum key length is %d bytes",
207 BLAKE2S_KEYBYTES);
208 goto error;
209 }
210 self->param.key_length = (uint8_t)key->len;
211 }
212
213 /* Initialize hash state. */
214 if (blake2s_init_param(&self->state, &self->param) < 0) {
215 PyErr_SetString(PyExc_RuntimeError,
216 "error initializing hash state");
217 goto error;
218 }
219
220 /* Set last node flag (must come after initialization). */
221 self->state.last_node = last_node;
222
223 /* Process key block if any. */
224 if (self->param.key_length) {
225 uint8_t block[BLAKE2S_BLOCKBYTES];
226 memset(block, 0, sizeof(block));
227 memcpy(block, key->buf, key->len);
228 blake2s_update(&self->state, block, sizeof(block));
229 secure_zero_memory(block, sizeof(block));
230 }
231
232 /* Process initial data if any. */
233 if (data != NULL) {
234 GET_BUFFER_VIEW_OR_ERROR(data, &buf, goto error);
235
236 if (buf.len >= HASHLIB_GIL_MINSIZE) {
237 Py_BEGIN_ALLOW_THREADS
238 blake2s_update(&self->state, buf.buf, buf.len);
239 Py_END_ALLOW_THREADS
240 } else {
241 blake2s_update(&self->state, buf.buf, buf.len);
242 }
243 PyBuffer_Release(&buf);
244 }
245
246 return (PyObject *)self;
247
248 error:
249 if (self != NULL) {
250 Py_DECREF(self);
251 }
252 return NULL;
253 }
254
255 /*[clinic input]
256 _blake2.blake2s.copy
257
258 Return a copy of the hash object.
259 [clinic start generated code]*/
260
261 static PyObject *
_blake2_blake2s_copy_impl(BLAKE2sObject * self)262 _blake2_blake2s_copy_impl(BLAKE2sObject *self)
263 /*[clinic end generated code: output=5b90131c4eae275e input=0b9d44942f0fe4b2]*/
264 {
265 BLAKE2sObject *cpy;
266
267 if ((cpy = new_BLAKE2sObject(Py_TYPE(self))) == NULL)
268 return NULL;
269
270 ENTER_HASHLIB(self);
271 cpy->param = self->param;
272 cpy->state = self->state;
273 LEAVE_HASHLIB(self);
274 return (PyObject *)cpy;
275 }
276
277 /*[clinic input]
278 _blake2.blake2s.update
279
280 data: object
281 /
282
283 Update this hash object's state with the provided bytes-like object.
284 [clinic start generated code]*/
285
286 static PyObject *
_blake2_blake2s_update(BLAKE2sObject * self,PyObject * data)287 _blake2_blake2s_update(BLAKE2sObject *self, PyObject *data)
288 /*[clinic end generated code: output=757dc087fec37815 input=97500db2f9de4aaa]*/
289 {
290 Py_buffer buf;
291
292 GET_BUFFER_VIEW_OR_ERROUT(data, &buf);
293
294 if (self->lock == NULL && buf.len >= HASHLIB_GIL_MINSIZE)
295 self->lock = PyThread_allocate_lock();
296
297 if (self->lock != NULL) {
298 Py_BEGIN_ALLOW_THREADS
299 PyThread_acquire_lock(self->lock, 1);
300 blake2s_update(&self->state, buf.buf, buf.len);
301 PyThread_release_lock(self->lock);
302 Py_END_ALLOW_THREADS
303 } else {
304 blake2s_update(&self->state, buf.buf, buf.len);
305 }
306 PyBuffer_Release(&buf);
307
308 Py_RETURN_NONE;
309 }
310
311 /*[clinic input]
312 _blake2.blake2s.digest
313
314 Return the digest value as a bytes object.
315 [clinic start generated code]*/
316
317 static PyObject *
_blake2_blake2s_digest_impl(BLAKE2sObject * self)318 _blake2_blake2s_digest_impl(BLAKE2sObject *self)
319 /*[clinic end generated code: output=40c566ca4bc6bc51 input=f41e0b8d6d937454]*/
320 {
321 uint8_t digest[BLAKE2S_OUTBYTES];
322 blake2s_state state_cpy;
323
324 ENTER_HASHLIB(self);
325 state_cpy = self->state;
326 blake2s_final(&state_cpy, digest, self->param.digest_length);
327 LEAVE_HASHLIB(self);
328 return PyBytes_FromStringAndSize((const char *)digest,
329 self->param.digest_length);
330 }
331
332 /*[clinic input]
333 _blake2.blake2s.hexdigest
334
335 Return the digest value as a string of hexadecimal digits.
336 [clinic start generated code]*/
337
338 static PyObject *
_blake2_blake2s_hexdigest_impl(BLAKE2sObject * self)339 _blake2_blake2s_hexdigest_impl(BLAKE2sObject *self)
340 /*[clinic end generated code: output=15153eb5e59c52eb input=c77a1321567e8952]*/
341 {
342 uint8_t digest[BLAKE2S_OUTBYTES];
343 blake2s_state state_cpy;
344
345 ENTER_HASHLIB(self);
346 state_cpy = self->state;
347 blake2s_final(&state_cpy, digest, self->param.digest_length);
348 LEAVE_HASHLIB(self);
349 return _Py_strhex((const char *)digest, self->param.digest_length);
350 }
351
352
353 static PyMethodDef py_blake2s_methods[] = {
354 _BLAKE2_BLAKE2S_COPY_METHODDEF
355 _BLAKE2_BLAKE2S_DIGEST_METHODDEF
356 _BLAKE2_BLAKE2S_HEXDIGEST_METHODDEF
357 _BLAKE2_BLAKE2S_UPDATE_METHODDEF
358 {NULL, NULL}
359 };
360
361
362
363 static PyObject *
py_blake2s_get_name(BLAKE2sObject * self,void * closure)364 py_blake2s_get_name(BLAKE2sObject *self, void *closure)
365 {
366 return PyUnicode_FromString("blake2s");
367 }
368
369
370
371 static PyObject *
py_blake2s_get_block_size(BLAKE2sObject * self,void * closure)372 py_blake2s_get_block_size(BLAKE2sObject *self, void *closure)
373 {
374 return PyLong_FromLong(BLAKE2S_BLOCKBYTES);
375 }
376
377
378
379 static PyObject *
py_blake2s_get_digest_size(BLAKE2sObject * self,void * closure)380 py_blake2s_get_digest_size(BLAKE2sObject *self, void *closure)
381 {
382 return PyLong_FromLong(self->param.digest_length);
383 }
384
385
386 static PyGetSetDef py_blake2s_getsetters[] = {
387 {"name", (getter)py_blake2s_get_name,
388 NULL, NULL, NULL},
389 {"block_size", (getter)py_blake2s_get_block_size,
390 NULL, NULL, NULL},
391 {"digest_size", (getter)py_blake2s_get_digest_size,
392 NULL, NULL, NULL},
393 {NULL}
394 };
395
396
397 static void
py_blake2s_dealloc(PyObject * self)398 py_blake2s_dealloc(PyObject *self)
399 {
400 BLAKE2sObject *obj = (BLAKE2sObject *)self;
401
402 /* Try not to leave state in memory. */
403 secure_zero_memory(&obj->param, sizeof(obj->param));
404 secure_zero_memory(&obj->state, sizeof(obj->state));
405 if (obj->lock) {
406 PyThread_free_lock(obj->lock);
407 obj->lock = NULL;
408 }
409 PyObject_Del(self);
410 }
411
412
413 PyTypeObject PyBlake2_BLAKE2sType = {
414 PyVarObject_HEAD_INIT(NULL, 0)
415 "_blake2.blake2s", /* tp_name */
416 sizeof(BLAKE2sObject), /* tp_basicsize */
417 0, /* tp_itemsize */
418 py_blake2s_dealloc, /* tp_dealloc */
419 0, /* tp_print */
420 0, /* tp_getattr */
421 0, /* tp_setattr */
422 0, /* tp_compare */
423 0, /* tp_repr */
424 0, /* tp_as_number */
425 0, /* tp_as_sequence */
426 0, /* tp_as_mapping */
427 0, /* tp_hash */
428 0, /* tp_call */
429 0, /* tp_str */
430 0, /* tp_getattro */
431 0, /* tp_setattro */
432 0, /* tp_as_buffer */
433 Py_TPFLAGS_DEFAULT, /* tp_flags */
434 py_blake2s_new__doc__, /* tp_doc */
435 0, /* tp_traverse */
436 0, /* tp_clear */
437 0, /* tp_richcompare */
438 0, /* tp_weaklistoffset */
439 0, /* tp_iter */
440 0, /* tp_iternext */
441 py_blake2s_methods, /* tp_methods */
442 0, /* tp_members */
443 py_blake2s_getsetters, /* tp_getset */
444 0, /* tp_base */
445 0, /* tp_dict */
446 0, /* tp_descr_get */
447 0, /* tp_descr_set */
448 0, /* tp_dictoffset */
449 0, /* tp_init */
450 0, /* tp_alloc */
451 py_blake2s_new, /* tp_new */
452 };
453