2021-04-22 20:55:39 -04:00
|
|
|
# test importing of .mpy files with native code (x64 only)
|
|
|
|
|
|
|
|
import sys, uio
|
|
|
|
|
|
|
|
try:
|
|
|
|
uio.IOBase
|
|
|
|
import uos
|
|
|
|
|
|
|
|
uos.mount
|
|
|
|
except (ImportError, AttributeError):
|
|
|
|
print("SKIP")
|
|
|
|
raise SystemExit
|
|
|
|
|
|
|
|
if not (sys.platform == "linux" and sys.maxsize > 2 ** 32):
|
|
|
|
print("SKIP")
|
|
|
|
raise SystemExit
|
|
|
|
|
|
|
|
|
|
|
|
class UserFile(uio.IOBase):
|
|
|
|
def __init__(self, data):
|
|
|
|
self.data = data
|
|
|
|
self.pos = 0
|
|
|
|
|
|
|
|
def read(self):
|
|
|
|
return self.data
|
|
|
|
|
|
|
|
def readinto(self, buf):
|
|
|
|
n = 0
|
|
|
|
while n < len(buf) and self.pos < len(self.data):
|
|
|
|
buf[n] = self.data[self.pos]
|
|
|
|
n += 1
|
|
|
|
self.pos += 1
|
|
|
|
return n
|
|
|
|
|
|
|
|
def ioctl(self, req, arg):
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
class UserFS:
|
|
|
|
def __init__(self, files):
|
|
|
|
self.files = files
|
|
|
|
|
|
|
|
def mount(self, readonly, mksfs):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def umount(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def stat(self, path):
|
|
|
|
if path in self.files:
|
|
|
|
return (32768, 0, 0, 0, 0, 0, 0, 0, 0, 0)
|
|
|
|
raise OSError
|
|
|
|
|
|
|
|
def open(self, path, mode):
|
|
|
|
return UserFile(self.files[path])
|
|
|
|
|
|
|
|
|
|
|
|
# these are the test .mpy files
|
|
|
|
user_files = {
|
|
|
|
# bad architecture
|
2021-04-23 15:26:42 -04:00
|
|
|
"/mod0.mpy": b"M\x05\xff\x00\x10",
|
2021-04-22 20:55:39 -04:00
|
|
|
# test loading of viper and asm
|
|
|
|
"/mod1.mpy": (
|
2021-04-23 15:26:42 -04:00
|
|
|
b"M\x05\x0b\x1f\x20" # header
|
|
|
|
b"\x20" # n bytes, bytecode
|
|
|
|
b"\x00\x08\x02m\x02m" # prelude
|
|
|
|
b"\x51" # LOAD_CONST_NONE
|
|
|
|
b"\x63" # RETURN_VALUE
|
|
|
|
b"\x00\x02" # n_obj, n_raw_code
|
2021-04-22 20:55:39 -04:00
|
|
|
b"\x22" # n bytes, viper code
|
|
|
|
b"\x00\x00\x00\x00\x00\x00" # dummy machine code
|
|
|
|
b"\x00\x00" # qstr0
|
|
|
|
b"\x01\x0c\x0aprint" # n_qstr, qstr0
|
|
|
|
b"\x00\x00\x00" # scope_flags, n_obj, n_raw_code
|
|
|
|
b"\x23" # n bytes, asm code
|
|
|
|
b"\x00\x00\x00\x00\x00\x00\x00\x00" # dummy machine code
|
|
|
|
b"\x00\x00\x00" # scope_flags, n_pos_args, type_sig
|
|
|
|
),
|
2021-04-23 15:26:42 -04:00
|
|
|
# test loading viper with truncated data
|
|
|
|
"/mod2.mpy": (
|
|
|
|
b"M\x05\x0b\x1f\x20" # header
|
|
|
|
b"\x20" # n bytes, bytecode
|
|
|
|
b"\x00\x08\x02m\x02m" # prelude
|
|
|
|
b"\x51" # LOAD_CONST_NONE
|
|
|
|
b"\x63" # RETURN_VALUE
|
|
|
|
b"\x00\x01" # n_obj, n_raw_code
|
|
|
|
b"\x12" # n bytes(=4), viper code
|
|
|
|
),
|
|
|
|
# test loading viper with additional scope flags and relocation
|
|
|
|
"/mod3.mpy": (
|
|
|
|
b"M\x05\x0b\x1f\x20" # header
|
|
|
|
b"\x20" # n bytes, bytecode
|
|
|
|
b"\x00\x08\x02m\x02m" # prelude
|
|
|
|
b"\x51" # LOAD_CONST_NONE
|
|
|
|
b"\x63" # RETURN_VALUE
|
|
|
|
b"\x00\x01" # n_obj, n_raw_code
|
|
|
|
b"\x12" # n bytes(=4), viper code
|
|
|
|
b"\x00\x00\x00\x00" # dummy machine code
|
|
|
|
b"\x00" # n_qstr
|
|
|
|
b"\x81\x60" # scope_flags: VIPERBSS | VIPERRODATA | VIPERRELOC (0xe0 encoded over two bytes)
|
|
|
|
b"\x00\x00" # n_obj, n_raw_code
|
|
|
|
b"\x06rodata" # rodata, 6 bytes
|
|
|
|
b"\x04" # bss, 4 bytes
|
|
|
|
b"\x03\x01\x00" # dummy relocation of rodata
|
|
|
|
),
|
2021-04-22 20:55:39 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
# create and mount a user filesystem
|
|
|
|
uos.mount(UserFS(user_files), "/userfs")
|
|
|
|
sys.path.append("/userfs")
|
|
|
|
|
|
|
|
# import .mpy files from the user filesystem
|
|
|
|
for i in range(len(user_files)):
|
|
|
|
mod = "mod%u" % i
|
|
|
|
try:
|
|
|
|
__import__(mod)
|
|
|
|
print(mod, "OK")
|
|
|
|
except ValueError as er:
|
|
|
|
print(mod, "ValueError", er)
|
2021-04-23 15:26:42 -04:00
|
|
|
except RuntimeError as er:
|
|
|
|
print(mod, "RuntimeError", er)
|
2021-04-22 20:55:39 -04:00
|
|
|
|
|
|
|
# unmount and undo path addition
|
|
|
|
uos.umount("/userfs")
|
|
|
|
sys.path.pop()
|