This section is intended to be a quick introduction to architecture of Sandhi without getting into unnecessary details.
Sandhi is fork of GRAS(GNU Radio Advanced Scheduler) at core, but it also incorporates other modules like Sciscipy etc.
GRAS is the application scheduler of Sandhi. It enables Sandhi to have closed-loop flowgraphs, dispatch threads, handle thread synchronization and much more.
To know more, checkout it's wiki
Essence of open-source development is to reuse the existing code and to build on top of it; several programs are generally written in high level programming languages such as Scilab/GNU-Octave since-
- It is easier to develop programs on them.
- Lot of libraries or toolboxes are availabe for such languages, which reduces the work.
Hence, Sandhi provides a way to reuse Scilab programs as functional blocks through a wrapper called Sciscipy. Sciscipy is essentially a wrapper written in C language to provide a mechanism to call Scilab's computational engine through Python while taking care of datatype conversion.
To know more, here's link to it's official page
Sandhi uses GNU Radio (GR) V3.6 with GNU Radio Companion(GRC) as its front-end. GNU Radio is open source software for implementing software radio visually through flowgraphs and blocks. A flow graph in GNU Radio is visual representation of data flow between two or more nodes, and these nodes are called blocks (which process the flowing data). Sandhi inherits this capability of intuitively implementing logic through flowgraph which makes it a good LabVIEW replacement.
Since sandhi is still in beta stages, some of user-specific block may be missing; but this is not really a problem, since a user with limited python experience can create block himself/herself using Sandhi's framework. To follow a descriptive, step-by-step and conventional guide check GNU Radio's Out of Tree Module. A block can be either coded in python or C++; but given python is easier to start with, we prefer coding in python. This section will present a rapid but unconventional approach: impatience is virtue
A block in Sandhi essentially requires two things:
- A xml file (front end block description which is parsed by GRC to display that block.)
- A python file (which describes how the block functions.)
Sandhi, by default, reads any xml block present in ~/.grc_gnuradio. Hence all our initial development will be done in the .grc_gnuradio folder.
Sample code is availabe with annotations here
We'd be pleased to include your block if it suits a majority of Sandhi-users; so to include your block with Sandhi either:
- Send us a pull request of your block, we'll accept it and integrate it in Sandhi for you
- Do it yourself and then send us a pull request on github (to get the warm fuzzy feeling of having helped us)
The CMake files in Sandhi check for dependencies, proper namespace and structure before Sandhi is compiled as source; So one has to modify CMakefiles and structure his/her code files properly for Sandhi to include one's block.
To integrate it, you need to
- Decide on a namespace and stick to it; here we'll use myblock as our namespace.
- Create a folder in sandhi/gr36 folder with proper name/prefix (For Example: gr-myblock)
- Fastest way is to restructure your files according to a given block in Sandhi. Copy the block and study its folder content and CMakeLists.txt files laziness is virtue.
- Copy your xml in ../gr-myblock/grc/ and edit its import section to reflect integration changes. For example: import myblock will become from gnuradio.myblock import myblock as myblock
- Modify it's CMakeLists.txt to reflect changes.
- Similarly modify CMakeLists.txt in ../gr-myblock/python/ and ../gr-myblock/doc/ the one in myblock/ folder itself; add additional dependeny check if your block requires any.
- The top_level CMakeLists.txt in gr36/ folder should include this line
_add_subdirectory(gr-myblock)_
This section gives hints on how to use Sciscipy in Sandhi to develop blocks. There are two ways of calling Sciscipy from work_function in python.
- By importing Scilab to use pre-built functions
from scilab import Scilab
sci_obj = Scilab()
output_items[0][:2] = sci_obj.rand(1,2) # This outputs a 1x2 Array of Random number generated by Scilab
- By importing Sciscipy to to perform a task in scilab and then return output to python
import sciscipy
# Function returns random number generated array computed from scilab.
def my_sci_func(x, y):
# Write your scilab code as strings
code_statement1 = "x =" + str(x) + ";"
code_statement2 = "y = " + str(y) + ";"
code_statement3 = "z = rand(x,y);"
combined_statement = code_statement1 + code_statement2 + code_statement3
sciscipy.eval(combined_statement)
# function return value
z = sciscipy.read("z")
return z
output_items[0][:2] = my_sci_func(1,2)
we realize this is the not the best way to code, but hey, if it works, it works