diff options
| author | Mattias Andrée <maandree@kth.se> | 2016-07-15 16:53:21 +0200 | 
|---|---|---|
| committer | Mattias Andrée <maandree@kth.se> | 2016-07-15 16:53:21 +0200 | 
| commit | 3665122b41901328b1c74d1e797bf9d0d5328f3a (patch) | |
| tree | a16aa89f64043625b6874df52031a2a90cf2637f | |
| parent | Implement libcoopgamma_set_gamma_recv (diff) | |
| download | libcoopgamma-3665122b41901328b1c74d1e797bf9d0d5328f3a.tar.gz libcoopgamma-3665122b41901328b1c74d1e797bf9d0d5328f3a.tar.bz2 libcoopgamma-3665122b41901328b1c74d1e797bf9d0d5328f3a.tar.xz | |
m + Implement check_error, next_payload, and next_header
Signed-off-by: Mattias Andrée <maandree@kth.se>
Diffstat (limited to '')
| -rw-r--r-- | src/libcoopgamma.c | 127 | ||||
| -rw-r--r-- | src/libcoopgamma.h | 12 | 
2 files changed, 124 insertions, 15 deletions
| diff --git a/src/libcoopgamma.c b/src/libcoopgamma.c index 3b2f08e..d5c25c7 100644 --- a/src/libcoopgamma.c +++ b/src/libcoopgamma.c @@ -764,6 +764,7 @@ int libcoopgamma_context_initialise(libcoopgamma_context_t* restrict this)    this->inbound_head = 0;    this->inbound_tail = 0;    this->inbound_size = 0; +  this->length = 0;    return 0;  } @@ -812,6 +813,7 @@ size_t libcoopgamma_context_marshal(const libcoopgamma_context_t* restrict this,    marshal_buffer(this->outbound + this->outbound_head, this->outbound_head - this->outbound_tail);    marshal_prim(this->inbound_head - this->inbound_tail, size_t);    marshal_buffer(this->inbound + this->inbound_head, this->inbound_head - this->inbound_tail); +  marshal_prim(this->length, size_t);    MARSHAL_EPILOGUE;  } @@ -844,6 +846,7 @@ int libcoopgamma_context_unmarshal(libcoopgamma_context_t* restrict this,    unmarshal_buffer(this->outbound, this->outbound_head);    this->inbound_size = this->inbound_head;    unmarshal_buffer(this->inbound, this->inbound_head); +  unmarshal_prim(this->length, size_t);    UNMARSHAL_EPILOGUE;  } @@ -1388,13 +1391,47 @@ static int send_message(libcoopgamma_context_t* restrict ctx, char* msg, size_t  } -static char* next_line(libcoopgamma_context_t* restrict ctx) +/** + * Get the next header of the inbound message + *  + * All headers must be read before the payload is read + *  + * @param   ctx  The state of the library, must be connected + * @return       The next header line, can never be `NULL`, + *               the empty string marks the end of the headers. + *               This memory segment must not be freed. + */ +static char* next_header(libcoopgamma_context_t* restrict ctx)  { +  char* rc = ctx->inbound + ctx->inbound_tail; +  ctx->inbound_tail += strlen(ctx->inbound) + 1; +  return rc;  } +/** + * Get the payload of the inbound message + *  + * Calling this function marks that the inbound message + * has been fully ready. You must call this function + * even if you do not expect a payload + *  + * @param   ctx  The state of the library, must be connected + * @param   n    Output parameter for the size of the payload + * @return       The payload (not NUL-terminated), `NULL` if + *               there is no payload. Failure is impossible. + *               This memory segment must not be freed. + */  static char* next_payload(libcoopgamma_context_t* restrict ctx, size_t* n)  { +  if ((*n = ctx->length)) +    { +      char* rc = ctx->inbound + ctx->inbound_tail; +      ctx->inbound_tail += *n; +      return rc; +    } +  else +    return NULL;  } @@ -1407,11 +1444,81 @@ static char* next_payload(libcoopgamma_context_t* restrict ctx, size_t* n)   *    * @param   ctx    The state of the library, must be connected   * @param   async  Information about the request - * @return         1 if the server sent an error, 0 on success, -1 on failure. - *                 Information about failure is copied `ctx`. + * @return         1 if the server sent an error (even indicating success), + *                 0 on success, -1 on failure. Information about failure + *                 is copied `ctx`.   */  static int check_error(libcoopgamma_context_t* restrict ctx, libcoopgamma_async_context_t* restrict async)  { +  char temp[3 * sizeof(uint64_t) + 1]; +  size_t old_tail = ctx->inbound_tail; +  char* line; +  char* value; +  int command_ok = 0; +  int have_in_response_to = 0; +  int have_error = 0; +  int bad = 0; +   +  for (;;) +    { +      line = next_header(ctx); +      value = strchr(line, ':') + 2; +      if (!*line) +	break; +      else if (!strcmp(line, "Command: error")) +	command_ok = 1; +      else if (strstr(line, "In response to: ") == line) +	{ +	  uint32_t id = (uint32_t)atol(value); +	  have_in_response_to = 1 + !!have_in_response_to; +	  if (id != async->message_id) +	    bad = 1; +	  else +	    { +	      sprintf(temp, "%" PRIu32, id); +	      if (strcmp(value, temp)) +		bad = 1; +	    } +	} +      else if (strstr(line, "Error: ") == line) +	{ +	  have_error = 1 + !!have_error; +	  ctx->error.server_side = 1; +	  ctx->error.custom = (strstr(value, "custom") == value); +	  if (ctx->error.custom) +	    { +	      if (value[6] == '\0') +		{ +		  ctx->error.number = 0; +		  continue; +		} +	      else if (value[6] != ' ') +		{ +		  bad = 1; +		  continue; +		} +	    } +	  ctx->error.number = (uint64_t)atoll(value); +	  sprintf(temp, "%" PRIu64, ctx->error.number); +	  if (strcmp(value, temp)) +	    bad = 1; +	} +    } +   +  if (command_ok == 0) +    { +      ctx->inbound_tail = old_tail; +      return 0; +    } +   +  if (bad || (have_in_response_to != 1) || (have_error != 1)) +    { +      errno = EBADMSG; +      copy_errno(ctx); +      return -1; +    } +   +  return 1;  } @@ -1468,7 +1575,7 @@ char** libcoopgamma_get_crtcs_recv(libcoopgamma_context_t* restrict ctx,    for (;;)      { -      line = next_line(ctx); +      line = next_header(ctx);        if (!*line)  	break;        else if (!strcmp(line, "Command: crtc-enumeration")) @@ -1582,7 +1689,7 @@ int libcoopgamma_get_gamma_info_recv(libcoopgamma_crtc_info_t* restrict info,    for (;;)      { -      line = next_line(ctx); +      line = next_header(ctx);        value = strchr(line, ':') + 2;        if (!*line)  	break; @@ -1697,8 +1804,7 @@ int libcoopgamma_get_gamma_send(libcoopgamma_filter_query_t* restrict query,   *    * @param   table  Output for the response, must be initialised   * @param   ctx    The state of the library, must be connected - * @param   async  Information about the request, that is needed to - *                 identify and parse the response, is stored here + * @param   async  Information about the request   * @return         Zero on success, -1 on error, in which case `ctx->error`   *                 (rather than `errno`) is read for information about the error   */ @@ -1725,7 +1831,7 @@ int libcoopgamma_get_gamma_recv(libcoopgamma_filter_table_t* restrict table,    for (;;)      { -      line = next_line(ctx); +      line = next_header(ctx);        value = strchr(line, ':') + 2;        if (!*line)  	break; @@ -1924,8 +2030,7 @@ int libcoopgamma_set_gamma_send(libcoopgamma_filter_t* restrict filter, libcoopg   * Apply, update, or remove a gamma ramp adjustment, receive response part   *    * @param   ctx    The state of the library, must be connected - * @param   async  Information about the request, that is needed to - *                 identify and parse the response, is stored here + * @param   async  Information about the request   * @return         Zero on success, -1 on error, in which case `ctx->error`   *                 (rather than `errno`) is read for information about the error   */ @@ -1937,7 +2042,7 @@ int libcoopgamma_set_gamma_recv(libcoopgamma_context_t* restrict ctx,    if (check_error(ctx, async))      return -(ctx->error.custom || ctx->error.number); -  while (*next_line(ctx)); +  while (*next_header(ctx));    (void) next_payload(ctx, &_n);    errno = EBADMSG; diff --git a/src/libcoopgamma.h b/src/libcoopgamma.h index 15636b6..3f9c935 100644 --- a/src/libcoopgamma.h +++ b/src/libcoopgamma.h @@ -666,6 +666,12 @@ typedef struct libcoopgamma_context     */    size_t inbound_size; +  /** +   * The value of 'Length' header in +   * the inbound message +   */ +  size_t length; +    } libcoopgamma_context_t; @@ -1285,8 +1291,7 @@ int libcoopgamma_get_gamma_send(libcoopgamma_filter_query_t* restrict, libcoopga   *    * @param   table  Output for the response, must be initialised   * @param   ctx    The state of the library, must be connected - * @param   async  Information about the request, that is needed to - *                 identify and parse the response, is stored here + * @param   async  Information about the request   * @return         Zero on success, -1 on error, in which case `ctx->error`   *                 (rather than `errno`) is read for information about the error   */ @@ -1313,8 +1318,7 @@ int libcoopgamma_set_gamma_send(libcoopgamma_filter_t* restrict, libcoopgamma_de   * Apply, update, or remove a gamma ramp adjustment, receive response part   *    * @param   ctx    The state of the library, must be connected - * @param   async  Information about the request, that is needed to - *                 identify and parse the response, is stored here + * @param   async  Information about the request   * @return         Zero on success, -1 on error, in which case `ctx->error`   *                 (rather than `errno`) is read for information about the error   */ | 
