45#include "match_string.h"
58static char *option[N_OPTIONS] = {
66 "tek2sdds <inputfile> <outputfile>\n"
67 " -signalname=<name>\n"
68 " [-description=<text>,<contents>]\n"
69 " [-mpllabels=<title>,<topline>]\n"
70 " [-majorOrder=row|column]\n"
72 " -signalname=<name> (required) Name of the signal\n"
73 " -description=<text>,<contents> (optional) Description text and contents\n"
74 " -mpllabels=<title>,<topline> (optional) MPL labels: title and topline\n"
75 " -majorOrder=row|column (optional) Major order of data\n"
77 "This program converts Tektronix ASCII format waveforms to SDDS format.\n"
79 "Program by Michael Borland (" __DATE__
" " __TIME__
", SVN revision: " SVN_VERSION
")\n";
83 char *TEK_name, *SDDS_name, *value_string;
87#define TEK_PARAMETERS 23
104 {
"BYT/NR",
"TEKBytesPerNumber", NULL,
SDDS_LONG},
106 {NULL, NULL, NULL, 0},
109char *TEK_PreambleString =
"WFMPRE ";
110char *TEK_DataMarker =
"CURVE ";
111char *TEK_PointsName =
"NR.PT";
112char *TEK_XIncrementName =
"XINCR";
113char *TEK_XZeroName =
"XZERO";
114char *TEK_XUnitsName =
"XUNIT";
115char *TEK_YUnitsName =
"YUNIT";
116char *TEK_YZeroName =
"YZERO";
117char *TEK_YMultiplierName =
"YMULT";
118char *TEK_EncodingName =
"ENCDG";
119char *TEK_BytesPerNumberName =
"BYT/NR";
120char *TEK_ByteOrderName =
"BYT.OR";
124long GetNextItem(
char *buffer,
long bufsize, FILE *fpi);
126int main(
int argc,
char **argv) {
128 SCANNED_ARG *scanned;
129 long i, i_arg, index, points, bytes_per_number;
131 char *input, *output, buffer[BUFSIZE];
132 char *signal_name, *ptr, *parameter_name;
133 char *mpl_title, *mpl_topline, *descrip_text, *descrip_contents;
136 double xIncrement, xZero, yMultiplier, yZero;
137 char *xUnits, *yUnits;
139 short columnMajorOrder = 0;
140 unsigned long majorOrderFlag;
142 xUnits = yUnits = NULL;
144 argc =
scanargs(&scanned, argc, argv);
149 input = output = signal_name = NULL;
150 mpl_title = mpl_topline = descrip_text = descrip_contents = NULL;
153 for (i_arg = 1; i_arg < argc; i_arg++) {
154 if (scanned[i_arg].arg_type == OPTION) {
157 switch (
match_string(scanned[i_arg].list[0], option, N_OPTIONS, 0)) {
158 case SET_MAJOR_ORDER:
160 scanned[i_arg].n_items--;
161 if (scanned[i_arg].n_items > 0 &&
162 (!
scanItemList(&majorOrderFlag, scanned[i_arg].list + 1, &scanned[i_arg].n_items, 0,
163 "row", -1, NULL, 0, SDDS_ROW_MAJOR_ORDER,
164 "column", -1, NULL, 0, SDDS_COLUMN_MAJOR_ORDER, NULL)))
165 SDDS_Bomb(
"invalid -majorOrder syntax/values");
166 if (majorOrderFlag & SDDS_COLUMN_MAJOR_ORDER)
167 columnMajorOrder = 1;
168 else if (majorOrderFlag & SDDS_ROW_MAJOR_ORDER)
169 columnMajorOrder = 0;
171 case SET_SIGNAL_NAME:
172 if (scanned[i_arg].n_items != 2) {
173 bomb(
"invalid -signalname syntax", USAGE);
175 signal_name = scanned[i_arg].list[1];
177 case SET_DESCRIPTION:
178 if (scanned[i_arg].n_items != 3) {
179 bomb(
"invalid -description syntax", USAGE);
181 descrip_text = scanned[i_arg].list[1];
182 descrip_contents = scanned[i_arg].list[2];
185 if (scanned[i_arg].n_items != 3) {
186 bomb(
"invalid -mpllabels syntax", USAGE);
188 mpl_title = scanned[i_arg].list[1];
189 mpl_topline = scanned[i_arg].list[2];
192 bomb(
"invalid option seen", USAGE);
197 input = scanned[i_arg].list[0];
199 output = scanned[i_arg].list[0];
201 bomb(
"too many filenames", USAGE);
212 if (fread(buffer, 1, strlen(TEK_PreambleString), fpi) != strlen(TEK_PreambleString) ||
213 strncmp(TEK_PreambleString, buffer, strlen(TEK_PreambleString)) != 0)
214 SDDS_Bomb(
"file does not appear to be in Tektronix format");
216 parameter_name = buffer;
217 while ((code = GetNextItem(buffer, BUFSIZE, fpi)) < 3) {
218 if (!(ptr = strchr(buffer,
':')))
219 SDDS_Bomb(
"error parsing input file--missing colon on parameter tag");
221 if (strcmp(TEK_DataMarker, parameter_name) == 0)
224 while (TEK_parameter[index].TEK_name) {
225 if (strcmp(TEK_parameter[index].TEK_name, parameter_name) == 0)
229 if (!TEK_parameter[index].TEK_name) {
230 fprintf(stderr,
"warning: parameter %s is not recognized\n", parameter_name);
233 if (TEK_parameter[index].value_string) {
234 fprintf(stderr,
"error: duplicate entries for parameter %s\n", parameter_name);
239 if (code == 2 || code == 3)
245 if (fread(buffer, 1, strlen(TEK_DataMarker), fpi) != strlen(TEK_DataMarker) ||
246 strncmp(TEK_DataMarker, buffer, strlen(TEK_DataMarker)) != 0)
247 SDDS_Bomb(
"CURVE item missing or not in right place");
251 SDDS_table.layout.data_mode.column_major = columnMajorOrder;
254 while (TEK_parameter[index].TEK_name) {
255 if (!TEK_parameter[index].value_string) {
259 if (strcmp(TEK_parameter[index].TEK_name, TEK_XIncrementName) == 0) {
260 if (sscanf(TEK_parameter[index].value_string,
"%lf", &xIncrement) != 1)
261 SDDS_Bomb(
"unable to scan value for x increment");
262 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_XZeroName) == 0) {
263 if (sscanf(TEK_parameter[index].value_string,
"%lf", &xZero) != 1)
264 SDDS_Bomb(
"unable to scan value for x zero");
265 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_YZeroName) == 0) {
266 if (sscanf(TEK_parameter[index].value_string,
"%lf", &yZero) != 1)
267 SDDS_Bomb(
"unable to scan value for y zero");
268 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_YMultiplierName) == 0) {
269 if (sscanf(TEK_parameter[index].value_string,
"%lf", &yMultiplier) != 1)
270 SDDS_Bomb(
"unable to scan value for y multiplier");
271 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_XUnitsName) == 0) {
272 xUnits = TEK_parameter[index].value_string;
274 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_YUnitsName) == 0) {
275 yUnits = TEK_parameter[index].value_string;
277 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_PointsName) == 0) {
278 if (sscanf(TEK_parameter[index].value_string,
"%ld", &points) != 1)
279 SDDS_Bomb(
"unable to scan value for number of points");
280 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_EncodingName) == 0) {
281 if (strcmp(TEK_parameter[index].value_string,
"ASCII") == 0)
283 else if (strcmp(TEK_parameter[index].value_string,
"BINARY") == 0)
286 SDDS_Bomb(
"data encoding is neither ASCII nor BINARY");
287 }
else if (strcmp(TEK_parameter[index].TEK_name, TEK_BytesPerNumberName) == 0) {
288 if (sscanf(TEK_parameter[index].value_string,
"%ld", &bytes_per_number) != 1)
289 SDDS_Bomb(
"unable to scan value bytes per number");
290 }
else if (strcmp(TEK_parameter[index].value_string, TEK_ByteOrderName) == 0) {
292 if (strcmp(TEK_parameter[index].value_string,
"LSB") != 0) {
296 if (
SDDS_DefineParameter(&SDDS_table, TEK_parameter[index].SDDS_name, NULL, NULL, TEK_parameter[index].TEK_name,
297 NULL, TEK_parameter[index].type, TEK_parameter[index].value_string) < 0)
312 data =
tmalloc(
sizeof(*data) * points);
313 time =
tmalloc(
sizeof(*time) * points);
315 for (i = 0; i < points; i++) {
317 if (!(code = GetNextItem(buffer, BUFSIZE, fpi)))
318 SDDS_Bomb(
"insufficient data in input file");
323 time[i] = xZero + i * xIncrement;
324 if (sscanf(buffer,
"%lf", data + i) != 1)
326 data[i] = yZero + data[i] * yMultiplier;
330 fread(buffer,
sizeof(
char), 4, fpi);
331 for (i = 0; i < points; i++) {
332 if (fread(&sdata,
sizeof(sdata), 1, fpi) != 1) {
333 fprintf(stderr,
"file ends unexpectedly\n");
337 time[i] = xZero + i * xIncrement;
339 data[i] = yZero + data[i] * yMultiplier;
343 if (!
SDDS_SetColumn(&SDDS_table, SDDS_SET_BY_NAME, time, points,
"t") ||
344 !
SDDS_SetColumn(&SDDS_table, SDDS_SET_BY_NAME, data, points, signal_name) ||
345 !SDDS_WriteTable(&SDDS_table) ||
351long GetNextItem(
char *buffer,
long bufsize, FILE *fpi) {
356 while (i < bufsize && (c = getc(fpi)) != EOF) {
357 if (c ==
',' || c ==
';' || c ==
'%')
361 if (c == EOF && i == 0)
370 fprintf(stderr,
"warning: invalid data seen--ignoring\n");
SDDS (Self Describing Data Set) Data Types Definitions and Function Prototypes.
int32_t SDDS_SetColumn(SDDS_DATASET *SDDS_dataset, int32_t mode, void *data, int64_t rows,...)
Sets the values for one data column in the current data table of an SDDS dataset.
int32_t SDDS_InitializeOutput(SDDS_DATASET *SDDS_dataset, int32_t data_mode, int32_t lines_per_row, const char *description, const char *contents, const char *filename)
Initializes the SDDS output dataset.
int32_t SDDS_DefineColumn(SDDS_DATASET *SDDS_dataset, const char *name, const char *symbol, const char *units, const char *description, const char *format_string, int32_t type, int32_t field_length)
Defines a data column within the SDDS dataset.
int32_t SDDS_WriteLayout(SDDS_DATASET *SDDS_dataset)
Writes the SDDS layout header to the output file.
int32_t SDDS_DefineParameter(SDDS_DATASET *SDDS_dataset, const char *name, const char *symbol, const char *units, const char *description, const char *format_string, int32_t type, char *fixed_value)
Defines a data parameter with a fixed string value.
void SDDS_PrintErrors(FILE *fp, int32_t mode)
Prints recorded error messages to a specified file stream.
void SDDS_Bomb(char *message)
Terminates the program after printing an error message and recorded errors.
int32_t SDDS_CopyString(char **target, const char *source)
Copies a source string to a target string with memory allocation.
void SDDS_RemovePadding(char *s)
Removes leading and trailing whitespace from a string.
#define SDDS_STRING
Identifier for the string data type.
#define SDDS_LONG
Identifier for the signed 32-bit integer data type.
#define SDDS_DOUBLE
Identifier for the double data type.
void * tmalloc(uint64_t size_of_block)
Allocates a memory block of the specified size with zero initialization.
void bomb(char *error, char *usage)
Reports error messages to the terminal and aborts the program.
char * delete_chars(char *s, char *t)
Removes all occurrences of characters found in string t from string s.
FILE * fopen_e(char *file, char *open_mode, long mode)
Opens a file with error checking, messages, and aborts.
long match_string(char *string, char **option, long n_options, long mode)
Matches a given string against an array of option strings based on specified modes.
int scanargs(SCANNED_ARG **scanned, int argc, char **argv)
long scanItemList(unsigned long *flags, char **item, long *items, unsigned long mode,...)
Scans a list of items and assigns values based on provided keywords and types.
char * str_tolower(char *s)
Convert a string to lower case.