1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556 |
- from cpython cimport PyBytes_AsString
- #from cpython cimport PyByteArray_AsString # cython still not exports that
- cdef extern from "Python.h":
- char* PyByteArray_AsString(bytearray ba) except NULL
- from libc.stdint cimport uint32_t, uint64_t, uintmax_t
- def _websocket_mask_cython(object mask, object data):
- """Note, this function mutates its `data` argument
- """
- cdef:
- Py_ssize_t data_len, i
- # bit operations on signed integers are implementation-specific
- unsigned char * in_buf
- const unsigned char * mask_buf
- uint32_t uint32_msk
- uint64_t uint64_msk
- assert len(mask) == 4
- if not isinstance(mask, bytes):
- mask = bytes(mask)
- if isinstance(data, bytearray):
- data = <bytearray>data
- else:
- data = bytearray(data)
- data_len = len(data)
- in_buf = <unsigned char*>PyByteArray_AsString(data)
- mask_buf = <const unsigned char*>PyBytes_AsString(mask)
- uint32_msk = (<uint32_t*>mask_buf)[0]
- # TODO: align in_data ptr to achieve even faster speeds
- # does it need in python ?! malloc() always aligns to sizeof(long) bytes
- if sizeof(size_t) >= 8:
- uint64_msk = uint32_msk
- uint64_msk = (uint64_msk << 32) | uint32_msk
- while data_len >= 8:
- (<uint64_t*>in_buf)[0] ^= uint64_msk
- in_buf += 8
- data_len -= 8
- while data_len >= 4:
- (<uint32_t*>in_buf)[0] ^= uint32_msk
- in_buf += 4
- data_len -= 4
- for i in range(0, data_len):
- in_buf[i] ^= mask_buf[i]
|