mirror of
https://github.com/AsahiLinux/m1n1
synced 2024-11-28 09:20:17 +00:00
proxyclient: initial files
Signed-off-by: Hector Martin <marcan@marcan.st>
This commit is contained in:
parent
5175c7a31b
commit
bd19584743
5 changed files with 578 additions and 0 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -1,2 +1,5 @@
|
|||
!build/.keep
|
||||
build/
|
||||
__pycache__
|
||||
*.pyc
|
||||
|
||||
|
|
35
proxyclient/chainload.py
Normal file
35
proxyclient/chainload.py
Normal file
|
@ -0,0 +1,35 @@
|
|||
#!/usr/bin/python
|
||||
|
||||
import serial, os, struct, sys, time
|
||||
from proxy import *
|
||||
from tgtypes import *
|
||||
|
||||
uartdev = os.environ.get("M1N1DEVICE", "/dev/ttyUSB0")
|
||||
usbuart = serial.Serial(uartdev, 115200)
|
||||
|
||||
iface = UartInterface(usbuart, debug=False)
|
||||
proxy = M1N1Proxy(iface, debug=False)
|
||||
|
||||
proxy.set_baud(1500000)
|
||||
|
||||
payload = open(sys.argv[1], "rb").read()
|
||||
|
||||
base = proxy.get_base()
|
||||
ba_addr = proxy.get_bootargs()
|
||||
|
||||
ba = iface.readstruct(ba_addr, BootArgs)
|
||||
|
||||
new_base = base + ((ba.top_of_kernel_data + 0xffff) & ~0xffff) - ba.phys_base
|
||||
|
||||
print("Loading %d bytes to 0x%x" % (len(payload), new_base))
|
||||
|
||||
iface.writemem(new_base + 0x4000, payload[0x4000:], True)
|
||||
|
||||
entry = new_base + 0x4000
|
||||
|
||||
print("Jumping to 0x%x" % entry)
|
||||
|
||||
proxy.vector(entry, ba_addr)
|
||||
|
||||
iface.nop()
|
||||
print("Proxy is alive again")
|
442
proxyclient/proxy.py
Executable file
442
proxyclient/proxy.py
Executable file
|
@ -0,0 +1,442 @@
|
|||
#!/usr/bin/python3
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
import os, sys, struct
|
||||
|
||||
def hexdump(s,sep=" "):
|
||||
return sep.join(["%02x"%x for x in s])
|
||||
|
||||
def ascii(s):
|
||||
s2 = ""
|
||||
for c in s:
|
||||
if c < 0x20 or c > 0x7e:
|
||||
s2 += "."
|
||||
else:
|
||||
s2 += c
|
||||
return s2
|
||||
|
||||
def pad(s,c,l):
|
||||
if len(s) < l:
|
||||
s += c * (l - len(s))
|
||||
return s
|
||||
|
||||
def chexdump(s,st=0):
|
||||
for i in range(0,len(s),16):
|
||||
print("%08x %s %s |%s|" % (
|
||||
i + st,
|
||||
hexdump(s[i:i+8], ' ').rjust(23),
|
||||
hexdump(s[i+8:i+16], ' ').rjust(23),
|
||||
ascii(s[i:i+16]),rjust(16)))
|
||||
|
||||
|
||||
class UartError(RuntimeError):
|
||||
pass
|
||||
|
||||
class UartTimeout(UartError):
|
||||
pass
|
||||
|
||||
class UartCMDError(UartError):
|
||||
pass
|
||||
|
||||
class UartChecksumError(UartError):
|
||||
pass
|
||||
|
||||
class UartRemoteError(UartError):
|
||||
pass
|
||||
|
||||
class UartInterface:
|
||||
REQ_NOP = 0x00AA55FF
|
||||
REQ_PROXY = 0x01AA55FF
|
||||
REQ_MEMREAD = 0x02AA55FF
|
||||
REQ_MEMWRITE = 0x03AA55FF
|
||||
REQ_BOOT = 0x04AA55FF
|
||||
|
||||
ST_OK = 0
|
||||
ST_BADCMD = -1
|
||||
ST_INVAL = -2
|
||||
ST_XFERERR = -3
|
||||
ST_CRCERR = -4
|
||||
|
||||
CMD_LEN = 56
|
||||
REPLY_LEN = 36
|
||||
|
||||
def __init__(self, device, debug=False):
|
||||
self.debug = debug
|
||||
self.dev = device
|
||||
self.dev.timeout = 0
|
||||
self.dev.flushOutput()
|
||||
self.dev.flushInput()
|
||||
self.pted = False
|
||||
#d = self.dev.read(1)
|
||||
#while d != "":
|
||||
#d = self.dev.read(1)
|
||||
self.dev.timeout = 1.5
|
||||
self.tty_enable = True
|
||||
|
||||
def checksum(self, data):
|
||||
sum = 0xDEADBEEF;
|
||||
for c in data:
|
||||
sum *= 31337
|
||||
sum += c ^ 0x5a
|
||||
sum &= 0xFFFFFFFF
|
||||
|
||||
return (sum ^ 0xADDEDBAD) & 0xFFFFFFFF
|
||||
|
||||
def readfull(self, size):
|
||||
d = b''
|
||||
while len(d) < size:
|
||||
block = self.dev.read(size - len(d))
|
||||
if not block:
|
||||
raise UartTimeout("Expected %d bytes, got %d bytes"%(size,len(d)))
|
||||
d += block
|
||||
return d
|
||||
|
||||
def cmd(self, cmd, payload=b""):
|
||||
if len(payload) > self.CMD_LEN:
|
||||
raise ValueError("Incorrect payload size %d"%len(payload))
|
||||
|
||||
payload = payload.ljust(self.CMD_LEN, b"\x00")
|
||||
command = struct.pack("<I", cmd) + payload
|
||||
command += struct.pack("<I", self.checksum(command))
|
||||
if self.debug:
|
||||
print("<<", hexdump(command))
|
||||
self.dev.write(command)
|
||||
|
||||
def unkhandler(self, s):
|
||||
if not self.tty_enable:
|
||||
return
|
||||
for c in s:
|
||||
if not self.pted:
|
||||
sys.stdout.write("TTY> ")
|
||||
self.pted = True
|
||||
if c == 10:
|
||||
self.pted = False
|
||||
sys.stdout.write(chr(c))
|
||||
sys.stdout.flush()
|
||||
|
||||
def reply(self, cmd):
|
||||
reply = b''
|
||||
while True:
|
||||
if not reply or reply[-1] != 255:
|
||||
reply = b''
|
||||
reply += self.readfull(1)
|
||||
if reply != b"\xff":
|
||||
self.unkhandler(reply)
|
||||
continue
|
||||
else:
|
||||
reply = b'\xff'
|
||||
reply += self.readfull(1)
|
||||
if reply != b"\xff\x55":
|
||||
self.unkhandler(reply)
|
||||
continue
|
||||
reply += self.readfull(1)
|
||||
if reply != b"\xff\x55\xaa":
|
||||
self.unkhandler(reply)
|
||||
continue
|
||||
reply += self.readfull(self.REPLY_LEN - 3)
|
||||
if self.debug:
|
||||
print(">>", hexdump(reply))
|
||||
cmdin, status, data, checksum = struct.unpack("<Ii24sI", reply)
|
||||
ccsum = self.checksum(reply[:-4])
|
||||
if checksum != ccsum:
|
||||
print("Reply checksum error: Expected 0x%08x, got 0x%08x"%(checksum, ccsum))
|
||||
raise UartChecksumError()
|
||||
|
||||
if cmdin != cmd:
|
||||
if cmdin == self.REQ_BOOT:
|
||||
# Proxy rebooted in the meantime, try again
|
||||
return self.reply(cmd)
|
||||
raise UartCMDError("Reply command mismatch: Expected 0x%08x, got 0x%08x"%(cmd, cmdin))
|
||||
if status != self.ST_OK:
|
||||
if status == self.ST_BADCMD:
|
||||
raise UartRemoteError("Reply error: Bad Command")
|
||||
elif status == self.ST_INVAL:
|
||||
raise UartRemoteError("Reply error: Invalid argument")
|
||||
elif status == self.ST_XFERERR:
|
||||
raise UartRemoteError("Reply error: Data transfer failed")
|
||||
elif status == self.ST_CRCERR:
|
||||
raise UartRemoteError("Reply error: Data checksum failed")
|
||||
else:
|
||||
raise UartRemoteError("Reply error: Unknown error (%d)"%status)
|
||||
return data
|
||||
|
||||
def nop(self):
|
||||
self.cmd(self.REQ_NOP)
|
||||
self.reply(self.REQ_NOP)
|
||||
|
||||
def proxyreq(self, req, reboot=False):
|
||||
self.cmd(self.REQ_PROXY, req)
|
||||
if reboot:
|
||||
return self.reply(self.REQ_BOOT)
|
||||
else:
|
||||
return self.reply(self.REQ_PROXY)
|
||||
|
||||
def writemem(self, addr, data, progress=False):
|
||||
checksum = self.checksum(data)
|
||||
size = len(data)
|
||||
req = struct.pack("<QQI", addr, size, checksum)
|
||||
self.cmd(self.REQ_MEMWRITE, req)
|
||||
if self.debug:
|
||||
print("<< DATA:")
|
||||
chexdump(data)
|
||||
for i in range(0, len(data), 8192):
|
||||
self.dev.write(data[i:i + 8192])
|
||||
if progress:
|
||||
sys.stdout.write(".")
|
||||
sys.stdout.flush()
|
||||
if progress:
|
||||
print()
|
||||
# should automatically report a CRC failure
|
||||
self.reply(self.REQ_MEMWRITE)
|
||||
|
||||
def readmem(self, addr, size):
|
||||
req = struct.pack("<QQ", addr, size)
|
||||
self.cmd(self.REQ_MEMREAD, req)
|
||||
reply = self.reply(self.REQ_MEMREAD)
|
||||
checksum = struct.unpack("<I",reply[:4])[0]
|
||||
data = self.readfull(size)
|
||||
if self.debug:
|
||||
print(">> DATA:")
|
||||
chexdump(data)
|
||||
ccsum = self.checksum(data)
|
||||
if checksum != ccsum:
|
||||
raise UartCRCError("Reply data checksum error: Expected 0x%08x, got 0x%08x"%(checksum, ccsum))
|
||||
return data
|
||||
|
||||
def readstruct(self, addr, stype):
|
||||
return stype.parse(self.readmem(addr, stype.sizeof()))
|
||||
|
||||
class ProxyError(RuntimeError):
|
||||
pass
|
||||
|
||||
class ProxyCMDError(ProxyError):
|
||||
pass
|
||||
|
||||
class ProxyRemoteError(ProxyError):
|
||||
pass
|
||||
|
||||
class AlignmentError(Exception):
|
||||
pass
|
||||
|
||||
class M1N1Proxy:
|
||||
S_OK = 0
|
||||
S_BADCMD = -1
|
||||
|
||||
P_NOP = 0x000
|
||||
P_EXIT = 0x001
|
||||
P_CALL = 0x002
|
||||
P_GET_BOOTARGS = 0x003
|
||||
P_GET_BASE = 0x004
|
||||
P_SET_BAUD = 0x005
|
||||
|
||||
P_WRITE64 = 0x100
|
||||
P_WRITE32 = 0x101
|
||||
P_WRITE16 = 0x102
|
||||
P_WRITE8 = 0x103
|
||||
P_READ64 = 0x104
|
||||
P_READ32 = 0x105
|
||||
P_READ16 = 0x106
|
||||
P_READ8 = 0x107
|
||||
P_SET64 = 0x108
|
||||
P_SET32 = 0x109
|
||||
P_SET16 = 0x10a
|
||||
P_SET8 = 0x10b
|
||||
P_CLEAR64 = 0x10c
|
||||
P_CLEAR32 = 0x10d
|
||||
P_CLEAR16 = 0x10e
|
||||
P_CLEAR8 = 0x10f
|
||||
P_MASK64 = 0x110
|
||||
P_MASK32 = 0x111
|
||||
P_MASK16 = 0x112
|
||||
P_MASK8 = 0x113
|
||||
|
||||
P_MEMCPY64 = 0x200
|
||||
P_MEMCPY32 = 0x201
|
||||
P_MEMCPY16 = 0x202
|
||||
P_MEMCPY8 = 0x203
|
||||
P_MEMSET64 = 0x204
|
||||
P_MEMSET32 = 0x205
|
||||
P_MEMSET16 = 0x206
|
||||
P_MEMSET8 = 0x207
|
||||
|
||||
P_DC_FLUSHRANGE = 0x300
|
||||
P_DC_INVALRANGE = 0x301
|
||||
P_DC_FLUSHALL = 0x302
|
||||
P_IC_INVALALL = 0x303
|
||||
|
||||
def __init__(self, iface, debug=False):
|
||||
self.debug = debug
|
||||
self.iface = iface
|
||||
|
||||
def request(self, opcode, *args, reboot=False):
|
||||
if len(args) > 6:
|
||||
raise ValueError("Too many arguments")
|
||||
args = list(args) + [0] * (6 - len(args))
|
||||
req = struct.pack("<7Q", opcode, *args)
|
||||
if self.debug:
|
||||
print("<<<< %08x: %08x %08x %08x %08x %08x %08x"%tuple([opcode] + args))
|
||||
reply = self.iface.proxyreq(req, reboot=reboot)
|
||||
rop, status, retval = struct.unpack("<QqQ", reply)
|
||||
if self.debug:
|
||||
print(">>>> %08x: %d %08x"%(rop, status, retval))
|
||||
if reboot:
|
||||
return
|
||||
if rop != opcode:
|
||||
raise ProxyCMDError("Reply opcode mismatch: Expected 0x%08x, got 0x%08x"%(opcode,rop))
|
||||
if status != self.S_OK:
|
||||
if status == self.S_BADCMD:
|
||||
raise ProxyRemoteError("Reply error: Bad Command")
|
||||
else:
|
||||
raise ProxyRemoteError("Reply error: Unknown error (%d)"%status)
|
||||
return retval
|
||||
|
||||
def nop(self):
|
||||
self.request(self.P_NOP)
|
||||
def exit(self):
|
||||
self.request(self.P_EXIT)
|
||||
def call(self, addr, *args):
|
||||
if len(args) > 4:
|
||||
raise ValueError("Too many arguments")
|
||||
return self.request(self.P_CALL, addr, *args)
|
||||
def vector(self, addr, *args):
|
||||
if len(args) > 4:
|
||||
raise ValueError("Too many arguments")
|
||||
self.request(self.P_CALL, addr, *args, reboot=True)
|
||||
def get_bootargs(self):
|
||||
return self.request(self.P_GET_BOOTARGS)
|
||||
def get_base(self):
|
||||
return self.request(self.P_GET_BASE)
|
||||
def set_baud(self, baudrate):
|
||||
self.iface.tty_enable = False
|
||||
try:
|
||||
self.request(self.P_SET_BAUD, baudrate, 16, 0x005aa5f0)
|
||||
finally:
|
||||
self.iface.tty_enable = True
|
||||
|
||||
def write64(self, addr, data):
|
||||
if addr & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_WRITE64, addr, data)
|
||||
def write32(self, addr, data):
|
||||
if addr & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_WRITE32, addr, data)
|
||||
def write16(self, addr, data):
|
||||
if addr & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_WRITE16, addr, data)
|
||||
def write8(self, addr, data):
|
||||
self.request(self.P_WRITE8, addr, data)
|
||||
|
||||
def read64(self, addr):
|
||||
if addr & 7:
|
||||
raise AlignmentError()
|
||||
return self.request(self.P_READ64, addr)
|
||||
def read32(self, addr):
|
||||
if addr & 3:
|
||||
raise AlignmentError()
|
||||
return self.request(self.P_READ32, addr)
|
||||
def read16(self, addr):
|
||||
if addr & 1:
|
||||
raise AlignmentError()
|
||||
return self.request(self.P_READ16, addr)
|
||||
def read8(self, addr):
|
||||
return self.request(self.P_READ8, addr)
|
||||
|
||||
def set64(self, addr, data):
|
||||
if addr & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_SET64, addr, data)
|
||||
def set32(self, addr, data):
|
||||
if addr & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_SET32, addr, data)
|
||||
def set16(self, addr, data):
|
||||
if addr & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_SET16, addr, data)
|
||||
def set8(self, addr, data):
|
||||
self.request(self.P_SET8, addr, data)
|
||||
|
||||
def clear64(self, addr, data):
|
||||
if addr & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR64, addr, data)
|
||||
def clear32(self, addr, data):
|
||||
if addr & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR32, addr, data)
|
||||
def clear16(self, addr, data):
|
||||
if addr & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR16, addr, data)
|
||||
def clear8(self, addr, data):
|
||||
self.request(self.P_CLEAR8, addr, data)
|
||||
|
||||
def mask64(self, addr, clear, set):
|
||||
if addr & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR64, clear, set)
|
||||
def mask32(self, addr, clear, set):
|
||||
if addr & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR32, clear, set)
|
||||
def mask16(self, addr, clear, set):
|
||||
if addr & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_CLEAR16, clear, set)
|
||||
def mask8(self, addr, clear, set):
|
||||
self.request(self.P_CLEAR8, clear, set)
|
||||
|
||||
def memcpy64(self, dst, src, size):
|
||||
if src & 7 or dst & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMCPY64, dst, src, size)
|
||||
def memcpy32(self, dst, src, size):
|
||||
if src & 3 or dst & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMCPY32, dst, src, size)
|
||||
def memcpy16(self, dst, src, size):
|
||||
if src & 1 or dst & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMCPY16, dst, src, size)
|
||||
def memcpy8(self, dst, src, size):
|
||||
self.request(self.P_MEMCPY8, dst, src, size)
|
||||
|
||||
def memset64(self, dst, src, size):
|
||||
if dst & 7:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMSET64, dst, src, size)
|
||||
def memset32(self, dst, src, size):
|
||||
if dst & 3:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMSET32, dst, src, size)
|
||||
def memset16(self, dst, src, size):
|
||||
if dst & 1:
|
||||
raise AlignmentError()
|
||||
self.request(self.P_MEMSET16, dst, src, size)
|
||||
def memset8(self, dst, src, size):
|
||||
self.request(self.P_MEMSET8, dst, src, size)
|
||||
|
||||
def dc_flushrange(self, addr, size):
|
||||
self.request(self.P_DC_FLUSHRANGE, addr, size)
|
||||
def dc_invalrange(self, addr, size):
|
||||
self.request(self.P_DC_INVALRANGE, addr, size)
|
||||
def dc_flushall(self):
|
||||
self.request(self.P_DC_FLUSHALL)
|
||||
def ic_invalall(self):
|
||||
self.request(self.P_IC_INVALALL)
|
||||
|
||||
if __name__ == "__main__":
|
||||
import serial
|
||||
uartdev = os.environ.get("M1N1DEVICE", "/dev/ttyUSB0")
|
||||
usbuart = serial.Serial(uartdev, 115200)
|
||||
uartif = UartInterface(usbuart, debug=True)
|
||||
print("Sending NOP...", end=' ')
|
||||
uartif.nop()
|
||||
print("OK")
|
||||
proxy = M1N1Proxy(uartif, debug=True)
|
||||
print("Sending Proxy NOP...", end=' ')
|
||||
proxy.nop()
|
||||
print("OK")
|
||||
print("Boot args: 0x%x" % proxy.get_bootargs())
|
70
proxyclient/shell.py
Executable file
70
proxyclient/shell.py
Executable file
|
@ -0,0 +1,70 @@
|
|||
#!/usr/bin/python
|
||||
|
||||
import atexit, serial, os, struct, code, traceback, readline, rlcompleter
|
||||
from proxy import *
|
||||
import __main__
|
||||
import builtins
|
||||
|
||||
class HistoryConsole(code.InteractiveConsole):
|
||||
def __init__(self, locals=None, filename="<console>",
|
||||
histfile=os.path.expanduser("~/.m1n1-history")):
|
||||
code.InteractiveConsole.__init__(self, locals, filename)
|
||||
self.init_history(histfile)
|
||||
|
||||
def init_history(self, histfile):
|
||||
readline.parse_and_bind("tab: complete")
|
||||
if hasattr(readline, "read_history_file"):
|
||||
try:
|
||||
readline.read_history_file(histfile)
|
||||
except FileNotFoundError:
|
||||
pass
|
||||
atexit.register(self.save_history, histfile)
|
||||
|
||||
def save_history(self, histfile):
|
||||
readline.set_history_length(1000)
|
||||
readline.write_history_file(histfile)
|
||||
|
||||
def showtraceback(self):
|
||||
type, value, tb = sys.exc_info()
|
||||
traceback.print_exception(type, value, tb)
|
||||
|
||||
saved_display = sys.displayhook
|
||||
|
||||
def display(val):
|
||||
global saved_display
|
||||
if isinstance(val, int) or isinstance(val, int):
|
||||
builtins._ = val
|
||||
print(hex(val))
|
||||
else:
|
||||
saved_display(val)
|
||||
|
||||
sys.displayhook = display
|
||||
|
||||
# convenience
|
||||
h = hex
|
||||
|
||||
uartdev = os.environ.get("M1N1DEVICE", "/dev/ttyUSB0")
|
||||
usbuart = serial.Serial(uartdev, 115200)
|
||||
|
||||
iface = UartInterface(usbuart, debug=False)
|
||||
iface.nop()
|
||||
proxy = M1N1Proxy(iface, debug=False)
|
||||
proxy.nop()
|
||||
|
||||
locals = __main__.__dict__
|
||||
|
||||
for attr in dir(iface):
|
||||
locals[attr] = getattr(iface,attr)
|
||||
for attr in dir(proxy):
|
||||
locals[attr] = getattr(proxy,attr)
|
||||
del attr
|
||||
|
||||
from armutils import *
|
||||
from tgtypes import *
|
||||
|
||||
base = proxy.get_base()
|
||||
ba_addr = proxy.get_bootargs()
|
||||
ba = iface.readstruct(ba_addr, BootArgs)
|
||||
|
||||
HistoryConsole(locals).interact("Have fun!")
|
||||
|
28
proxyclient/tgtypes.py
Normal file
28
proxyclient/tgtypes.py
Normal file
|
@ -0,0 +1,28 @@
|
|||
# SPDX-License-Identifier: MIT
|
||||
from construct import *
|
||||
|
||||
BootArgs = Struct(
|
||||
"revision" / Hex(Int16ul),
|
||||
"version" / Hex(Int16ul),
|
||||
Padding(4),
|
||||
"virt_base" / Hex(Int64ul),
|
||||
"phys_base" / Hex(Int64ul),
|
||||
"mem_size" / Hex(Int64ul),
|
||||
"top_of_kernel_data" / Hex(Int64ul),
|
||||
"video" / Struct(
|
||||
"base" / Hex(Int64ul),
|
||||
"display" / Hex(Int64ul),
|
||||
"stride" / Hex(Int64ul),
|
||||
"width" / Hex(Int64ul),
|
||||
"height" / Hex(Int64ul),
|
||||
"depth" / Hex(Int64ul),
|
||||
),
|
||||
"machine_type" / Hex(Int32ul),
|
||||
Padding(4),
|
||||
"devtree" / Hex(Int64ul),
|
||||
"devtree_size" / Hex(Int32ul),
|
||||
"cmdline" / PaddedString(608, "ascii"),
|
||||
Padding(4),
|
||||
"boot_flags" / Hex(Int64ul),
|
||||
"mem_size_actual" / Hex(Int64ul),
|
||||
)
|
Loading…
Reference in a new issue