| iMatix home page | << | < | > | >> |
SFL Version 2.11 |
#include "sflexdr.h" int exdr_write (byte *buffer, const char *format, ...)
Accepts a list of data items, prepares them according to the format string, and stores the result in the buffer. The buffer may be transmitted to another system, then decoded using exdr_read. Assumes nothing about system word sizes, etc. However, does assume that both systems use ASCII. If the buffer address is NULL, does not store the data items, but counts the effective size and returns that. The null-terminated format string can contain these sequences:
c | Single character value (may be multibyte) |
b | Single byte value |
w,d | Double byte value (16-bit short integer) |
q,l | Quad-byte value (32-bit long integer) |
s | Null-terminated string (address of string), or NULL |
B | Bool value (16-bit short integer) |
m | Memory descriptor size (16-bit integer), >= 0 |
M | Memory descriptor body (pointer to block), or NULL |
h | Huge memory descriptor size (31-bit integer), >= 0 |
H | Huge memory descriptor body (pointer), or NULL |
{ va_list argptr; /* Argument list pointer */ byte byte_value, /* Byte value from arguments */ *target, /* Pointer into target buffer */ *block; /* Source block for 'M' type */ char *string; /* Source string for 's' type */ dbyte dbyte_value; /* Network format dbyte value */ qbyte memory_size = 0, /* Memory descriptor size value */ qbyte_value; /* Network format qbyte value */ ASSERT (format); va_start (argptr, format); /* Start variable arguments list */ target = buffer; while (*format) { switch (*format++) { case 'c': /* Character */ case 'b': /* Single byte */ byte_value = (byte) va_arg (argptr, int); if (buffer) *(byte *) target = byte_value; target += 1; break; case 'd': /* Signed short integer */ case 'w': /* Unsigned short integer */ case 'B': /* Bool */ dbyte_value = htons ((short) va_arg (argptr, int)); if (buffer) { *(byte *) target++ = *((byte *) &dbyte_value); *(byte *) target++ = *((byte *) &dbyte_value + 1); } else target += 2; break; case 'l': /* Signed long (32-bit) */ case 'q': /* 4-byte unsigned value */ qbyte_value = htonl (va_arg (argptr, qbyte)); if (buffer) { *(byte *) target++ = *((byte *) &qbyte_value); *(byte *) target++ = *((byte *) &qbyte_value + 1); *(byte *) target++ = *((byte *) &qbyte_value + 2); *(byte *) target++ = *((byte *) &qbyte_value + 3); } else target += 4; break; case 's': /* Null-terminated string */ string = va_arg (argptr, char *); if (string) { if (buffer) strcpy ((char *) target, string); target += strlen (string) + 1; } else /* Store NULL as single null byte */ { if (buffer) *(byte *) target++ = 0; else target += 1; } break; case 'm': /* Memory descriptor size */ memory_size = va_arg (argptr, int); dbyte_value = htons ((dbyte) memory_size); if (buffer) { *(byte *) target++ = *((byte *) &dbyte_value); *(byte *) target++ = *((byte *) &dbyte_value + 1); } else target += 2; break; case 'M': /* Memory descriptor body */ block = va_arg (argptr, byte *); if (block) { if (buffer) memcpy (target, block, (size_t) memory_size); target += (size_t) memory_size; } else ASSERT (memory_size == 0); break; case 'h': /* Huge memory descriptor size */ memory_size = va_arg (argptr, qbyte); qbyte_value = htonl (memory_size); if (buffer) { *(byte *) target++ = *((byte *) &qbyte_value); *(byte *) target++ = *((byte *) &qbyte_value + 1); *(byte *) target++ = *((byte *) &qbyte_value + 2); *(byte *) target++ = *((byte *) &qbyte_value + 3); } else target += 4; break; case 'H': /* Huge memory descriptor body */ block = va_arg (argptr, byte *); if (block) { if (buffer) memcpy (target, block, (size_t) memory_size); target += (size_t) memory_size; } else ASSERT (memory_size == 0); break; } } va_end (argptr); /* End variable arguments list */ return ((int) (target - buffer)); }
| << | < | > | >> | Copyright © 1996-2000 iMatix Corporation |