zero scan and documentation
This commit is contained in:
parent
2b698fc08a
commit
f30f6f1ad5
|
@ -110,11 +110,14 @@ buildroot-clean:
|
||||||
-docker container stop upsilon-buildroot
|
-docker container stop upsilon-buildroot
|
||||||
-docker container rm upsilon-buildroot
|
-docker container rm upsilon-buildroot
|
||||||
|
|
||||||
###### TFTP
|
###### Execute
|
||||||
|
|
||||||
tftp:
|
tftp:
|
||||||
cd ../boot && py3tftp --host 192.168.1.100 -p 6969 -v
|
cd ../boot && py3tftp --host 192.168.1.100 -p 6969 -v
|
||||||
|
|
||||||
|
copy:
|
||||||
|
scp ../boot/mmio.py ../linux/comm.py upsilon:~/
|
||||||
|
|
||||||
###### External projects
|
###### External projects
|
||||||
|
|
||||||
clone: f4pga buildroot litex opensbi
|
clone: f4pga buildroot litex opensbi
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
"""
|
||||||
|
Copyright 2023 (C) Peter McGoron
|
||||||
|
|
||||||
|
This file is a part of Upsilon, a free and open source software project.
|
||||||
|
For license terms, refer to the files in `doc/copying` in the Upsilon
|
||||||
|
source distribution.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from pssh.clients import SSHClient
|
||||||
|
import numpy as np
|
||||||
|
import matplotlib.pyplot as plt
|
||||||
|
import pandas as pd
|
||||||
|
import sys
|
||||||
|
|
||||||
|
def sign_extend(value, bits):
|
||||||
|
is_signed = (value >> (bits - 1)) & 1 == 1
|
||||||
|
if not is_signed:
|
||||||
|
return value
|
||||||
|
return -((~value + 1) & ((1 << (bits - 1)) - 1))
|
||||||
|
|
||||||
|
client = SSHClient('192.168.1.50', user='root', pkey='~/.ssh/upsilon_key')
|
||||||
|
client.scp_send('../linux/noise_test.py', '/root/noise_test.py')
|
||||||
|
out = client.run_command('micropython noise_test.py')
|
||||||
|
|
||||||
|
current_dac = None
|
||||||
|
current_adc = []
|
||||||
|
x_ax = []
|
||||||
|
y_ax = []
|
||||||
|
for line in out.stdout:
|
||||||
|
l = line.split(' ')
|
||||||
|
if l[0] != current_dac:
|
||||||
|
if current_dac is not None:
|
||||||
|
m = np.mean(current_adc)
|
||||||
|
sdev = np.std(current_adc)
|
||||||
|
print(current_dac, m, sdev)
|
||||||
|
x_ax.append(current_dac)
|
||||||
|
y_ax.append(m)
|
||||||
|
current_adc = [sign_extend(int(l[1]), 18)]
|
||||||
|
current_dac = l[0]
|
||||||
|
else:
|
||||||
|
current_adc.append(sign_extend(int(l[1]),18))
|
||||||
|
|
||||||
|
df = pd.DataFrame({"x": x_ax, "y": y_ax})
|
||||||
|
df.to_csv(f"{sys.argv[1]}.csv")
|
||||||
|
plt.plot(df.x, df.y)
|
||||||
|
plt.show()
|
|
@ -0,0 +1,46 @@
|
||||||
|
Copyright 2023 (C) Peter McGoron.
|
||||||
|
|
||||||
|
This file is a part of Upsilon, a free and open source software project.
|
||||||
|
For license terms, refer to the files in `doc/copying` in the Upsilon
|
||||||
|
source distribution.
|
||||||
|
|
||||||
|
__________________________________________________________________________
|
||||||
|
|
||||||
|
This manual describes the controller software programming. This guide does not
|
||||||
|
describe client programming (programs that run on the client and interface with
|
||||||
|
the controller). It does not describe Verilog: see `verilog_manual.md` for
|
||||||
|
that.
|
||||||
|
|
||||||
|
# Preqreuisites
|
||||||
|
|
||||||
|
You must know basic Linux shell (change directories, edit files with `vi`)
|
||||||
|
and basic SSH usage (sftp, ssh).
|
||||||
|
|
||||||
|
Knowledge of Micropython (a subset of Python) is required for scripting.
|
||||||
|
|
||||||
|
I assume that you have the controller running and accessable. See `docker.md`
|
||||||
|
for the easy quick-start guide.
|
||||||
|
|
||||||
|
# Programming in MicroPython
|
||||||
|
|
||||||
|
## Introduction to MicroPython
|
||||||
|
|
||||||
|
MicroPython is a programming language that is very similar to Python. It is
|
||||||
|
stripped down and designed to run on very small devices. If you have written
|
||||||
|
Python, you will be able to use MicroPython without issue. If you are not
|
||||||
|
a hardcore Python programmer, you might not even notice a difference.
|
||||||
|
|
||||||
|
Everything you need to know is [here](https://docs.micropython.org).
|
||||||
|
|
||||||
|
## Standard Library
|
||||||
|
|
||||||
|
There are two modules of the standard library: `mmio` and `comm`.
|
||||||
|
|
||||||
|
`mmio` are wrappers that handle reads and writes from MMIO pins. This file
|
||||||
|
is automatically generated by the build process. This file is generated in
|
||||||
|
the `gateware` directory (if you use the Docker build system, the file is
|
||||||
|
automatically copied to `boot/mmio.py`).
|
||||||
|
|
||||||
|
`comm` contains higher level wrappers for DAC and ADC pins. This module is
|
||||||
|
documented well enough that you should be able to read it and understand
|
||||||
|
how to use it.
|
|
@ -116,3 +116,9 @@ UART.
|
||||||
Run `litex_term /dev/ttyUSB1`. You should get messages in the window with
|
Run `litex_term /dev/ttyUSB1`. You should get messages in the window with
|
||||||
the TFTP server that the FPGA has connected to the server. Eventually you
|
the TFTP server that the FPGA has connected to the server. Eventually you
|
||||||
will get a login prompt: you have sucessfully loaded Upsilon onto your FPGA.
|
will get a login prompt: you have sucessfully loaded Upsilon onto your FPGA.
|
||||||
|
|
||||||
|
## Copy Library
|
||||||
|
|
||||||
|
Run `make copy` to copy the Micropython Upsilon library to the FPGA. After
|
||||||
|
this the modules `comm` and `mmio` are available when running scripts in
|
||||||
|
`/root`.
|
||||||
|
|
|
@ -1,27 +0,0 @@
|
||||||
Copyright 2023 (C) Peter McGoron.
|
|
||||||
|
|
||||||
This file is a part of Upsilon, a free and open source software project.
|
|
||||||
For license terms, refer to the files in `doc/copying` in the Upsilon
|
|
||||||
source distribution.
|
|
||||||
|
|
||||||
__________________________________________________________________________
|
|
||||||
|
|
||||||
The User Manual is targeted towards non-programmers using Upsilon.
|
|
||||||
|
|
||||||
# Preqreuisites
|
|
||||||
|
|
||||||
You will need to know the basics of Git. Git is the system used to track
|
|
||||||
changes and update Upsilon. you will need to know the wwhat a git repository is,
|
|
||||||
how to pull changes from a repository, what commit hashes are and how to make
|
|
||||||
branches.
|
|
||||||
|
|
||||||
You must know basic Linux shell (change directories, edit files with `vi`)
|
|
||||||
and basic SSH usage (sftp, ssh).
|
|
||||||
|
|
||||||
Knowledge of Micropython (a subset of Python) is required for scripting.
|
|
||||||
|
|
||||||
|
|
||||||
# Building and Booting
|
|
||||||
|
|
||||||
Follow `docs/docker.md` to setup the build environment, build Upsilon, and
|
|
||||||
boot Upsilon.
|
|
|
@ -11,11 +11,43 @@
|
||||||
#
|
#
|
||||||
# TODO: Devicetree?
|
# TODO: Devicetree?
|
||||||
|
|
||||||
|
import argparse
|
||||||
import json
|
import json
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
class CSRGenerator:
|
class MMIORegister:
|
||||||
|
def __init__(self, name, read_only=False, number=1, exntype=None):
|
||||||
|
"""
|
||||||
|
Describes a MMIO register.
|
||||||
|
:param name: The name of the MMIO register. This name must be the
|
||||||
|
same as the pin name used in ``csr.json``, except for any
|
||||||
|
numerical suffix.
|
||||||
|
:param read_only: True if the register is read only. Defaults to
|
||||||
|
``False``.
|
||||||
|
:param number: The number of MMIO registers with the same name
|
||||||
|
and number suffixes. The number suffixes must go from 0
|
||||||
|
to ``number - 1`` with no gaps.
|
||||||
|
"""
|
||||||
|
self.name = name
|
||||||
|
self.read_only = read_only
|
||||||
|
self.number = number
|
||||||
|
self.exntype = exntype
|
||||||
|
|
||||||
|
def mmio_factory(dac_num, exntype):
|
||||||
|
def f(name, read_only=False):
|
||||||
|
return MMIORegister(name, read_only, numer=dac_num, exntype=exntype)
|
||||||
|
return f
|
||||||
|
|
||||||
|
|
||||||
|
class MicroPythonCSRGenerator:
|
||||||
def __init__(self, csrjson, bitwidthjson, registers, outf):
|
def __init__(self, csrjson, bitwidthjson, registers, outf):
|
||||||
|
"""
|
||||||
|
This class generates a MicroPython wrapper for MMIO registers.
|
||||||
|
|
||||||
|
:param csrjson: Filename of a LiteX "csr.json" file.
|
||||||
|
:param bitwidthjson: Filename of an Upsilon "bitwidthjson" file.
|
||||||
|
:param registers: A list of
|
||||||
|
"""
|
||||||
self.registers = registers
|
self.registers = registers
|
||||||
self.csrs = json.load(open(csrjson))
|
self.csrs = json.load(open(csrjson))
|
||||||
self.bws = json.load(open(bitwidthjson))
|
self.bws = json.load(open(bitwidthjson))
|
||||||
|
@ -70,13 +102,17 @@ class CSRGenerator:
|
||||||
assert len(acc) == 2
|
assert len(acc) == 2
|
||||||
self.print(f'{indent}return {acc[0]} | ({acc[1]} << 32)\n')
|
self.print(f'{indent}return {acc[0]} | ({acc[1]} << 32)\n')
|
||||||
|
|
||||||
def print_fun(self, optype, name, regnum, pfun):
|
def print_fun(self, optype, reg, pfun):
|
||||||
"""Print out a read/write function for an MMIO register.
|
"""Print out a read/write function for an MMIO register.
|
||||||
* {optype} is set to "read" or "write" (the string).
|
|
||||||
* {name} is set to the name of the MMIO register, without number suffix.
|
:param optype: is set to "read" or "write" (the string).
|
||||||
* {regnum} is set to the amount of that type oF MMIO register exists.
|
:param reg: is the dictionary containing the register info.
|
||||||
* {pfun} is set to {self.print_write_register} or {self.print_read_register}
|
:param pfun: is set to {self.print_write_register} or {self.print_read_register}
|
||||||
"""
|
"""
|
||||||
|
name = reg['name']
|
||||||
|
regnum = reg['total']
|
||||||
|
exntype = reg['exntype]'
|
||||||
|
|
||||||
self.print(f'def {optype}_{name}(')
|
self.print(f'def {optype}_{name}(')
|
||||||
|
|
||||||
printed_argument = False
|
printed_argument = False
|
||||||
|
@ -101,26 +137,42 @@ class CSRGenerator:
|
||||||
self.print(f'num == {i}:\n')
|
self.print(f'num == {i}:\n')
|
||||||
pfun('\t\t', 'val', name, i)
|
pfun('\t\t', 'val', name, i)
|
||||||
self.print(f'\telse:\n')
|
self.print(f'\telse:\n')
|
||||||
self.print(f'\t\traise Exception("invalid {name}", regnum)\n')
|
self.print(f'\t\traise {exntype}(regnum)\n')
|
||||||
self.print('\n')
|
self.print('\n')
|
||||||
|
|
||||||
def print_file(self):
|
def print_file(self):
|
||||||
self.print('import machine\n')
|
self.print('import machine\n')
|
||||||
|
self.print('class InvalidDACException(Exception):\n\tpass\n')
|
||||||
|
self.print('class InvalidADCException(Exception):\n\tpass\n')
|
||||||
for reg in self.registers:
|
for reg in self.registers:
|
||||||
self.print_fun('read', reg['name'], reg['total'], self.print_read_register)
|
self.print_fun('read', reg, self.print_read_register)
|
||||||
if not reg['read_only']:
|
if not reg['read_only']:
|
||||||
self.print_fun('write', reg['name'], reg['total'], self.print_write_register)
|
self.print_fun('write', reg, self.print_write_register)
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
dac_num = 8
|
dac_num = 8
|
||||||
adc_num = 8
|
adc_num = 8
|
||||||
|
dac_reg = mmio_factory(dac_num, "InvalidDACException")
|
||||||
|
adc_reg = mmio_factory(adc_num, "InvalidADCException")
|
||||||
|
|
||||||
registers = [
|
registers = [
|
||||||
{"read_only": False, "name": "dac_sel", "total": dac_num},
|
dac_reg("dac_sel"),
|
||||||
{"read_only": True, "name": "dac_finished", "total": dac_num},
|
dac_reg("dac_finished", read_only=True),
|
||||||
{"read_only": False, "name": "dac_arm", "total": dac_num},
|
dac_reg("dac_arm"),
|
||||||
{"read_only": True, "name": "dac_recv_buf", "total": dac_num},
|
dac_reg("dac_recv_buf", read_only=True),
|
||||||
{"read_only": False, "name": "dac_send_buf", "total": dac_num},
|
dac_reg("dac_send_buf"),
|
||||||
|
|
||||||
|
adc_reg("adc_finished", read_only=True),
|
||||||
|
adc_reg("adc_arm"),
|
||||||
|
adc_reg("adc_recv_buf", read_only=True),
|
||||||
|
adc_reg("adc_sel"),
|
||||||
|
|
||||||
|
MMIORegister("cl_in_loop", read_only=True),
|
||||||
|
MMIORegister("cl_cmd"),
|
||||||
|
MMIORegister("cl_word_in"),
|
||||||
|
MMIORegister("cl_start_cmd"),
|
||||||
|
MMIORegister("cl_finish_cmd", read_only=True),
|
||||||
|
|
||||||
# {"read_only": False, "name": "wf_arm", "total": dac_num},
|
# {"read_only": False, "name": "wf_arm", "total": dac_num},
|
||||||
# {"read_only": False, "name": "wf_halt_on_finish", "total": dac_num},
|
# {"read_only": False, "name": "wf_halt_on_finish", "total": dac_num},
|
||||||
# {"read_only": True, "name": "wf_finished", "total": dac_num},
|
# {"read_only": True, "name": "wf_finished", "total": dac_num},
|
||||||
|
@ -129,17 +181,5 @@ if __name__ == "__main__":
|
||||||
# {"read_only": False, "name": "wf_refresh_start", "total": dac_num},
|
# {"read_only": False, "name": "wf_refresh_start", "total": dac_num},
|
||||||
# {"read_only": True, "name": "wf_refresh_finished", "total": dac_num},
|
# {"read_only": True, "name": "wf_refresh_finished", "total": dac_num},
|
||||||
# {"read_only": False, "name": "wf_start_addr", "total": dac_num},
|
# {"read_only": False, "name": "wf_start_addr", "total": dac_num},
|
||||||
|
|
||||||
{"read_only": True, "name": "adc_finished", "total": adc_num},
|
|
||||||
{"read_only": False, "name": "adc_arm", "total": adc_num},
|
|
||||||
{"read_only": True, "name": "adc_recv_buf", "total": adc_num},
|
|
||||||
{"read_only": False, "name": "adc_sel", "total": adc_num},
|
|
||||||
|
|
||||||
{"read_only": True, "name": "cl_in_loop", "total": 1},
|
|
||||||
{"read_only": False, "name": "cl_cmd", "total": 1},
|
|
||||||
{"read_only": False, "name": "cl_word_in", "total": 1},
|
|
||||||
{"read_only": False, "name": "cl_word_out", "total": 1},
|
|
||||||
{"read_only": False, "name": "cl_start_cmd", "total": 1},
|
|
||||||
{"read_only": True, "name": "cl_finish_cmd", "total": 1},
|
|
||||||
]
|
]
|
||||||
CSRGenerator("csr.json", "csr_bitwidth.json", registers, sys.stdout).print_file()
|
MicroPythonCSRGenerator("csr.json", "csr_bitwidth.json", registers, sys.stdout).print_file()
|
||||||
|
|
|
@ -57,6 +57,9 @@ from litedram.frontend.dma import LiteDRAMDMAReader
|
||||||
from liteeth.phy.mii import LiteEthPHYMII
|
from liteeth.phy.mii import LiteEthPHYMII
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
Keep this diagram up to date! This is the wiring diagram from the ADC to
|
||||||
|
the named Verilog pins.
|
||||||
|
|
||||||
Refer to `A7-constraints.xdc` for pin names.
|
Refer to `A7-constraints.xdc` for pin names.
|
||||||
DAC: SS MOSI MISO SCK
|
DAC: SS MOSI MISO SCK
|
||||||
0: 1 2 3 4 (PMOD A top, right to left)
|
0: 1 2 3 4 (PMOD A top, right to left)
|
||||||
|
|
|
@ -1,25 +1,48 @@
|
||||||
|
# Copyright 2023 (C) Peter McGoron
|
||||||
|
# This file is a part of Upsilon, a free and open source software project.
|
||||||
|
# For license terms, refer to the files in `doc/copying` in the Upsilon
|
||||||
|
# source distribution.
|
||||||
|
#
|
||||||
|
# Upsilon Micropython Standard Library.
|
||||||
|
|
||||||
from mmio import *
|
from mmio import *
|
||||||
|
|
||||||
def dac_write_value(val, num):
|
# Write a 20 bit twos-complement value to a DAC.
|
||||||
write_dac_send_buf(1 << 20 | val & 0xFFFFF, num) # 20 bit DAC
|
def dac_write_volt(val, num):
|
||||||
|
"""
|
||||||
|
Write a 20 bit twos-complement value to a DAC.
|
||||||
|
|
||||||
|
:param val: Two's complement 20 bit integer. The number is bitmasked
|
||||||
|
to the appropriate length, so negative Python integers are also
|
||||||
|
accepted. This DOES NOT check if the integer actually fits in 20
|
||||||
|
bits.
|
||||||
|
|
||||||
|
:param num: DAC number.
|
||||||
|
:raises Exception:
|
||||||
|
"""
|
||||||
|
write_dac_send_buf(1 << 20 | (val & 0xFFFFF), num)
|
||||||
write_dac_arm(1, num)
|
write_dac_arm(1, num)
|
||||||
write_dac_arm(0, num)
|
write_dac_arm(0, num)
|
||||||
|
|
||||||
def dac_read_value(val, num):
|
# Read a register from a DAC.
|
||||||
|
def dac_read_reg(val, num):
|
||||||
write_dac_send_buf(1 << 23 | val, num)
|
write_dac_send_buf(1 << 23 | val, num)
|
||||||
write_dac_arm(1, num)
|
write_dac_arm(1, num)
|
||||||
write_dac_arm(0, num)
|
write_dac_arm(0, num)
|
||||||
return read_dac_recv_buf(num)
|
return read_dac_recv_buf(num)
|
||||||
|
|
||||||
|
# Initialize a DAC by setting it's output value to 0, and
|
||||||
|
# removing the output restriction from the settings register.
|
||||||
def dac_init(num):
|
def dac_init(num):
|
||||||
write_dac_sel(0,num)
|
write_dac_sel(0,num)
|
||||||
dac_write_value(0, num)
|
dac_write_volt(0, num)
|
||||||
write_dac_send_buf(1 << 22 | 1 << 2, num)
|
write_dac_send_buf(1 << 21 | 1 << 1, num)
|
||||||
write_dac_arm(1, num)
|
write_dac_arm(1, num)
|
||||||
write_dac_arm(0, num)
|
write_dac_arm(0, num)
|
||||||
return dac_read_value(1 << 22, num)
|
return dac_read_reg(1 << 21, num)
|
||||||
|
|
||||||
def adc_read_value(num):
|
# Read a value from an ADC.
|
||||||
|
def adc_read(num):
|
||||||
write_adc_arm(1, num)
|
write_adc_arm(1, num)
|
||||||
write_adc_arm(0, num)
|
write_adc_arm(0, num)
|
||||||
return read_from_adc(num)
|
return read_adc_recv_buf(num)
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
from comm import *
|
||||||
|
from time import sleep_ms
|
||||||
|
|
||||||
|
for i in range(-300,300):
|
||||||
|
dac_write_volt(i, 0)
|
||||||
|
for j in range(0,20):
|
||||||
|
print(i, adc_read(0))
|
||||||
|
|
Loading…
Reference in New Issue