55#include "match_string.h"
65 CLO_REMOVE_COMMUTATION_OFFSET_ONLY,
69char *commandline_option[N_OPTIONS] = {
76 "removeCommutationOffsetOnly",
80 "sddsremoveoffsets [<input-file>] [<output-file>]\n"
81 " [-pipe=[input],[output]] \n"
83 " [-commutationMode=<string>] \n"
84 " [-removeCommutationOffsetOnly] \n"
85 " [-fhead=<value>] \n"
86 " [-majorOrder=row|column] \n"
89 " -columns=<name> Specify data columns to adjust\n"
90 " -verbose Enable verbose output\n"
91 " -pipe=[input],[output] Use pipes for input/output\n"
92 " -majorOrder=row|column Specify output major order\n"
93 " -commutationMode=<string> Commutation mode (a, b, ab1, ab2)\n"
94 " -fhead=<value> Fraction of head rows for offset calculation\n"
95 " -removeCommutationOffsetOnly Remove only commutation offset\n\n"
97 " Removes offset from BPM waveform data. Adjusts data such that the resulting file has a zero average.\n"
98 " Supports commutation modes for specific offset handling strategies.\n"
99 "Program by Louis Emery and Jialun Luo, ANL (" __DATE__
" " __TIME__
", SVN revision: " SVN_VERSION
")\n";
101static long commutation_modes = 4;
102static char *commutation_mode_list[4] = {
"a",
"b",
"ab1",
"ab2"};
104static long pattern_length = 4;
106long resolve_column_names(
SDDS_DATASET *sdds_in,
char ***column, int32_t *columns);
108int main(
int argc,
char **argv) {
111 unsigned long major_order_flag;
112 char *input = NULL, *output = NULL;
115 char **input_column = NULL, **output_column = NULL;
119 long commutation_mode = 0;
120 long remove_commutation_offset_only = 0;
121 unsigned long pipe_flags = 0;
122 long tmpfile_used = 0, no_warnings = 0;
123 short column_major_order = -1;
124 double *data, offset1 = 0, offset2 = 0, ave_offset = 0;
128 argc =
scanargs(&s_arg, argc, argv);
132 for (i_arg = 1; i_arg < argc; i_arg++) {
133 if (s_arg[i_arg].arg_type == OPTION) {
134 switch (
match_string(s_arg[i_arg].list[0], commandline_option, N_OPTIONS, UNIQUE_MATCH)) {
135 case CLO_MAJOR_ORDER:
136 major_order_flag = 0;
137 s_arg[i_arg].n_items--;
138 if (s_arg[i_arg].n_items > 0 && (!
scanItemList(&major_order_flag, s_arg[i_arg].list + 1, &s_arg[i_arg].n_items, 0,
"row", -1, NULL, 0, SDDS_ROW_MAJOR_ORDER,
"column", -1, NULL, 0, SDDS_COLUMN_MAJOR_ORDER, NULL)))
139 SDDS_Bomb(
"invalid -majorOrder syntax/values");
140 if (major_order_flag & SDDS_COLUMN_MAJOR_ORDER)
141 column_major_order = 1;
142 else if (major_order_flag & SDDS_ROW_MAJOR_ORDER)
143 column_major_order = 0;
145 case CLO_COMMUTATION_MODE:
146 if (s_arg[i_arg].n_items != 2)
147 bomb(
"invalid -commutationMode syntax", NULL);
148 if ((commutation_mode =
match_string(
str_tolower(s_arg[i_arg].list[1]), commutation_mode_list, commutation_modes, EXACT_MATCH)) < 0)
149 SDDS_Bomb(
"invalid commutationMode given!");
154 case CLO_REMOVE_COMMUTATION_OFFSET_ONLY:
155 remove_commutation_offset_only = 1;
158 if (s_arg[i_arg].n_items != 2)
159 bomb(
"invalid -fhead syntax", NULL);
160 if (sscanf(s_arg[i_arg].list[1],
"%lf", &fhead) != 1 || fhead <= 0 || fhead > 1)
161 fprintf(stderr,
"Error: invalid -fhead syntax\n");
164 if (!
processPipeOption(s_arg[i_arg].list + 1, s_arg[i_arg].n_items - 1, &pipe_flags))
169 SDDS_Bomb(
"only one -columns option may be given");
170 if (s_arg[i_arg].n_items < 2)
172 input_column =
tmalloc(
sizeof(*input_column) * (columns = s_arg[i_arg].n_items - 1));
173 for (i = 0; i < columns; i++)
174 input_column[i] = s_arg[i_arg].list[i + 1];
181 input = s_arg[i_arg].list[0];
183 output = s_arg[i_arg].list[0];
189 processFilenames(
"sddsremoveoffsets", &input, &output, pipe_flags, no_warnings, &tmpfile_used);
192 SDDS_Bomb(
"supply the names of columns for offset removal with the -columns option");
197 if (!resolve_column_names(&sdds_in, &input_column, &columns))
200 SDDS_Bomb(
"no columns selected for offset removal");
205 output_column =
tmalloc(
sizeof(*output_column) * columns);
206 for (i = 0; i < columns; i++)
207 output_column[i] = input_column[i];
209 if (!
SDDS_InitializeOutput(&sdds_out, SDDS_BINARY, 1,
"Remove offset from BPM waveforms",
"Modified data", output) ||
212 if (column_major_order != -1)
213 sdds_out.layout.data_mode.column_major = column_major_order;
215 sdds_out.layout.data_mode.column_major = sdds_in.layout.data_mode.column_major;
223 for (i = 0; i < columns; i++) {
227 long repeat_offset = (data[0] == data[1]) ? 0 : 1;
229 double sum1 = 0, sum2 = 0;
230 unsigned long count1 = 0, count2 = 0;
231 unsigned long half_pattern_length = pattern_length / 2;
232 for (j = 0; j < fhead * rows; j++) {
233 if ((j - repeat_offset) % pattern_length < half_pattern_length) {
241 offset1 = sum1 / count1;
242 offset2 = sum2 / count2;
243 if (!remove_commutation_offset_only) {
244 for (j = 0; j < rows; j++) {
245 if ((j - repeat_offset) % pattern_length < half_pattern_length)
251 ave_offset = (offset1 + offset2) / 2.0;
252 for (j = 0; j < rows; j++) {
253 if ((j - repeat_offset) % pattern_length < half_pattern_length)
254 data[j] = data[j] - offset1 + ave_offset;
256 data[j] = data[j] - offset2 + ave_offset;
261 printf(
"offset1 = %f \t offset2 = %f\n", offset1, offset2);
263 for (j = 0; j < rows; j++) {
266 double new_mean = new_sum / rows;
267 printf(
"New average: %f\n", new_mean);
288long resolve_column_names(
SDDS_DATASET *sdds_in,
char ***column, int32_t *columns) {
292 for (i = 0; i < *columns; i++) {
SDDS (Self Describing Data Set) Data Types Definitions and Function Prototypes.
int32_t SDDS_InitializeCopy(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source, char *filename, char *filemode)
int32_t SDDS_CopyPage(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source)
int32_t SDDS_SetColumnFromDoubles(SDDS_DATASET *SDDS_dataset, int32_t mode, double *data, int64_t rows,...)
Sets the values for a single data column using double-precision floating-point numbers.
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_WritePage(SDDS_DATASET *SDDS_dataset)
Writes the current data table to the output file.
int32_t SDDS_WriteLayout(SDDS_DATASET *SDDS_dataset)
Writes the SDDS layout header to the output file.
void SDDS_SetError(char *error_text)
Records an error message in the SDDS error stack.
char ** SDDS_GetColumnNames(SDDS_DATASET *SDDS_dataset, int32_t *number)
Retrieves the names of all columns in the SDDS dataset.
void SDDS_PrintErrors(FILE *fp, int32_t mode)
Prints recorded error messages to a specified file stream.
void SDDS_RegisterProgramName(const char *name)
Registers the executable program name for use in error messages.
void SDDS_Bomb(char *message)
Terminates the program after printing an error message and recorded errors.
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.
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.
long replaceFileAndBackUp(char *file, char *replacement)
Replaces a file with a replacement file and creates a backup of the original.
int scanargs(SCANNED_ARG **scanned, int argc, char **argv)
long processPipeOption(char **item, long items, unsigned long *flags)
void processFilenames(char *programName, char **input, char **output, unsigned long pipeFlags, long noWarnings, long *tmpOutputUsed)
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.