A Non-GUI Python API class for the Tiny SA Ultra written to support several personal projects
This uses official resources and documentation but is NOT endorsed by the official tinySA product
- The tinySA Ultra
- Requirements
- Library Usage
- Error Handling
- Example Implementations
- List of tinySA Commands and their Library Commands
- List of Commands Removed from Library
- Additional Library Commands
- Table of Command and Device Compatibility
- Notes for Beginners
- References
- Publications and Integration
- Licensing
This project requires numpy and pyserial.
Use 'pip install -r requirements.txt' to install the following dependencies:
pyserial
numpy
These dependencies cover only the API. Additional dependencies should be installed to follow the included examples and tests. These can be installed with 'pip install -r test_requirements.txt':
matplotlib
This library is currently only available as the tinySA class in 'tinySA_python.py' in this repository. It is very much under development and missing error checking and handling.
Several usage examples are provided in the Example Implementations section, including working with the hardware and plotting results with matplotlib.
Some error handling has been implemented for the individual functions. Most functions have a list of acceptable formats for input, which is included in the documentation and the 'libraryHelp' function. The 'tinySAHelp' function will get output from the current version of firmware running on the connected tinySA device.
Detailed error messages can be returned by toggling 'verbose' on.
From the official wiki USB Interface page:
There is limited error checking against incorrect parameters of incorrect mode
* Frequencies can be specified using an integer optionally postfixed with a the letter 'k' for kilo 'M' for Mega or 'G' for Giga. E.g. 0.1M (100kHz), 500k (0.5MHz) or 12000000 (12MHz)
* Levels are specified in dB(m) and can be specified using a floating point notation. E.g. 10 or 2.5
* Time is specified in seconds optionally postfixed with the letters 'm' for mili or 'u' for micro. E.g. 1 (1 second), 2.5 (2.5 seconds), 120m (120 milliseconds)
This library has been tested on Windows, but not yet on Unix systems. The primary difference should be the format of the serial port connection, but there may be smaller bugs in format that have not been detected yet.
There are several ways to list avilable serial ports.
-
Open Device Manager, scroll down to Ports (COM & LPT), and expand the menu. There should be a COM# port listing "USB Serial Device(COM #)". If your tinySA Ultra is set up to work with Serial, this will be it.
-
This uses the pyserial library requirement already installed for this library. It probably also works on Linux systems, but has not been tested yet.
import serial.tools.list_ports
ports = serial.tools.list_ports.comports()
for port, desc, hwid in ports:
print(f"Port: {port}, Description: {desc}, Hardware ID: {hwid}")
Example output for this method (on Windows) is as follows:
Port: COM4, Description: Standard Serial over Bluetooth link (COM4), Hardware ID: BTHENUM\{00001101-0000-1000-8000-00805F9B34FB}_LOCALMFG&0000\7&D0D1EE&0&000000000000_00000000
Port: COM3, Description: Standard Serial over Bluetooth link (COM3), Hardware ID: BTHENUM\{00001101-0000-1000-8000-00805F9B34FB}_LOCALMFG&0002\7&D0D1EE&0&B8B3DC31CBA8_C00000000
Port: COM10, Description: USB Serial Device (COM10), Hardware ID: USB VID:PID=0483:5740 SER=400 LOCATION=1-3
"COM10" is the port location of the tinySA Ultra that is used in the examples in this README.
TODO
This library returns strings as cleaned bytearrays. The command and first \r\n
pair are removed from the front, and the ch>
is removed from the end of the tinySA serial return.
The original message format:
bytearray(b'deviceid\r\ndeviceid 0\r\nch>')
Cleaned version:
bytearray(b'deviceid 0\r')
Show the process for initializing, opening the serial port, getting device info, and disconnecting
# import the library class
from src.tinySA_python import tinySA
# create a new tinySA object
tsa = tinySA()
# attempt to connect to previously discovered serial port
success = tsa.connect(port='COM10')
# if port open, then get device information and disconnect
if success == False:
print("ERROR: could not connect to port")
else:
msg = tsa.info()
print(msg)
tsa.disconnect()
Example output for this method is as follows:
bytearray(b'tinySA ULTRA\r\n2019-2024 Copyright @Erik Kaashoek\r\n2016-2020 Copyright @edy555\r\nSW licensed under GPL. See: https://github.com/erikkaashoek/tinySA\r\nVersion: tinySA4_v1.4-143-g864bb27\r\nBuild Time: Jan 10 2024 - 11:14:08\r\nKernel: 4.0.0\r\nCompiler: GCC 7.2.1 20170904 (release) [ARM/embedded-7-branch revision 255204]\r\nArchitecture: ARMv7E-M Core Variant: Cortex-M4F\r\nPort Info: Advanced kernel mode\r\nPlatform: STM32F303xC Analog & DSP\r')
Currently, the following can be used to turn on or off returned error messages.
- the 'verbose' option. When enabled, detailed messages are printed out.
# detailed messages are ON
tsa.setVerbose(True)
# detailed messages are OFF
tsa.setVerbose(False)
- the 'errorByte' option. When enabled, if there is an error with the command or configuration,
b'ERROR'
is returned instead of the defaultb''
.
# when an error occurs, b'ERROR' is returned
tsa.setErrorByteReturn(True)
# when an error occurs, the default b'' might be returned
tsa.setErrorByteReturn(False)
There are three options for help() with this library.
# the default help function
# 1 = help for this library, other values call the tinySA help function
tsa.help(1)
# calling the library help function directly
tsa.libraryHelp()
# calling the tinySA help directly
tsa.tinySAHelp()
All three return a bytearray in the format bytearray(b'commands:......')
TODO when error checking is complete to show multiple examples
See other sections for the following examples:
This example shows several examples for common data requests:
# import the library class for the tinySA
from src.tinySA_python import tinySA
# create a new tinySA object
tsa = tinySA()
# attempt to connect to previously discovered serial port
success = tsa.connect(port='COM10')
# if port open, then complete task(s) and disconnect
if success == False:
print("ERROR: could not connect to port")
else:
#detailed messages
tsa.setVerbose(True) #detailed messages
# get current trace data on screen
msg = tsa.data(val=2)
print(msg)
# set current device ID
msg = tsa.deviceid(3)
print(msg)
# get current device ID
msg = tsa.deviceid()
print(msg)
# get device information
msg = tsa.info()
print(msg)
# pause sweeping
msg = tsa.pause()
print(msg)
# resume sweeping
msg = tsa.resume()
print(msg)
# get current battery voltage (mV)
msg = tsa.vbat()
print(msg)
tsa.disconnect()
The capture()
function can be used to capture the screen and output it to an image file. Note that the screen size varies by device, and the serial read
This example truncates the last hex value, so a single padding x00
value has been added. This will eventually be investigated, but it's not hurting the output right now.
# import the library class for the tinySA
from src.tinySA_python import tinySA
# imports FOR THE EXAMPLE
import numpy as np
from PIL import Image
import struct
def convert_data_to_image(data_bytes, width, height):
# this is not a particularly pretty example, and the data_bytes is sometimes a byte short (TODO:fix)
# calculate the expected data size
expected_size = width * height * 2 # 16 bits per pixel (RGB565), 2 bytes per pixel
# error checking - brute force, but fine while developing
if len(data_bytes) < expected_size:
print(f"Data size is too small. Expected {expected_size} bytes, got {len(data_bytes)} bytes.")
# if the data size is off by 1 byte, add a padding byte
if len(data_bytes) == expected_size - 1:
print("Data size is 1 byte smaller than expected. Adding 1 byte of padding.")
# add a padding byte (0x00) to make the size match
data_bytes.append(0)
else:
return
elif len(data_bytes) > expected_size:
# truncate the data to the expected size (in case it's larger than needed)
data_bytes = data_bytes[:expected_size]
print("Data is larger than the expected size. trunacting. check data.")
# unpack the byte array to get pixel values (RGB565 format)
num_pixels = width * height
# unpacking as unsigned shorts (2 bytes each)
x = struct.unpack(f">{num_pixels}H", data_bytes)
# convert the RGB565 to RGBA
arr = np.array(x, dtype=np.uint32)
arr = 0xFF000000 + ((arr & 0xF800) >> 8) + ((arr & 0x07E0) << 5) + ((arr & 0x001F) << 19)
# reshape array to match the image dimensions. (height, width) format
arr = arr.reshape((height, width))
# create the image
img = Image.frombuffer('RGBA', (width, height), arr.tobytes(), 'raw', 'RGBA', 0, 1)
# save the image
img.save("capture_example.png")
# show the image
img.show()
# create a new tinySA object
tsa = tinySA()
# attempt to connect to previously discovered serial port
success = tsa.connect(port='COM10')
# if port closed, then return error message
if success == False:
print("ERROR: could not connect to port")
else: # port open, complete task(s) and disconnect
# detailed messages turned on
tsa.setVerbose(True)
# get the trace data
data_bytes = tsa.capture()
print(data_bytes)
tsa.disconnect()
# processing after disconnect (just for this example)
# test with 480x320 resolution for tinySA Ultra
convert_data_to_image(data_bytes, 480, 320)
Capture On-Screen Trace Data of a Frequency Sweep from 100 kHz to 800 kHz
This example plots the last/current sweep of data from the tinySA device.
data()
gets the trace data. frequencies()
gets the frequency values used.
byteArrayToNumArray(byteArr)
takes in the returned trace data and frequency
bytearrays and converts them to arrays that are then plotted using matplotlib
# import the library class for the tinySA
from src.tinySA_python import tinySA
# import matplotlib FOR THE EXAMPLE
import matplotlib.pyplot as plt
# functions used in this example
def byteArrayToNumArray(byteArr, enc="utf-8"):
# decode the bytearray to a string
decodedStr = byteArr.decode(enc)
# split the string by newline characters
stringVals = decodedStr.splitlines()
# convert each value to a float
floatVals = [float(val) for val in stringVals]
return floatVals
# create a new tinySA object
tsa = tinySA()
# attempt to connect to previously discovered serial port
success = tsa.connect(port='COM10')
# if port closed, then return error message
if success == False:
print("ERROR: could not connect to port")
else: # port open, complete task(s) and disconnect
# detailed messages turned on
tsa.setVerbose(True)
# get the trace data
data_bytes = tsa.data()
print(data_bytes)
# get the frequencies used by the last sweep
freq_bytes = tsa.frequencies()
tsa.disconnect()
# processing after disconnect (just for this example)
dataVals = byteArrayToNumArray(data_bytes)
print(len(dataVals)) # length of 450 data points
freqVals = byteArrayToNumArray(freq_bytes)
print(len(freqVals)) # length of 450 data points
# create the plot
plt.plot(freqVals, dataVals)
# add labels and title
plt.xlabel('Frequency (Hz)')
plt.ylabel('Signal Strength (dB)')
plt.title('Plot of Last Data Sweep')
# show the plot
plt.show()
Plotted On-Screen Trace Data of a Frequency Sweep from 100 kHz to 800 kHz
This list and the following list in the Additional Library Commands section describe the functions in this library.
This section is sorted by the tinySA (Ultra) commands, and includes:
- A brief description of what the command does
- What the original usage looked like
- The tinySA_python function call, or calls if multiple options exist
- Example return, or example format of return
- Any additional notes about the usage
All of the listed commands are included in this API to some degree, but error checking may be incomplete.
Quick Link Table:
- Status: Getting works, setting does not.
- Description: Gets the frequency correction set by CORRECT FREQUENCY menu in the expert menu settings
- Original Usage:
actual_freq [{frequency in Hz}]
- Library Function Call:
actual_freq(val=None|Int)
- Example Return: 3000000000
- Notes: freq in Hz going by the returns. Should be able to set the value with this, according to documentation, but its probably a format issue in the library.
- Status: Done
- Description: Enables/disables the build in Automatic Gain Control
- Original Usage:
agc 0..7|auto
- Library Function Call:
agc(val="auto"|0..7)
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Sets the internal attenuation
- Original Usage:
attenuate [auto|0-31]
- Library Function Call:
attenuate(val="auto"|0..31)
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: Sent by tinySA when in auto refresh mode
- Original Usage:
????
- Library Function Call:
bulk()
- Example Return:
format: "bulk\r\n{X}{Y}{Width}{Height} {Pixeldata}\r\n"
- Notes: All numbers are binary coded 2 bytes little endian. The pixel data is encoded as 2 bytes per pixel
- Status: Done
- Description: Sets or cancels one of the measurement modes
- Original Usage:
calc off|minh|maxh|maxd|aver4|aver16|quasip
- Library Function Call:
calc(val="off"|"minh"|"maxh"|"maxd"|"aver4"|"aver16"|"quasip")
- Example Return: empty bytearray
- Notes:
- the commands are the same as those listed in the MEASURE menu
- tinySA Calc Menu:
- OFF disables any calculation
- MIN HOLD sets the display to hold the minimum value measured. Reset the hold by selecting again. This setting is used to see stable signals that are within the noise
- MAX HOLD sets the display to hold the maximum value measured. Reset the hold by selecting again. This setting can be used for many measurements such as showing the power envelope of a modulated signal.
- MAX DECAY sets the display to hold the maximum value measured for a certain number of scans after which the maximum will start to decay. The default number of scans to hold is 20. This default can be changed in the SETTINGS menu. This setting is used instead of MAX HOLD to reduce the impact of spurious signals
- AVER 4 sets the averaging to new_measurement = old_measurement*3/4+measured_value/4. By default the averaging is linear power averaging
- AVER 16 sets the averaging to new_measurement = old_measurement*15/16+measured_value/16. By default the averaging is linear power averaging
- Status: Done
- Description: Disables or sets the caloutput to a specified frequency in MHz
- Original Usage:
caloutput off|30|15|10|4|3|2|1
- Library Function :
caloutput(val="off"|30|15|10|4|3|2|1)
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Requests a screen dump to be sent in binary format of HEIGHTxWIDTH pixels of each 2 bytes
- Original Usage:
capture
- Library Function Call:
capture()
- Example Return:
format:'\x00\x00\x00\x00\x00\x00\x00\...x00\x00\x00'
- Notes: tinySA original: 320x240, tinySA Ultra: 480x320
- Status: Done
- Description: Resets the configuration data to factory defaults
- Original Usage:
clearconfig
- Library Function Call:
clearconfig()
- Example Return:
b'Config and all cal data cleared. \r\nDo reset manually to take effect. Then do touch cal and save.\r'
- Notes: Requires password '1234'. Hardcoded. Other functions need to be used with this to complete the process.
- Status: Done
- Description: Sets or gets the colors used
- Original Usage:
color [{id} {rgb24}]
- Library Function Call:
color(ID=None|0..31, RGB=None(default:'0xF8FCF8')|'0x000000'..'0xFFFFFF')
- Example Return: If ID='None' used:
0: 0x000000\r\n 1: 0xF8FCF8\r\n 2: 0x808080\r\n 3: 0xE0E4E0\r\n 4: 0x000000\r\n 5: 0xD0D0D0\r\n 6: 0xF8FC00\r\n 7: 0x40FC40\r\n 8: 0xF800F8\r\n 9: 0xF84040\r\n 10: 0x18E000\r\n 11: 0xF80000\r\n 12: 0x0000F8\r\n 13: 0xF8FCF8\r\n 14: 0x808080\r\n 15: 0x00FC00\r\n 16: 0x808080\r\n 17: 0x000000\r\n 18: 0xF8FCF8\r\n 19: 0x0000F8\r\n 20: 0xF88080\r\n 21: 0x00FC00\r\n 22: 0x888C88\r\n 23: 0xD8DCD8\r\n 24: 0x282828\r\n 25: 0xC0C4C0\r\n 26: 0xF8FCF8\r\n 27: 0x00FC00\r\n 28: 0x00FCF8\r\n 29: 0xF8FC00\r\n 30: 0x000000\r\n 31: 0x000000\r'
- Notes: the hex value currently must be passed in as a string
- Status: Done
- Description: Sets or gets the frequency level correction table
- Original Usage:
correction {low|lna|ultra|ultra_lna|direct|direct_lna|harm|harm_lna|out|out_direct|out_adf|out_ultra|off|on} [{0-19} {frequency(Hz)} {value(dB)}]
- Library Function Call:
correct(tableName, slot, freq, val)
- Example Return: empty bytearray
- Notes: See Correction Wiki. The current content of the table is shown by entering
correction low
without any arguments. The data in the table can be modified by specifying the slot number and the new values. There MUST be one entry in the low table for 30MHz and the correction for that frequency MUST be zero. - Future Work: Confirm table format across devices. Value currently limited between -10 and 35, but this needs to be more specific.
- Status: Done
- Description: Sets or gets the dac value
- Original Usage:
dac [0..4095]
- Library Function Call(s):
dac(val=None|0..4095)
- Example Return:
b'usage: dac {value(0-4095)}\r\ncurrent value: 1922\r'
- Notes:
- Status: Done
- Description: Gets the trace data
- Original Usage:
data 0..2
- Library Function Call:
data(val=0|1|2)
- Example Return:
format bytearray(b'7.593750e+00\r\n-8.437500e+01\r\n-8.693750e+01\r\n...\r')
- Notes: 0 = temp value, 1 = stored trace, 2 = measurement. strength in decibels (dB)
- Status: Done
- Description: Sets or gets a user settable integer number ID that can be use to identify a specific tinySA connected to the PC
- Original Usage:
deviceid [{number}]
- Library Function Call:
deviceid(id=None|{int number})
- Example Return:
'deviceid 0\r'
- Notes:
- Status: TODO
- Description: ??
- Original Usage:
direct {start|stop|on|off} {freq(Hz)}
- Library Function Call:
direct()
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Sets the external attenuation/amplification
- Original Usage:
ext_gain -100..100
- Library Function Call:
ext_gain(val=-100...100)
- Example Return: empty bytearray
- Notes: Works in both input and output mode
- Status: TODO
- Description: Sent by tinySA when in auto refresh mode
- Original Usage:
- Library Function Call:
- Example Return:
format: "fill\r\n{X}{Y}{Width}{Height} {Color}\r\n"
- Notes: All numbers returned are binary coded 2 bytes little endian. Similar to 'bulk'???
- Status: Done
- Description: Pauses the sweep and sets the measurement frequency in Hz.
- Original Usage:
freq {frequency}
- Library Function Call:
freq()
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Gets the frequency correction.
- Original Usage:
freq_corr
- Library Function Call:
freq_corr()
- Example Return:
b'0 ppb\r'
- Notes: This command returns the frequency correction, in parts per billion (ppb).
- Status: Done
- Description: Gets the frequencies used by the last sweep
- Original Usage:
frequencies
- Library Function Call:
frequencies()
- Example Return:
b'1500000000\r\n... \r\n3000000000\r'
- Notes:
- Status: TODO libraryHelp()
- Description: Gets a list of the available commands
- Original Usage:
help
- Library Function Calls:
help(val=0|1)
tinySAHelp()
libraryHelp()
- Example Return:
- Notes: 0 = tinySAHelp(), 1=libraryHelp(). Both functions can also be called directly. libraryHelp() has more information about this library and the inputs.
- Status: TODO. needs error checking added
- Description: Measures the input level at each of the indicated frequencies.
- Original Usage:
hop {start(Hz)} {stop(Hz)} {step(Hz) | points} [outmask]
- Library Function Call:
- Example Return: empty bytearray
- Notes: Ultra only. From tinysa-org: if the 3rd parameter is below 450 it is assumed to be points, otherwise as step frequency Outmask selects if the frequency (1) or level (2) is output.
- Status: Done
- Description: Sets the intermediate frequency (IF) to automatic or a specific value
- Original Usage:
if (0|433M..435M )
- Library Function Call:
setIF(val=0|433M..435M|'auto')
- Example Return: empty bytearray
- Notes: Val input of 0 is 'auto'. Added explicit 'auto' to match other library funcs.
- Status: Done
- Description: Sets intermediate frequency (IF) to a specific value
- Original Usage:
if1 (975M..979M )
- Library Function Call:
setIF1(val=0|975M..979M|'auto')
- Example Return: empty bytearray
- Notes: Val input of 0 is 'auto'. Added explicit 'auto' to match other library funcs.
- Status: Done
- Description: Displays various software/firmware and hardware information
- Original Usage:
info
- Library Function Call:
info()
- Example Return:
b'tinySA ULTRA\r\n2019-2024 Copyright @Erik Kaashoek\r\n2016-2020 Copyright edy555\r\nSW licensed under GPL. See: https://github.com/erikkaashoek/tinySA\r\nVersion: tinySA4_v1.-143-g864bb27\r\nBuild Time: Jan 10 2024 - 11:14:08\r\nKernel: 4.0.0\r\nCompiler: GCC 7.2.1 20170904 (release) [ARM/embedded-7-branch revision 255204]\r\nArchitecture: ARMv7E-M Core Variant: Cortex-M4F\r\nPort Info: Advanced kernel mode\r\nPlatform:STM32F303xC Analog & DSP\r'
- Notes:
- Status: Done
- Description: Sets the output level
- Original Usage:
level -76..13
- Library Function Call:
level(val=-76...13)
- Example Return: empty bytearray
- Notes: Not all values in the range are available.
- Status: Done
- Description: Sets the output level delta for low output mode level sweep
- Original Usage:
levelchange -70..+70
- Library Function Call:
levelchange(val=-70...70)
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: Sets or gets the level calibration data
- Original Usage:
leveloffset low|high|switch [output] {error}
- Library Function Call:
- Example Return: empty bytearray
- Notes: For the output corrections first ensure correct output levels at maximum output level. For the low output set the output to -50dBm and measure and correct the level with "leveloffset switch error" where For all output leveloffset commands measure the level with the leveloffset to zero and calculate error = measured level - specified level
- Status: TODO
- Description:
- Original Usage:
line off|{level}
- Library Function Call:
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Loads a previously stored preset to the connected device
- Original Usage:
load 0..4
- Library Function Call:
load(val=0|1|2|3|4)
- Example Return: empty bytearray
- Notes: 0 is the startup preset
- Status: Done
- Description: Set lna usage off/on
- Original Usage:
lna off|on
- Library Function Call:
lna(val="off"|"on")
- Example Return: empty bytearray
- Notes: Should not be enabled when AGC is enabled - tinySA wiki SETTINGS2
- Status: TODO
- Description: sets or dumps marker info
- Original Usage:
marker {id} on|off|peak|{freq}| {index}
- Library Function Call:
marker()
- Example Return: empty bytearray
- Notes: where id=1..4 index=0..num_points-1 Marker levels will use the selected unit Marker peak will activate the marker (if not done already), position the marker on the strongest signal and display the marker info The frequency must be within the selected sweep range mode
- Status: TODO
- Description: The menu command can be used to activate any menu item based on the index of the menu item
- Original Usage:
menu {#} [{#} [{#} [{#}]]]
- Library Function Call:
menu([])
- Example Return: empty bytearray
- Notes: where # is the menu entry number starting with 1 at the top. Example: menu 6 2 will toggle the waterfall option tinySA Menu Tree
- Status: TODO
- Description: Sets the mode of the tinySA
- Original Usage:
mode low|high input|output
- Library Function Call:
mode(freq)
- Example Return: empty bytearray
- Notes: tinySA Wiki MODE
- LOW INPUT activates the 0.1-350MHz input mode
- HIGH INPUT activates the 240MHz-960MHz input mode
- LOW OUTPUT activates the 0.1-350MHz output mode
- HIGH OUTPUT activates the 240MHz-960MHz output mode
- Status: TODO
- Description: Set the modulation in output mode
- Original Usage:
modulation off|AM_1kHz|AM_10Hz|NFM|WFM|extern
- Library Function Call:
modulation()
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: ??
- Original Usage:
- Library Function Call:
- Example Return: empty bytearray
- Notes: Check documentation on this to see if more options
- Status: Done
- Description: Sets the output on or off
- Original Usage:
output on|off
- Library Function Call:
output(val="off"|"on")
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Pauses the sweeping in either input or output mode
- Original Usage:
pause
- Library Function Call:
pause()
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: sets the rbw to either automatic or a specific value
- Original Usage:
rbw auto|3..600
- Library Function Call:
rbw(val="auto"|3..600)
- Example Return: empty bytearray
- Notes: the number specifies the target rbw in kHz. Frequencies listed in official documentation: 3 kHz, 10 kHz, 30 kHz, 100 kHz, 300 kHz, 600 kHz
- Status: Done
- Description: Loads a previously stored preset from the device
- Original Usage:
recall 0..4
- Library Function Call:
recal(val=0|1|2|3|4)
- Example Return: empty bytearray
- Notes: Same functionality as
load()
. 0 is the startup preset.
- Status: Done
- Description: Enables or disables the auto refresh mode
- Original Usage:
refresh on|off
- Library Function Call:
refresh(val="off"|"on")
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Triggers a signal for the removal/release of the touch screen
- Original Usage:
release
- Library Function Call:
release()
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: does nothing
- Original Usage:
remark [use any text]
- Library Function Call:
- Example Return: empty bytearray
- Notes: ?? included due to documentation
- Status: Done
- Description: Sets the number of (re)measurements that should be taken at every frequency
- Original Usage:
repeat 1..1000
- Library Function Call:
repeat(val=1.1000)
- Example Return: empty bytearray
- Notes: increasing the repeat reduces the noise per frequency, repeat 1 is the normal scanning mode. Has int(val) conversion.
- Status: Done
- Description: Resets the tinySA
- Original Usage:
reset
- Library Function Call:
reset()
- Example Return: empty bytearray, serial error message. depends on the system.
- Notes: Disconnects the serial.
- Status: Done
- Description: Resumes the sweeping in either input or output mode
- Original Usage:
resume
- Library Function Call:
resume()
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Saves the current setting to a preset
- Original Usage:
save 0..4
- Library Function Call:
save(val=0..4)
- Example Return: empty bytearray
- Notes: where 0 is the startup preset
- Status: Done
- Description: Saves the device configuration data
- Original Usage:
saveconfig
- Library Function Call:
saveconfig()
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: Performs a scan and optionally outputs the measured data
- Original Usage:
scan {start(Hz)} {stop(Hz)} [points] [outmask]
- Library Function Call:
- Example Return: empty bytearray
- Notes: where the outmask is a binary OR of 1=frequencies, 2=measured data, 4=stored data and points is maximum 290
- Status: TODO
- Description: Performs a scan of unlimited amount of points and send the data in binary form
- Original Usage:
scanraw {start(Hz)} {stop(Hz)} [points][option]
- Library Function Call:
- Example Return: empty bytearray
- Notes: From Documentation 1: The measured data is send as '{' ('x' MSB LSB)*points '}' where the 16 bit data is scaled by 32.
From Documentation 2: The measured data is the level in dBm and is send as '{' ('x' MSB LSB)*points '}'. To get the dBm level from the 16 bit data, divide by 32 and subtract 128 for the tinySA and 174 for the tinySA Ultra. The option, when present, can be either 0,1,2 or 3 being the sum of 1=unbuffered and 2=continuous
UNDERGROING TESTING
- Status: TODO
- Description: Deletes a specific file on the sd card
- Original Usage:
sd_delete {filename}
- Library Function Call:
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Displays list of filenames with extension and sizes
- Original Usage:
sd_list
- Library Function Call:
sd_list()
- Example Return: format example:
-0.bmp 307322
- Notes:
- Status: TODO
- Description: Reads a specific file on the sd card
- Original Usage:
sd_read {filename}
- Library Function Call:
sd_read(filename)
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: performs one or all selftests
- Original Usage:
selftest 0 0..9
,selftest (1-3) [arg]
- Library Function Call:
selftest(val=0..9)
- Example Return: empty bytearray
- Notes: MUST have cable connected to RF and CAL
- Status: Done
- Description: Enables or disables spur reduction
- Original Usage:
spur on|off
- Library Function Call:
spur(val="off"|"on")
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Displays the current device status (paused/resumed)
- Original Usage:
status
- Library Function Call:
status()
- Example Return:
b'Resumed'
- Notes:
- Status: TODO
- Description: Set sweep boundaries or execute a sweep
- Original Usage:
sweep [(start|stop|center|span|cw {frequency}) | ({start(Hz)} {stop(Hz)} [0..290] ) ]
- Library Function Call:
- Example Return: empty bytearray
- Notes: sweep without arguments lists the current sweep settings, the frequencies specified should be within the permissible range. The sweep commands apply both to input and output modes
- Status: TODO
- Description: Sets the sweeptime
- Original Usage:
sweep {time(Seconds)}
- Library Function Call:
- Example Return: empty bytearray
- Notes: the time specified may end in a letter where m=mili and u=micro
- Status: Done
- Description: specifies the text entry for the active keypad
- Original Usage:
text keypadtext
- Library Function Call:
text(val="")
- Example Return: empty bytearray
- Notes: where keypadtext is the text used. Example: text 12M. Currently does not work for entering file names.
- Status: Done
- Description: lists information of the threads in the tinySA
- Original Usage:
threads
- Library Function Call:
threads()
- Example Return:
b'stklimit| |stk free| addr|refs|prio| state| name\r\n20000200|2000054C|00000248|200016A8| 0| 128| CURRENT| main\r\n20001530|2000157C|0000008C|20001650| 0| 1| READY| idle\r\n200053D8|200056C4|00000250|200059B0| 0| 127| READY| sweep\r'
- Notes:
- Status: Done
- Description: sends the coordinates of a touch
- Original Usage:
touch {X coordinate} {Y coordinate}
- Library Function Call:
touch(x=0,y=0)
- Example Return: empty bytearray
- Notes: The upper left corner of the screen is "0 0"
- Status: Done
- Description: starts the touch calibration
- Original Usage:
touchcal
- Library Function Call:
touchcal()
- Example Return: empty bytearray
- Notes: is there a way to cancel this?
- Status: Done
- Description: starts the touch test
- Original Usage:
touchtest
- Library Function Call:
touchtest()
- Example Return: empty bytearray
- Notes: instructions on screen "touch panel, draw lines, press button to complete"
- Status: TODO
- Description: displays all or one trace information or sets trace related information
- Original Usage:
trace [{0..2} | dBm|dBmV|dBuV| V|W |store|clear|subtract | (scale|reflevel) auto|{level}]
- Library Function Call:
- Example Return: empty bytearray
- Notes:
- Status: TODO
- Description: sets the trigger type or level
- Original Usage:
trigger auto|normal|single|{level(dBm)}
- Library Function Call:
- Example Return: empty bytearray
- Notes: the trigger level is always set in dBm
- Status: Done
- Description: turn on/config tiny SA ultra mode
- Original Usage:
ultra off|on|auto|start|harm {freq}
- Library Function Call:
ultra(val="on"|"off")
- Example Return: empty bytearray
- Notes: limited input until more documentation found
- Status: Done
- Description: Get port current baud rate/ gets the current serial config
- Original Usage:
usart_cfg
- Library Function Call:
usart_cfg()
- Example Return:
b'Serial: 115200 baud\r'
- Notes: default is 115,200
- Status: Done
- Description: Get the current battery voltage
- Original Usage:
vbat
- Library Function Call:
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: Sets or gets the battery offset value
- Original Usage:
vbat_offset [{0..4095}]
- Library Function Call:
vbat_offset(val=None|0..4095)
- Example Return:
b'300\r'
- Notes:
- Status: Done
- Description: returns the version text
- Original Usage:
version
- Library Function Call:
version()
- Example Return: empty bytearray
- Notes:
- Status: Done
- Description: wait for a single sweep to finish and pauses sweep or waits for specified number of seconds
- Original Usage:
wait [{seconds}]
- Library Function Call:
- Example Return: empty bytearray
- Notes:
- Status: TODO: documentation
- Description:
- Original Usage:
zero {level}\r\n174dBm
- Library Function Call:
- Example Return: empty bytearray
- Notes:
''' Full list of help commands commands: freq time dac nf saveconfig clearconfig zero sweep pause resume wait repeat status caloutput save recall trace trigger marker line usart_cfg vbat_offset color if if1 lna2 agc actual_freq freq_corr attenuate level sweeptime leveloffset levelchange modulation rbw mode spur lna direct ultra load ext_gain output deviceid correction calc menu text remark
Other commands: version reset data frequencies scan hop scanraw test touchcal touchtest usart capture refresh touch release vbat help info selftest sd_list sd_read sd_delete threads '''
The list of commands from 'help' that are still 'unknown' or don’t appear to have an impact on the tinySA via this Python API:
''' usart
'''
- Status: REMOVED. NOT ON DEVELOPER'S DUT
- Description: Sets the abortion enabled status (on/off) or aborts the previous command.
- Original Usage:
abort [off|on]
- Library Function Call:
abort(val=None|"off"|"on")
- Example Return: ????
- Notes: When used without parameters the previous command still running will be aborted. Abort must be enabled before using the "abort on" command. Additional error checking has been added with the 'verbose' option.
- Status: REMOVED until more documentation is confirmed
- Description: ??
- Original Usage:
lna2 0..7|auto
- Library Function Call:
lna2(val="auto"|0..7)
- Example Return: empty bytearray
- Notes:
- Status: REMOVED. Not recognized by DUT.
- Description: Restarts the tinySA after the specified number of seconds
- Original Usage:
restart {seconds}
- Library Function Call:
restart(val=0...)
- Example Return: empty bytearray
- Notes: where 0 seconds stops the restarting process. has int(val) conversion until it's clear if float() causes issues.
- Status: REMOVED
- Description: Sets the sweep voltage
- Original Usage:
sweep_voltage {0-3.3}
- Library Function Call:
sweep_voltage()
- Example Return: empty bytearray
- Notes: device does not recognize this
This library was developed with the tinySA Ultra, and there's some commands that might have been added/dropped between devices. This table is a record of CURRENT known compatibility with THIS library, and to what level the PC can interact with the device. It is HIGHLY likely to NOT be complete.
If a last checked firmware version is known, that is included in the header in the parenthesis.
?? is for an unfinished TODO list item, not an unknown. When a function is complete its compatibility is added.
Device Command | tinySA () | tinySA Ultra () | tinySA Ultra + () |
---|---|---|---|
abort | ?? | ||
actual_freq | Get Only | ||
agc | Set | ||
attenuate | Set | ||
bulk | ?? | ||
calc | Set | ||
caloutput | Set | ||
capture | Get | ||
clearconfig | Reset | ||
color | Set and Get | ||
correction | Set and Get | ||
dac | Set and Get | ||
data | Get | ||
deviceid | Set and Get | ||
direct | ?? | ||
ext_gain | Set | ||
fill | ?? | ||
freq | Set | ||
freq_corr | Get | ||
frequencies | Get | ||
help | Get | ||
hop | No | ?? | |
if | Set | ||
if1 | Set | ||
info | Get | ||
level | Set | ||
levelchange | Set | ||
leveloffset | ?? | ||
line | ?? | ||
load | Load to Device | ||
lna | Set | ||
marker | ?? | ||
menu | ?? | ||
mode | ?? | ||
modulation | ?? | ||
nf | ?? | ||
output | ?? | ||
pause | Set | ||
rbw | ?? | ||
recall | Load to Device | ||
refresh | |||
release | |||
remark | |||
repeat | |||
reset | |||
resume | |||
save | |||
saveconfig | |||
scan | |||
scanraw | |||
sd_delete | |||
sd_list | Get | ||
sd_read | |||
selftest | |||
spur | |||
status | |||
sweep | |||
sweeptime | |||
text | Set | ||
threads | Get | ||
touch | ?? | ||
touchcal | ?? | ||
touchtest | |||
trace | |||
trigger | |||
ultra | |||
usart_cfg | |||
vbat | |||
vbat_offset | |||
version | |||
wait | |||
zero |
This is a brief section for anyone that might have jumped in with a bit too much ambition. It is highly suggested to read the manual.
Very useful, important documentation can be found at:
- The tinySA wiki
- The getting started first use page
- Frequently asked questions (FAQs) can be found on the Wiki FAQs page
Running list of acronyms that get tossed around with little to no explanation. Googling is recommended if you are not familiar with these terms as they're essential to understanding device usage.
- AGC - Automatic Gain Control. This controls the overall dynamic range of the output when the input level(s) changes.
- Baud - Baud, or baud rate. The rate that information is transferred in a communication channel. A baud rate of 9600 means a max of 9600 bits per second is transmitted.
- dB - dB (decibel) and dBm (decibel-milliwatts). dB (unitless) quantifies the ratio between two values, whereas dBm expresses the absolute power level (always relative to 1mW).
- DUT - Device Under Test. Used here to refer to the singular device used while initially writing the API.
- IF - Intermediate Frequency. A frequency to which a carrier wave is shifted as an intermediate step in transmission or reception - Wikipedia
- LNA - Low Noise Amplifier. An electronic component that amplifies a very low-power signal without significantly degrading its signal-to-noise ratio - Wikipedia
- RBW - Resolution Bandwidth. Frequency span of the final filter (IF filter) that is applied to the input signal. Determines the fast-Fourier transform (FFT) bin size.
-
https://groups.io/g/tinysa/topic/tinysa_screen_capture_using/82218670
-
The firmware on github at https://github.com/erikkaashoek/tinySA
This API was written to support the work in:
L. Linkous, E. Karincic, M. Suche and E. Topsakal, "Reinforcement Learning Controlled Mechanically Reconfigurable Antennas," 2025 United States National Committee of URSI National Radio Science Meeting (USNC-URSI NRSM), Boulder, CO, USA, 2025
Other publications featuring the code in this repo will be added as they become public.
The code in this repository has been released under GPL-2.0 for lack of better idea right now. This licensing does not take priority over the official releases and the decisions of the tinySA team.