/* HEADER: ; TITLE: Frankenstein Cross Assemblers; VERSION: 2.0; DESCRIPTION: " Reconfigurable Cross-assembler producing Intel (TM) Hex format object records. "; SYSTEM: UNIX, MS-Dos ; FILENAME: fraosub.c; WARNINGS: "This software is in the public domain. Any prior copyright claims are relinquished. This software is distributed with no warranty whatever. The author takes no responsibility for the consequences of its use." ; SEE-ALSO: frasmain.c; AUTHORS: Mark Zenier; */ /* description output pass utility routines history September 27, 1987 March 15, 1988 release 1.1 WIDTH September 14, 1990 Dosify, 6 char unique names */ #include "config.h" #include "lzoe/lzoe.h" #include "file/file.h" #include #include "frasmdat.h" #include "fragcon.h" #include "as1600_types.h" #include "protos.h" #include "intermed.h" #include "intvec.h" #include "icart/icartrom.h" #include "collect.h" static int source_offset = 32; static int num_hex_source = 4; static int num_hex_per_line = 8; extern FILE *smapf; extern char *srcmapfn; static const char *currentfnm = NULL; static int linenumber = 0; int listlineno = 0; static char *lineLbuff = NULL; static int lineLflag = FALSE; static int show_noncode_lines = TRUE; static int show_coded_lines = TRUE; #define LISTMODESTK (256) static int listmodestk[LISTMODESTK]; static unsigned short *outrs16 = NULL; static size_t outrs16_alloc = 0; static size_t nextresult; static int genlocctr, resultloc; static unsigned mode_set = 0, mode_clr = 0, type_flag = 0; static const char *oeptr; static const char *over_file = NULL; static int over_line = 0; #define MAXIMPWID 24 static int widthmask[MAXIMPWID+1] = { /* 0 */ 1L, /* 1 */ 1L, /* 2 */ (1L << 2 ) - 1, /* 3 */ (1L << 3 ) - 1, /* 4 */ (1L << 4 ) - 1, /* 5 */ (1L << 5 ) - 1, /* 6 */ (1L << 6 ) - 1, /* 7 */ (1L << 7 ) - 1, /* 8 */ (1L << 8 ) - 1, /* 9 */ (1L << 9 ) - 1, /* 10 */ (1L << 10 ) - 1, /* 11 */ (1L << 11 ) - 1, /* 12 */ (1L << 12 ) - 1, /* 13 */ (1L << 13 ) - 1, /* 14 */ (1L << 14 ) - 1, /* 15 */ (1L << 15 ) - 1, /* 16 */ (1L << 16 ) - 1, /* 17 */ (1L << 17 ) - 1, /* 18 */ (1L << 18 ) - 1, /* 19 */ (1L << 19 ) - 1, /* 20 */ (1L << 20 ) - 1, /* 21 */ (1L << 21 ) - 1, /* 22 */ (1L << 22 ) - 1, /* 23 */ (1L << 23 ) - 1, /* 24 */ (1L << 24 ) - 1 }; static int count_nl(const char *s) { const char *nl = s; int cnt = 0; while ((nl = strchr(nl, '\n')) != NULL) { cnt++; nl++; } return cnt; } static void parsemode(const char *modestr) { int action = 0; unsigned set = 0, clr = 0, bit; while (*modestr) { while (*modestr && *modestr == ',') modestr++; bit = action = 0; do { while (*modestr && strchr(" \t\n\r", *modestr)) modestr++; if (!*modestr) break; if (!action) { action = *modestr++; continue; } switch (*modestr++) { case 'R': case 'r': bit |= ICARTROM_READ; break; case 'W': case 'w': bit |= ICARTROM_WRITE; break; case 'N': case 'n': bit |= ICARTROM_NARROW; break; case 'B': case 'b': bit |= ICARTROM_BANKSW; break; case '-': case '+': case '=': frp2error("Mode syntax: Action char where mode " "char expected"); break; default: frp2error("Mode syntax: Unknown mode character"); break; } } while (*modestr && *modestr != ','); if (!action) break; switch (action) { case '+': { set |= bit; clr &= ~bit; break; } case '-': { set &= ~bit; clr |= bit; break; } case '=': { set = bit; clr = ~bit; break; } case 'R' : case 'W' : case 'B' : case 'N': case 'r' : case 'w' : case 'b' : case 'n': frp2error("Mode syntax: Missing action character"); break; default: frp2error("Mode syntax: Unknown action character"); break; } } mode_set = set & 0xF; mode_clr = clr & 0xF; } static void markrange(int startaddr, int endaddr) { const char *err; if ((err = collect_addseg( NULL, startaddr, endaddr - startaddr + 1, currpag, mode_set, mode_clr )) != NULL) frp2error(err); } void outphase(void) /* description process all the lines in the intermediate file globals the output expression pointer line number file name the binary output array and counts */ { irec_union *rec; if (srcmapfn) sm_outpath(); while ((rec = pass2_next_rec()) != NULL) { linenumber = rec->irec.line; if (rec->irec.type == REC_LIST_LINE) { if (listflag) flushlisthex(); if (lineLbuff) free(lineLbuff); lineLbuff = strdup(rec->string.string); lineLflag = show_coded_lines; } switch (rec->irec.type) { case REC_LIST_MODE: //'N': { int i, lm; lm = rec->list_mode.mode; if (lm == LIST_PREV) { for (i = LISTMODESTK - 1; i > 0; i--) listmodestk[i] = listmodestk[i - 1]; lm = listmodestk[LISTMODESTK-1]; } else { for (i = 1; i < LISTMODESTK - 1; i++) listmodestk[i] = listmodestk[i + 1]; listmodestk[0] = 0; listmodestk[LISTMODESTK-1] = lm; } switch (lm) { case LIST_ON: lineLflag = show_noncode_lines; show_coded_lines = TRUE; show_noncode_lines = TRUE; break; case LIST_CODE: lineLflag = FALSE; show_coded_lines = TRUE; show_noncode_lines = FALSE; break; case LIST_OFF: lineLflag = FALSE; show_coded_lines = FALSE; show_noncode_lines = FALSE; break; } break; } case REC_ERROR: //'E': /* error */ { if(listflag) flushsourceline(); fputs(rec->string.string, loutf); fputc('\n', loutf); listlineno += 1 + count_nl(rec->string.string); break; } case REC_LIST_LINE: //'L': /* listing */ { break; } case REC_SET_EQU: { if (listflag && show_noncode_lines) { char hbuf[12]; sprintf(hbuf, "0x%X", rec->set_equ.value); fprintf(loutf, "%-*.*s", source_offset, source_offset, hbuf); if(lineLflag) { fputs(lineLbuff, loutf); lineLflag = FALSE; listlineno += count_nl(lineLbuff); } else { fputc('\n', loutf); listlineno++; } } else lineLflag = FALSE; break; } case REC_COMMENT: //'C': /* comment / uncounted listing */ { if (listflag && show_noncode_lines) { fprintf(loutf,"%-*.*s", source_offset, source_offset, rec->string.string); if(lineLflag) { fputs(lineLbuff, loutf); /* already has newline */ lineLflag = FALSE; listlineno += count_nl(lineLbuff); } else { fputc('\n', loutf); listlineno++; } } else lineLflag = FALSE; break; } case REC_USER_COMMENT: //'S': /* user comment */ { /* User comments always get printed in the listing, but the code that generates the comment never does. */ if(listflag) { fprintf(loutf,"%s\n", rec->string.string); listlineno += 1 + count_nl(rec->string.string); lineLflag = FALSE; } else lineLflag = FALSE; break; } case REC_LOC_SET: //'P': /* location set */ { currseg = rec->loc_set.seg; currpag = rec->loc_set.pag; locctr = rec->loc_set.loc; genlocctr = rec->loc_set.loc; type_flag = rec->loc_set.type; parsemode (rec->loc_set.mode); break; } case REC_RESERVE_RANGE: //'R': /* reserve range */ { markrange(genlocctr, rec->reserve_range.endaddr); break; } case REC_MARK_RANGE: //'M': /* mark range with mode */ { unsigned old_set = mode_set, old_clr = mode_clr; parsemode(rec->mark_range.mode); markrange(rec->mark_range.lo, rec->mark_range.hi); mode_set = old_set; mode_clr = old_clr; break; } case REC_DATA_BLOCK: //'D': /* data */ { oeptr = rec->string.string; nextresult = 0; resultloc = genlocctr; outeval(); if (listflag && (show_coded_lines || show_noncode_lines)) { listhex(); } outhexblock(); break; } case REC_FILE_START: case REC_FILE_EXIT: { currentfnm = rec->string.string; over_file = NULL; over_line = 0; break; } case REC_CFGVAR_INT: { collect_cfg_var(rec->cfgvar_int.var, NULL, rec->cfgvar_int.value); break; } case REC_CFGVAR_STR: { collect_cfg_var(rec->cfgvar_str.var, rec->cfgvar_str.value, 0); break; } case REC_SRCFILE_OVER: { over_file = rec->srcfile_over.file; over_line = rec->srcfile_over.line; break; } case REC_LISTING_COLUMN: { num_hex_source = rec->listing_column.hex_source; num_hex_per_line = rec->listing_column.hex_no_src; source_offset = rec->listing_column.source_col; break; } case REC_OVERWRITE: { collect_set_overwrite_flags(rec->overwrite.err_if_overwritten, rec->overwrite.force_overwrite); break; } default: { frp2error("unknown intermediate file command"); break; } } pass2_release_rec(rec); } flushhex(); if(listflag) flushlisthex(); if (srcmapfn) sm_flush(); } static void push_outresult(const unsigned short val) { if (nextresult >= outrs16_alloc) { outrs16_alloc = outrs16_alloc ? outrs16_alloc << 1 : 256; outrs16 = REALLOC(outrs16, unsigned short, outrs16_alloc); if (!outrs16) { fprintf(stderr, "Out of memory allocating result buffer\n"); exit(1); } } outrs16[nextresult++] = val; } void outeval(void) /* description convert the polish form character string in the intermediate file 'D' line to binary values in the output result array. globals the output expression pointer the output result array */ { register int etop = 0; int offset = 0; estkm1p = &estk[0]; while( *oeptr != '\0') { switch(*oeptr) { case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': etop = safe_shls32(etop, 4) + ((*oeptr) - '0'); break; case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': etop = safe_shls32(etop, 4) + ((*oeptr) - 'a' + 10); break; case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': etop = safe_shls32(etop, 4) + ((*oeptr) - 'A' + 10); break; #define FRAERR frp2error #include "fraeuni.h" #include "fraebin.h" case IFC_CLASSIFY: { frp2warn("Unexpected IFC_CLASSIFY in stage 2"); etop = 0; break; } case IFC_SYMB: { struct symel *tsy; tsy = symbindex[etop]; if(tsy->seg <= 0) { frp2undef(tsy); etop = 0; } else { if(tsy->seg == SSG_SET) { frp2warn( "forward reference to SET symbol"); } etop = tsy->value; } } break; case IFC_CURRLOC: etop = genlocctr + offset; break; case IFC_PROGCTR: etop = genlocctr; break; case IFC_DUP: if(estkm1p >= &estk[PESTKDEPTH-1]) { frp2error("expression stack overflow"); } else { (++estkm1p)->v = etop; } break; case IFC_LOAD: if(estkm1p >= &estk[PESTKDEPTH-1]) { frp2error("expression stack overflow"); } else { (++estkm1p)->v = etop; } etop = 0; break; case IFC_CLR: etop = 0; break; case IFC_CLRALL: etop = 0; estkm1p = &estk[0]; break; case IFC_POP: etop = (estkm1p--)->v; break; case IFC_TESTERR: if(etop) { frp2error("expression fails validity test"); } break; case IFC_SWIDTH: if( etop > 0 && etop <= MAXIMPWID) { if( estkm1p->v < -(widthmask[etop-1]+1) || estkm1p->v > widthmask[etop-1] ) { frp2error("expression exceeds available field width"); } etop = ((estkm1p--)->v) & widthmask[etop]; } else { frp2error("unimplemented width"); } break; case IFC_WIDTH: if( etop > 0 && etop <= MAXIMPWID) { if( estkm1p->v < -(widthmask[etop-1]+1) || estkm1p->v > widthmask[etop] ) { frp2error("expression exceeds available field width"); } etop = ((estkm1p--)->v) & widthmask[etop]; } else { frp2error("unimplemented width"); } break; case IFC_IWIDTH: if( etop > 0 && etop <= MAXIMPWID) { unsigned sign_check = estkm1p->v & 0xFFFF8000; if (!(etop == 16 && (sign_check == 0xFFFF8000 || !sign_check)) && (estkm1p->v < 0 || estkm1p->v > widthmask[etop])) { frp2error("expression exceeds available field width"); } etop = ((estkm1p--)->v) & widthmask[etop]; } else { frp2error("unimplemented width"); } break; case IFC_EMU8: if (etop >= -128 && etop <= 255 ) { push_outresult(etop & 0xFF); } else { push_outresult(0); frp2error("expression exceeds available field width"); } offset++; etop = 0; break; case IFC_EMS7: if (etop >= -128 && etop <= 127 ) { push_outresult((etop & 0x7F) | (etop & 0x80 ? -0x80 : 0)); } else { push_outresult(0); frp2error("expression exceeds available field width"); } offset++; etop = 0; break; case IFC_EM16: if(etop >= -32768L && etop <= 65535L) { push_outresult(etop); } else { push_outresult(0); frp2error("expression exceeds available field width"); } offset++; etop = 0; break; case IFC_EMBR16: if(etop >= -32768L && etop <= 65535L) { push_outresult((0x00FF & (etop >> 8)) | (0xFF00 & (etop << 8))); } else { push_outresult(0); frp2error("expression exceeds available field width"); } offset++; etop = 0; break; default: break; } oeptr++; } genlocctr += offset; } static int lhaddr, lhnextaddr; static int lhnew, lhnext = 0; static int listbuffhex_size = 0; static unsigned short *listbuffhex = NULL; void flushlisthex(void) /* description output the residue of the hexidecimal values for the previous assembler statement. globals the new hex list flag */ { listouthex(); lhnew = TRUE; } void listhex(void) /* description buffer the output result to block the hexadecimal listing on the output file to NUMHEXPERL bytes per listing line. globals The output result array and count the hex line buffer and counts */ { register size_t cht; register int inhaddr = resultloc; if (lhnew) { lhaddr = lhnextaddr = resultloc; lhnew = FALSE; } for (cht = 0; cht != nextresult; cht++) { if(lhnextaddr != inhaddr || lhnext >= (lineLflag ? num_hex_source : num_hex_per_line ) ) { listouthex(); lhaddr = lhnextaddr = inhaddr; } if (lhnext >= listbuffhex_size) { listbuffhex_size = listbuffhex_size ? listbuffhex_size << 1 : 256; listbuffhex = REALLOC(listbuffhex, unsigned short, listbuffhex_size); if (!listbuffhex) { fprintf(stderr, "Out of memory allocating listbuffhex\n"); exit(1); } } listbuffhex[lhnext++] = outrs16[cht]; lhnextaddr++; inhaddr++; } } void listouthex(void) /* description print a line of hexidecimal on the listing globals the hex listing buffer */ { register int cn; int cols = 0; /* initially, no text on the line */ if (lhnext > 0) { cols = 7; /* if we have data, then we printed an address */ if (currpag >= 0) fprintf(loutf, "%.4X:%.1X ", 0xFFFF & (int)lhaddr, currpag); else fprintf(loutf, "%.4X ", 0xFFFF & (int)lhaddr); for (cn = 0; cn < lhnext; cn++) { cols += 5; fprintf(loutf, "%.4X ", listbuffhex[cn]); } if (!lineLflag) { listlineno++; fputc('\n', loutf); } } else { if (!show_noncode_lines) lineLflag = FALSE; } if (lineLflag) { if(lineLbuff[0] != '\n') { int c = source_offset > cols ? source_offset - cols : 0; fprintf(loutf,"%*c", c, ' '); fputs(lineLbuff, loutf); listlineno += count_nl(lineLbuff); lineLflag = FALSE; } else { fputc('\n', loutf); listlineno++; } } lhnext = 0; } static const char *sm_last_file = NULL; static int sm_start = -2, sm_end = -2, sm_line = -2, sm_list = -2; static unsigned int sm_type = ~2; void sm_outpath(void) { path_t *path = as1600_search_path; #ifndef NO_GETCWD char *cwdbuf = (char *)malloc(65536); if (cwdbuf) { char *cwd = getcwd(cwdbuf, 65536); if (cwd) fprintf(smapf, "CWD %s\n", cwd); free(cwdbuf); } #endif while (path) { fprintf(smapf, "PATH %s\n", path->name); path = path->next; } if (listflag) fprintf(smapf, "LISTING %s\n", loutfn); } void sm_flush() { const char *dispfile = over_file ? over_file : currentfnm; if (sm_start > 0) fprintf(smapf, "%4X %4X %2X %d %d\n", sm_start, sm_end, sm_type, sm_line, sm_list); if (dispfile != sm_last_file) { fprintf(smapf, "FILE %s\n", dispfile); sm_last_file = dispfile; } sm_start = sm_end = sm_line = -2; } void sm_outrange(int lo, int hi) { int list = show_coded_lines ? listlineno : 0; const char *dispfile = over_file ? over_file : currentfnm; int displine = over_file ? over_line : linenumber; int displist = over_file ? 0 : list; if (dispfile != sm_last_file || lo != sm_end + 1 || linenumber != sm_line || list != sm_list || type_flag != sm_type) { sm_flush(); sm_line = displine; sm_start = lo; sm_end = hi; sm_type = type_flag; sm_list = displist; return; } sm_end = hi; return; } static int nextoutaddr, blockaddr; void outhexblock(void) /* description buffer the output result to group adjacent output data into longer lines. globals the output result array the intel hex line buffer */ { const char *err; blockaddr = resultloc; nextoutaddr = blockaddr + currseg; if (nextoutaddr > 0xFFFF) { frp2error("Address overflow (pass 2)\n"); return; } if ((err = collect_addseg( outrs16, nextoutaddr, nextresult, currpag, mode_set, mode_clr )) != NULL) frp2error(err); if (srcmapfn) sm_outrange(nextoutaddr, nextoutaddr + nextresult - 1); } void flushhex(void) /* description flush the intel hex line buffer at the end of the second pass globals the intel hex line buffer */ { collect_flush(); } #define UBUFSZ (32) static struct { int line; const char *file; const char *symb; } undef_filt[UBUFSZ]; static unsigned undef_filt_idx = 0; void frp2undef(struct symel *symp) /* description second pass - print undefined symbol error message on the output listing device. If the the listing flag is false, the output device is the standard output, and the message format is different. parameters a pointer to a symbol table element globals the count of errors */ { int i; if(listflag) flushsourceline(); /* filter out redundant errors that can happen due to multiple refs within the same polish expression, or multiple references on the same line */ for (i = 0; i < UBUFSZ; i++) { if (!undef_filt[i].file) continue; if (undef_filt[i].line == linenumber && undef_filt[i].file == currentfnm && undef_filt[i].symb == symp->symstr) return; } /* insert this error in the filter */ undef_filt[undef_filt_idx].file = currentfnm; undef_filt[undef_filt_idx].line = linenumber; undef_filt[undef_filt_idx].symb = symp->symstr; undef_filt_idx = (undef_filt_idx + 1) % UBUFSZ; /* Now print the error. */ if ((symp->flags & SFLAG_ARRAY) == 0) { fprintf(loutf, "%s:%d: ERROR - undefined symbol %s\n", currentfnm, linenumber, symp->symstr); listlineno++; } else { const char *symstr = symp->symstr; int first = 1; do { const char *aidx_str = strchr(symstr, 0x01); unsigned aidx_val = 0; if (aidx_str && (aidx_str[1] & 0x80) == 0x80 && (aidx_str[2] & 0x80) == 0x80 && (aidx_str[3] & 0x80) == 0x80 && (aidx_str[4] & 0x80) == 0x80 && (aidx_str[5] & 0x80) == 0x80) { aidx_val = ((aidx_str[1] & 0x7F) << 0) | ((aidx_str[2] & 0x7F) << 7) | ((aidx_str[3] & 0x7F) << 14) | ((aidx_str[4] & 0x7F) << 21) | ((aidx_str[5] & 0x0F) << 28); symstr = aidx_str + 6; } else { if (!first) { fprintf(loutf, "\n"); listlineno++; } fprintf(loutf, "%s:%d: INTERNAL ERROR IN ARRAY ENCODING", currentfnm, linenumber); break; } if (first) fprintf(loutf, "%s:%d: ERROR - undefined array index %.*s", currentfnm, linenumber, (int)(aidx_str - symp->symstr), symp->symstr); first = 0; fprintf(loutf, "[%d]", aidx_val); } while (*symstr); fprintf(loutf, "\n"); listlineno++; } errorcnt++; } void frp2warn(const char *str) /* description second pass - print a warning message on the listing file, varying the format for console messages. parameters the message globals the count of warnings */ { if(listflag) flushsourceline(); fprintf(loutf, "%s:%d: WARNING - %s\n", currentfnm, linenumber, str); listlineno += 1 + count_nl(str); warncnt++; } void frp2error(const char *str) /* description second pass - print a message on the listing file parameters message globals count of errors */ { flushsourceline(); fprintf(loutf, "%s:%d: ERROR - %s\n", currentfnm, linenumber, str); listlineno += 1 + count_nl(str); errorcnt++; } void flushsourceline(void) /* description flush listing line buffer before an error for that line is printed */ { if(listflag && lineLflag && show_noncode_lines) { fputs("\t\t\t", loutf); fputs(lineLbuff, loutf); listlineno += count_nl(lineLbuff); } lineLflag = FALSE; } typedef struct tz_info { int pm; int hour; int min; int tot_minutes; } tz_info; LOCAL tz_info compute_tz(const struct tm *const t, const struct tm *gmt) { tz_info tz = { 0, 0, 0, 0 }; int hr_diff = ((t->tm_hour - gmt->tm_hour + 12) % 24) - 12; int ah_diff = hr_diff < 0 ? -hr_diff : hr_diff; int mn_diff = hr_diff < 0 ? (gmt->tm_min - t->tm_min) : (t->tm_min - gmt->tm_min); int abs_minutes; if (mn_diff < 0) { mn_diff += 60; if (ah_diff > 0) ah_diff--; } abs_minutes = ah_diff * 60 + mn_diff; tz.pm = hr_diff >= 0 ? '+' : '-'; tz.hour = ah_diff; tz.min = mn_diff; tz.tot_minutes = hr_diff < 0 ? -abs_minutes : abs_minutes; return tz; } /* Format time for TODAY_STR_xxx, w/NUL termination. */ /* Returns size of the output. */ int format_time_string(const struct tm *const t, const struct tm *const gmt, const char *const fmt, char *const bufbeg, const int space) { const int y_hi = t->tm_year / 100 + 19; const int y_lo = t->tm_year % 100; const int mon = t->tm_mon + 1; const int hr12 = t->tm_hour == 0 ? 12 : t->tm_hour > 12 ? t->tm_hour - 12 : t->tm_hour; const int ampm = t->tm_hour < 12 ? 'A' : 'P'; char * buf = bufbeg; char *const bufend = bufbeg + space - 6; /* room for +hhmm and \0 */ int i = 0; while (fmt[i] && buf < bufend) { const int c0 = fmt[i + 0]; const int c1 = fmt[i + 1]; if (c0 != '%') { *buf++ = c0; i++; continue; } /* Saw %; commit to consuming two chars */ i += 2; # define TWODIG(x) \ *buf++ = ((x) / 10) % 10 + '0'; \ *buf++ = (x) % 10 + '0'; switch (c1) { case 'Y': { TWODIG(y_hi); TWODIG(y_lo); continue; } case 'y': { TWODIG(y_lo); continue; } case 'm': { TWODIG(mon); continue; } case 'd': { TWODIG(t->tm_mday); continue; } case 'H': { TWODIG(t->tm_hour); continue; } case 'M': { TWODIG(t->tm_min); continue; } case 'S': { TWODIG(t->tm_sec); continue; } case 'I': { TWODIG(hr12); continue; } case 'p': { *buf++ = ampm; *buf++ = 'M'; continue; } case '%': { *buf++ = '%'; continue; } case 'z': { tz_info tz = compute_tz(t, gmt); *buf++ = tz.pm; TWODIG(tz.hour); TWODIG(tz.min); continue; } case 0: { fraerror("% at end of format string in TODAY_STR_xxx"); return 0; } default: { fraerror("unknown format character in TODAY_STR_xxx"); return 0; } } # undef TWODIG } *buf++ = 0; return buf - bufbeg; } /* Unpack time values as constant expressions for TODAY_VAL_xxx */ /* Returns an intvec with the unpacked exprs */ intvec_t *unpack_time_exprs(const struct tm *const t, const struct tm *const gmt, const char *const fmt) { const int year = t->tm_year + 1900; const int y_lo = t->tm_year % 100; const int mon = t->tm_mon + 1; const int hr12 = t->tm_hour == 0 ? 12 : t->tm_hour > 12 ? t->tm_hour - 12 : t->tm_hour; const int ampm = t->tm_hour >= 12; intvec_t *const RESTRICT iv = intvec_new(); int i = 0; while (fmt[i]) { const int c0 = fmt[i + 0]; const int c1 = fmt[i + 1]; if (c0 != '%') { i++; continue; } /* Saw %; commit to consuming two chars */ i += 2; # define CST_EXPR(x) \ intvec_push(iv, exprnode(PCCASE_CONS,0,IGP_CONSTANT,0,(x),SYMNULL)) switch (c1) { case 'Y': { CST_EXPR(year); continue; } case 'y': { CST_EXPR(y_lo); continue; } case 'm': { CST_EXPR(mon); continue; } case 'd': { CST_EXPR(t->tm_mday); continue; } case 'H': { CST_EXPR(t->tm_hour); continue; } case 'M': { CST_EXPR(t->tm_min); continue; } case 'S': { CST_EXPR(t->tm_sec); continue; } case 'I': { CST_EXPR(hr12); continue; } case 'p': { CST_EXPR(ampm); continue; } case '%': { /* ignore */ continue; } case 'z': { tz_info tz = compute_tz(t, gmt); CST_EXPR(tz.tot_minutes); continue; } case 0: { fraerror("% at end of format string in TODAY_VAL_xxx"); iv->len = 0; return iv; } default: { fraerror("unknown format character in TODAY_VAL_xxx"); iv->len = 0; return iv; } } } return iv; }