/* * Copyright (c) 2013, Kelvin W Sherlock * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #include #include #include namespace { // private... %%{ machine lexer; action addx { value = (value << 4) + digittoint(fc); } ws = [ \t]; value = '0x' xdigit+ @addx | '$' xdigit+ @addx | [A-Za-z_] @{ stringValue.push_back(fc); } [A-Za-z0-9_]+ @{ stringValue.push_back(fc); } ; name = [A-Za-z0-9_]+ @{ name.push_back(fc); } ; main := name ws* '=' ws* value ; write data; }%% bool ParseLine(const char *p, const char *pe, std::map &map) { // trap = number // or trap = [previously defined trap] std::string name; std::string stringValue; uint32_t value; value = 0; const char *eof = pe; int cs; %%write init; %%write exec; if (cs < %%{ write first_final; }%% ) { return false; } // name lookup if (!stringValue.empty()) { auto iter = map.find(stringValue); if (iter == map.end()) { fprintf(stderr, "Undefined trap: %s\n", stringValue.c_str()); return false; } value = iter->second; } // if loading globals, wouldn't need to do this... if (value > 0xafff || value < 0xa000) { fprintf(stderr, "Invalid trap number: $%04x\n", value); return false; } map.emplace(name, (uint16_t)value); return true; } } namespace Debug { void LoadTrapFile(const std::string &path, std::map &map) { FILE *fp; fp = fopen(path.c_str(), "r"); if (!fp) { fprintf(stderr, "Unable to open trap file %s\n", path.c_str()); return; } /* * getline(3) is 2008 posix. it allocates (and resizes as appropriate) * the buffer. * */ char *lineBuffer = NULL; size_t lineSize = 0; for(;;) { char *line; ssize_t length; length = getline(&lineBuffer, &lineSize, fp); if (!length) continue; //? if (length < 0) break; // eof or error. line = lineBuffer; // skip any leading space. while (length && isspace(*line)) { ++line; --length; } if (!length) continue; // comments if (*line == '#') continue; // strip any trailing space. // (will be \n terminated unless there was no \n) while (length && isspace(line[length - 1])) { line[--length] = 0; } if (!length) continue; if (!ParseLine(line, line + length, map)) { fprintf(stderr, "Error in trap definition: %s\n", line); } } fclose(fp); } } #ifdef TEST int main(int argc, char **argv) { for (int i = 1; i < argc; ++i) { std::string f(argv[i]); std::map map; Debug::LoadTrapFile(f, map); for(const auto kv : map) { printf("%s -> %04x\n", kv.first.c_str(), kv.second); } } return 0; } #endif