2015-05-04 16:35:40 +03:00
|
|
|
try:
|
|
|
|
import ustruct as struct
|
|
|
|
except:
|
2016-12-19 19:40:43 +03:00
|
|
|
try:
|
|
|
|
import struct
|
|
|
|
except ImportError:
|
|
|
|
print("SKIP")
|
2017-06-10 20:03:01 +03:00
|
|
|
raise SystemExit
|
2016-12-19 19:40:43 +03:00
|
|
|
|
2014-04-10 03:45:38 +03:00
|
|
|
print(struct.calcsize("<bI"))
|
|
|
|
print(struct.unpack("<bI", b"\x80\0\0\x01\0"))
|
2014-04-10 22:19:32 +03:00
|
|
|
print(struct.calcsize(">bI"))
|
|
|
|
print(struct.unpack(">bI", b"\x80\0\0\x01\0"))
|
2014-04-11 03:47:21 +03:00
|
|
|
|
|
|
|
# 32-bit little-endian specific
|
|
|
|
#print(struct.unpack("bI", b"\x80\xaa\x55\xaa\0\0\x01\0"))
|
2014-04-19 03:13:15 +03:00
|
|
|
|
2016-09-19 16:59:49 +03:00
|
|
|
print(struct.pack("<l", 1))
|
|
|
|
print(struct.pack(">l", 1))
|
2014-04-19 03:13:15 +03:00
|
|
|
print(struct.pack("<i", 1))
|
|
|
|
print(struct.pack(">i", 1))
|
|
|
|
print(struct.pack("<h", 1))
|
|
|
|
print(struct.pack(">h", 1))
|
|
|
|
print(struct.pack("<b", 1))
|
|
|
|
print(struct.pack(">b", 1))
|
|
|
|
|
|
|
|
print(struct.pack("<bI", -128, 256))
|
|
|
|
print(struct.pack(">bI", -128, 256))
|
2014-05-12 23:45:50 +03:00
|
|
|
|
|
|
|
print(struct.calcsize("100sI"))
|
|
|
|
print(struct.calcsize("97sI"))
|
|
|
|
print(struct.unpack("<6sH", b"foo\0\0\0\x12\x34"))
|
|
|
|
print(struct.pack("<6sH", b"foo", 10000))
|
2014-06-25 22:25:53 +03:00
|
|
|
|
|
|
|
s = struct.pack("BHBI", 10, 100, 200, 300)
|
|
|
|
v = struct.unpack("BHBI", s)
|
|
|
|
print(v == (10, 100, 200, 300))
|
2014-12-05 23:13:52 +00:00
|
|
|
|
2015-04-05 00:03:43 +01:00
|
|
|
# network byte order
|
|
|
|
print(struct.pack('!i', 123))
|
2015-09-03 23:06:18 +01:00
|
|
|
|
modstruct: Improve compliance with python3
While checking whether we can enable -Wimplicit-fallthrough, I encountered
a diagnostic in mp_binary_set_val_array_from_int which led to discovering
the following bug:
```
>>> struct.pack("xb", 3)
b'\x03\x03'
```
That is, the next value (3) was used as the value of a padding byte, while
standard Python always fills "x" bytes with zeros. I initially thought
this had to do with the unintentional fallthrough, but it doesn't.
Instead, this code would relate to an array.array with a typecode of
padding ('x'), which is ALSO not desktop Python compliant:
```
>>> array.array('x', (1, 2, 3))
array('x', [1, 0, 0])
```
Possibly this is dead code that used to be shared between struct-setting
and array-setting, but it no longer is.
I also discovered that the argument list length for struct.pack
and struct.pack_into were not checked, and that the length of binary data
passed to array.array was not checked to be a multiple of the element
size.
I have corrected all of these to conform more closely to standard Python
and revised some tests where necessary. Some tests for micropython-specific
behavior that does not conform to standard Python and is not present
in CircuitPython was deleted outright.
2020-09-12 13:57:31 -05:00
|
|
|
# too short / too long arguments
|
|
|
|
buf = bytearray(b'>>>123<<<')
|
|
|
|
try:
|
|
|
|
struct.pack_into('bb', buf, 0, 3)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
|
|
|
try:
|
|
|
|
struct.pack_into('bb', buf, 0, 3, 1, 4)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
|
|
|
try:
|
|
|
|
struct.pack('bb', 3)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
|
|
|
try:
|
|
|
|
struct.pack('bb', 3, 1, 4)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
2017-09-01 10:53:29 +10:00
|
|
|
# check that we get an error if the buffer is too small
|
|
|
|
try:
|
|
|
|
struct.unpack('I', b'\x00\x00\x00')
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
2015-09-03 23:06:18 +01:00
|
|
|
# first arg must be a string
|
|
|
|
try:
|
|
|
|
struct.pack(1, 2)
|
|
|
|
except TypeError:
|
|
|
|
print('TypeError')
|
2015-10-31 18:43:55 +03:00
|
|
|
|
2016-05-14 15:54:09 +03:00
|
|
|
# make sure that unknown types are detected
|
|
|
|
try:
|
|
|
|
struct.pack("z", 1)
|
|
|
|
except:
|
|
|
|
print("Unknown type")
|
|
|
|
|
2015-10-31 18:43:55 +03:00
|
|
|
# Initially repitition counters were supported only for strings,
|
|
|
|
# but later were implemented for all.
|
|
|
|
print(struct.unpack("<3B2h", b"foo\x12\x34\xff\xff"))
|
|
|
|
print(struct.pack("<3B", 1, 2, 3))
|
2015-12-23 19:11:27 -08:00
|
|
|
|
|
|
|
# pack_into
|
|
|
|
buf = bytearray(b'>>>123<<<')
|
|
|
|
struct.pack_into('<bbb', buf, 3, 0x41, 0x42, 0x43)
|
|
|
|
print(buf)
|
|
|
|
struct.pack_into('<bbb', buf, -6, 0x44, 0x45, 0x46)
|
|
|
|
print(buf)
|
|
|
|
|
2017-09-01 11:11:09 +10:00
|
|
|
# check that we get an error if the buffer is too small
|
|
|
|
try:
|
|
|
|
struct.pack_into('I', bytearray(1), 0, 0)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
2015-12-23 19:11:27 -08:00
|
|
|
try:
|
|
|
|
struct.pack_into('<bbb', buf, 7, 0x41, 0x42, 0x43)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
try:
|
|
|
|
struct.pack_into('<bbb', buf, -10, 0x41, 0x42, 0x43)
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
|
|
|
|
# unpack_from
|
|
|
|
buf = b'0123456789'
|
|
|
|
print(struct.unpack_from('<b', buf, 4))
|
|
|
|
print(struct.unpack_from('<b', buf, -4))
|
|
|
|
try:
|
|
|
|
print(struct.unpack_from('<b', buf, 10))
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
|
|
|
try:
|
|
|
|
print(struct.unpack_from('<b', buf, -11))
|
|
|
|
except:
|
|
|
|
print('struct.error')
|
modstruct: Improve compliance with python3
While checking whether we can enable -Wimplicit-fallthrough, I encountered
a diagnostic in mp_binary_set_val_array_from_int which led to discovering
the following bug:
```
>>> struct.pack("xb", 3)
b'\x03\x03'
```
That is, the next value (3) was used as the value of a padding byte, while
standard Python always fills "x" bytes with zeros. I initially thought
this had to do with the unintentional fallthrough, but it doesn't.
Instead, this code would relate to an array.array with a typecode of
padding ('x'), which is ALSO not desktop Python compliant:
```
>>> array.array('x', (1, 2, 3))
array('x', [1, 0, 0])
```
Possibly this is dead code that used to be shared between struct-setting
and array-setting, but it no longer is.
I also discovered that the argument list length for struct.pack
and struct.pack_into were not checked, and that the length of binary data
passed to array.array was not checked to be a multiple of the element
size.
I have corrected all of these to conform more closely to standard Python
and revised some tests where necessary. Some tests for micropython-specific
behavior that does not conform to standard Python and is not present
in CircuitPython was deleted outright.
2020-09-12 13:57:31 -05:00
|
|
|
|
|
|
|
# check padding bytes
|
|
|
|
print(struct.pack("xb", 3))
|
2021-04-16 12:39:23 -07:00
|
|
|
# Make sure pack doesn't reuse a larger value and error
|
|
|
|
print(struct.pack("xH", 0x100))
|