SCPI parser and status register model implementation (device side) written in C
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
scpi_parser/example/example.c

168 lines
3.3 KiB

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "scpi.h"
// ------- TESTING ----------
static void send_cmd(const char *cmd)
{
printf("\n> %s\n", cmd);
scpi_handle_string(cmd);
}
int main(void)
{
send_cmd("*IDN?\n"); // builtin commands..
send_cmd("*SRE 4\n"); // enable SRQ on error
send_cmd("FOO:BAR:BAZ\n"); // invalid command causes error
send_cmd("SYST:ERR:COUNT?\n"); // error subsystem
send_cmd("SYST:ERR:NEXT?; COUNT?; NEXT?\n"); // semicolon works according to spec
send_cmd("DATA:BLOB #216abcdefghijklmnop\n"); // binary data block
send_cmd("APPLY:SINE 50, 1.0, 2.17\n"); // floats
send_cmd("DISP:TEXT \"Hello world\"\n"); // string
// test user error
send_cmd("*CLS\n"); // string
send_cmd("USERERROR\n"); // string
send_cmd("SYST:ERR:COUNT?\n");
send_cmd("SYST:ERR:NEXT?\n");
send_cmd("ERROR_FALLBACK\n"); // test fallback to closest related error
// test chardata
send_cmd("CHARD FOOBAR123_MOO_abcdef_HELLO, 12\n");
send_cmd("SYST:ERR:ALL?\n");
}
// ---- Test device impl ----
// Newline sequence for responses.
// Device accepts botn \r\n and \n in incomming messages.
const char *scpi_eol = "\r\n";
const SCPI_error_desc scpi_user_errors[] = {
{10, "Custom error"},
{0} // terminator
};
void scpi_send_byte_impl(uint8_t b)
{
putchar(b); // device sends a byte
}
const char *scpi_user_IDN(void)
{
return "MightyPork,Test SCPI device,0,0.1";
}
/** Error callback */
void scpi_user_error(int16_t errno, const char * msg)
{
printf("### ERROR ADDED: %d, %s ###\n", errno, msg);
}
/** Service request impl */
void scpi_user_SRQ(void)
{
// NOTE: Actual instrument should send SRQ event somehow
printf("[Service Request]\n");
}
// ---- INSTRUMENT COMMANDS ----
void cmd_APPL_SIN_cb(const SCPI_argval_t *args)
{
printf("cb APPLy:SINe %d, %f, %f\n", args[0].INT, args[1].FLOAT, args[2].FLOAT);
}
void cmd_DISP_TEXT_cb(const SCPI_argval_t *args)
{
printf("cb DISPlay:TEXT %s\n", args[0].STRING);
}
void cmd_DATA_BLOB_cb(const SCPI_argval_t *args)
{
printf("cb DATA:BLOB <%d>\n", args[0].BLOB_LEN);
}
void cmd_DATA_BLOB_data(const uint8_t *bytes)
{
printf("binary data: \"%s\"\n", bytes);
}
void cmd_USRERR_cb(const SCPI_argval_t *args)
{
(void) args;
printf("cb USRERR - raising user error 10.\n");
scpi_add_error(10, "Custom error message...");
}
void cmd_CHARD_cb(const SCPI_argval_t *args)
{
printf("CHARData cb: %s, arg2 = %d\n", args[0].CHARDATA, args[1].INT);
}
void cmd_ERROR_FALLBACK_cb(const SCPI_argval_t *args)
{
(void) args;
printf("Testing the error fallback feature...\n");
scpi_add_error(-427, NULL);
}
// Command definition (mandatory commands are built-in)
const SCPI_command_t scpi_commands[] = {
{
.levels = {"APPLy", "SINe"},
.params = {SCPI_DT_INT, SCPI_DT_FLOAT, SCPI_DT_FLOAT},
.callback = cmd_APPL_SIN_cb
},
{
.levels = {"DISPlay", "TEXT"},
.params = {SCPI_DT_STRING},
.callback = cmd_DISP_TEXT_cb
},
{
.levels = {"DATA", "BLOB"},
.params = {SCPI_DT_BLOB},
.callback = cmd_DATA_BLOB_cb,
.blob_chunk = 4,
.blob_callback = cmd_DATA_BLOB_data
},
{
.levels = {"USeRERRor"},
.callback = cmd_USRERR_cb
},
{
.levels = {"ERROR_FALLBACK"},
.callback = cmd_ERROR_FALLBACK_cb
},
{
.levels = {"CHARData"},
.params = {SCPI_DT_CHARDATA, SCPI_DT_INT},
.callback = cmd_CHARD_cb
},
{/*END*/}
};