scpi: Rephrase length logic in data block reception, comment/group code

Slightly rephrase the SCPI code which parses the responses that carry
(binary) data blocks. Be explicit about NUL termination when parsing the
leading length spec in the response, obsoleting the array initializer.
Add lots of comments and group source code lines to better reflect
what's happening from the protocol's perspective.

Fix the returned error code in the path which reads responses of
excessive length in chunks. The previous implementation detected errors
but always returned code 0 (success).
This commit is contained in:
Gerhard Sittig 2016-12-30 13:27:29 +01:00 committed by Uwe Hermann
parent 904401e8fe
commit 26e8c6a2b2
1 changed files with 27 additions and 8 deletions

View File

@ -743,55 +743,74 @@ SR_PRIV int sr_scpi_get_block(struct sr_scpi_dev_inst *scpi,
{ {
int ret; int ret;
GString* response; GString* response;
char buf[10];
char buf[10] = { 0 };
long llen; long llen;
long datalen; long datalen;
/*
* Assume an initial maximum length, optionally gets adjusted below.
* Prepare a NULL return value for when error paths will be taken.
*/
response = g_string_sized_new(1024); response = g_string_sized_new(1024);
*scpi_response = NULL; *scpi_response = NULL;
/* Get (the first chunk of) the response. */
ret = sr_scpi_get_data(scpi, command, &response); ret = sr_scpi_get_data(scpi, command, &response);
if (ret != SR_OK) { if (ret != SR_OK) {
g_string_free(response, TRUE); g_string_free(response, TRUE);
return ret; return ret;
} }
/*
* SCPI protocol data blocks are preceeded with a length spec.
* The length spec consists of a '#' marker, one digit which
* specifies the character count of the length spec, and the
* respective number of characters which specify the data block's
* length. Raw data bytes follow (thus one must no longer assume
* that the received input stream would be an ASCIIZ string).
*
* Get the data block length, and strip off the length spec from
* the input buffer, leaving just the data bytes.
*/
if (response->str[0] != '#') { if (response->str[0] != '#') {
g_string_free(response, TRUE); g_string_free(response, TRUE);
return SR_ERR_DATA; return SR_ERR_DATA;
} }
buf[0] = response->str[1]; buf[0] = response->str[1];
buf[1] = '\0';
ret = sr_atol(buf, &llen); ret = sr_atol(buf, &llen);
if ((ret != SR_OK) || (llen == 0)) { if ((ret != SR_OK) || (llen == 0)) {
g_string_free(response, TRUE); g_string_free(response, TRUE);
return ret; return ret;
} }
memcpy(buf, &response->str[2], llen); memcpy(buf, &response->str[2], llen);
buf[llen] = '\0';
ret = sr_atol(buf, &datalen); ret = sr_atol(buf, &datalen);
if ((ret != SR_OK) || (datalen == 0)) { if ((ret != SR_OK) || (datalen == 0)) {
g_string_free(response, TRUE); g_string_free(response, TRUE);
return ret; return ret;
} }
// strip header
g_string_erase(response, 0, 2 + llen); g_string_erase(response, 0, 2 + llen);
/*
* If the initially assumed length does not cover the data block
* length, then re-allocate the buffer size to the now known
* length, and keep reading more chunks of response data.
*/
if (response->len < (unsigned long)(datalen)) { if (response->len < (unsigned long)(datalen)) {
int oldlen = response->len; int oldlen = response->len;
g_string_set_size(response, datalen); g_string_set_size(response, datalen);
g_string_set_size(response, oldlen); g_string_set_size(response, oldlen);
} }
while (response->len < (unsigned long)(datalen)) { while (response->len < (unsigned long)(datalen)) {
if (sr_scpi_get_data(scpi, NULL, &response) != SR_OK) { ret = sr_scpi_get_data(scpi, NULL, &response);
if (ret != SR_OK) {
g_string_free(response, TRUE); g_string_free(response, TRUE);
return ret; return ret;
} }
} }
/* Convert received data to byte array. */
*scpi_response = g_byte_array_new_take( *scpi_response = g_byte_array_new_take(
(guint8*)g_string_free(response, FALSE), datalen); (guint8*)g_string_free(response, FALSE), datalen);