123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438 |
- /**
- * Example implementation for bidirectional communication with openMSX.
- *
- * requires: libxml2
- * compile: g++ `xml2-config --cflags` `xml2-config --libs` openmsx-control.cc
- */
- #include <string>
- #include <list>
- #include <iostream>
- #include <unistd.h>
- #include <sys/types.h>
- #include <sys/select.h>
- #include <libxml/parser.h>
- using std::cout;
- using std::endl;
- using std::list;
- using std::string;
- class OpenMSXComm
- {
- public:
- // main loop
- void start();
- // send a command to openmsx
- void sendCommand(const string& command);
- private:
- // therse methods get called when openMSX has send the corresponding tag
- void openmsx_cmd_ok(const string& msg);
- void openmsx_cmd_nok(const string& msg);
- void openmsx_cmd_info(const string& msg);
- void openmsx_cmd_warning(const string& msg);
- void openmsx_cmd_update(const string& name, const string& value);
- // XML parsing call-back functions
- static void cb_start_element(OpenMSXComm* comm, const xmlChar* name,
- const xmlChar** attrs);
- static void cb_end_element(OpenMSXComm* comm, const xmlChar* name);
- static void cb_text(OpenMSXComm* comm, const xmlChar* chars, int len);
- void parseReply(const char** attrs);
- void parseLog(const char** attrs);
- void parseUpdate(const char** attrs);
- void doReply();
- void doLog();
- void doUpdate();
- void deprecated();
- // commands being executed
- list<string> commandStack;
- // XML parsing
- enum State {
- START,
- TAG_OPENMSX,
- TAG_REPLY,
- TAG_LOG,
- TAG_UPDATE,
- TAG_OK,
- TAG_NOK,
- TAG_INFO,
- TAG_WARNING,
- } state;
- unsigned unknownLevel;
- string content;
- xmlSAXHandler sax_handler;
- xmlParserCtxt* parser_context;
- enum ReplyStatus {
- REPLY_UNKNOWN,
- REPLY_OK,
- REPLY_NOK
- } replyStatus;
- enum LogLevel {
- LOG_UNKNOWN,
- LOG_INFO,
- LOG_WARNING
- } logLevel;
- enum UpdateType {
- UPDATE_UNKNOWN,
- UPDATE_LED
- } updateType;
- string updateName;
- // communication with openmsx process
- int fd_out;
- };
- void OpenMSXComm::openmsx_cmd_ok(const string& msg)
- {
- const string& command = commandStack.front();
- cout << "CMD: '" << command << "' executed" << endl;
- if (!msg.empty()) {
- cout << msg << endl;
- }
- commandStack.pop_front();
- }
- void OpenMSXComm::openmsx_cmd_nok(const string& msg)
- {
- const string& command = commandStack.front();
- cout << "CMD: '" << command << "' failed" << endl;
- if (!msg.empty()) {
- cout << msg << endl;
- }
- commandStack.pop_front();
- }
- void OpenMSXComm::openmsx_cmd_info(const string& msg)
- {
- cout << "INFO: " << msg << endl;
- }
- void OpenMSXComm::openmsx_cmd_warning(const string& msg)
- {
- cout << "WARNING: " << msg << endl;
- }
- void OpenMSXComm::openmsx_cmd_update(const string& name, const string& value)
- {
- cout << "UPDATE: " << name << " " << value << endl;
- }
- void OpenMSXComm::deprecated()
- {
- static bool alreadyPrinted = false;
- if (!alreadyPrinted) {
- alreadyPrinted = true;
- cout << "The openMSX you're running still uses the old communication protocol." << endl
- << "Because of this some stuff will not work (LED status for example)." << endl
- << "Please upgrade to openMSX 0.4.0 or higher." << endl;
- }
- }
- void OpenMSXComm::cb_start_element(OpenMSXComm* comm, const xmlChar* name,
- const xmlChar** attrs)
- {
- if (comm->unknownLevel) {
- ++(comm->unknownLevel);
- return;
- }
- switch (comm->state) {
- case START:
- if (strcmp((const char*)name, "openmsx-output") == 0) {
- comm->state = TAG_OPENMSX;
- } else {
- ++(comm->unknownLevel);
- }
- break;
- case TAG_OPENMSX:
- if (strcmp((const char*)name, "reply") == 0) {
- comm->state = TAG_REPLY;
- comm->parseReply((const char**)attrs);
- } else if (strcmp((const char*)name, "log") == 0) {
- comm->state = TAG_LOG;
- comm->parseLog((const char**)attrs);
- } else if (strcmp((const char*)name, "update") == 0) {
- comm->state = TAG_UPDATE;
- comm->parseUpdate((const char**)attrs);
- } // backwards compatibilty stuff
- else if (strcmp((const char*)name, "ok") == 0) {
- comm->state = TAG_OK;
- comm->replyStatus = REPLY_OK;
- comm->deprecated();
- } else if (strcmp((const char*)name, "nok") == 0) {
- comm->state = TAG_NOK;
- comm->replyStatus = REPLY_NOK;
- comm->deprecated();
- } else if (strcmp((const char*)name, "info") == 0) {
- comm->state = TAG_INFO;
- comm->logLevel = LOG_INFO;
- comm->deprecated();
- } else if (strcmp((const char*)name, "warning") == 0) {
- comm->state = TAG_WARNING;
- comm->logLevel = LOG_WARNING;
- comm->deprecated();
- } else {
- ++(comm->unknownLevel);
- }
- break;
- default:
- ++(comm->unknownLevel);
- break;
- }
- comm->content.clear();
- }
- void OpenMSXComm::parseReply(const char** attrs)
- {
- replyStatus = REPLY_UNKNOWN;
- if (attrs) {
- for ( ; *attrs; attrs += 2) {
- if (strcmp(attrs[0], "result") == 0) {
- if (strcmp(attrs[1], "ok") == 0) {
- replyStatus = REPLY_OK;
- } else if (strcmp(attrs[1], "nok") == 0) {
- replyStatus = REPLY_NOK;
- }
- }
- }
- }
- }
- void OpenMSXComm::parseLog(const char** attrs)
- {
- logLevel = LOG_UNKNOWN;
- if (attrs) {
- for ( ; *attrs; attrs += 2) {
- if (strcmp(attrs[0], "level") == 0) {
- if (strcmp(attrs[1], "info") == 0) {
- logLevel = LOG_INFO;
- } else if (strcmp(attrs[1], "warning") == 0) {
- logLevel = LOG_WARNING;
- }
- }
- }
- }
- }
- void OpenMSXComm::parseUpdate(const char** attrs)
- {
- updateType = UPDATE_UNKNOWN;
- if (attrs) {
- for ( ; *attrs; attrs += 2) {
- if (strcmp(attrs[0], "type") == 0) {
- if (strcmp(attrs[1], "led") == 0) {
- updateType = UPDATE_LED;
- }
- } else if (strcmp(attrs[0], "name") == 0) {
- updateName = attrs[1];
- }
- }
- }
- }
- void OpenMSXComm::cb_end_element(OpenMSXComm* comm, const xmlChar* name)
- {
- if (comm->unknownLevel) {
- --(comm->unknownLevel);
- return;
- }
- switch (comm->state) {
- case TAG_OPENMSX:
- comm->state = START;
- break;
- case TAG_REPLY:
- comm->doReply();
- comm->state = TAG_OPENMSX;
- break;
- case TAG_LOG:
- comm->doLog();
- comm->state = TAG_OPENMSX;
- break;
- case TAG_UPDATE:
- comm->doUpdate();
- comm->state = TAG_OPENMSX;
- break;
- // backwards compatibilty stuff
- case TAG_OK:
- comm->openmsx_cmd_ok(comm->content);
- comm->state = TAG_OPENMSX;
- break;
- case TAG_NOK:
- comm->openmsx_cmd_nok(comm->content);
- comm->state = TAG_OPENMSX;
- break;
- case TAG_INFO:
- comm->openmsx_cmd_info(comm->content);
- comm->state = TAG_OPENMSX;
- break;
- case TAG_WARNING:
- comm->openmsx_cmd_warning(comm->content);
- comm->state = TAG_OPENMSX;
- break;
- default:
- break;
- }
- }
- void OpenMSXComm::doReply()
- {
- switch (replyStatus) {
- case REPLY_OK:
- openmsx_cmd_ok(content);
- break;
- case REPLY_NOK:
- openmsx_cmd_nok(content);
- break;
- }
- }
- void OpenMSXComm::doLog()
- {
- switch (logLevel) {
- case LOG_INFO:
- openmsx_cmd_info(content);
- break;
- case LOG_WARNING:
- openmsx_cmd_warning(content);
- break;
- }
- }
- void OpenMSXComm::doUpdate()
- {
- switch (updateType) {
- case UPDATE_LED:
- openmsx_cmd_update(updateName, content);
- break;
- }
- }
- void OpenMSXComm::cb_text(OpenMSXComm* comm, const xmlChar* chars, int len)
- {
- switch (comm->state) {
- case TAG_REPLY:
- case TAG_LOG:
- case TAG_UPDATE:
- case TAG_OK:
- case TAG_NOK:
- case TAG_INFO:
- case TAG_WARNING:
- comm->content.append((const char*)chars, len);
- break;
- default:
- break;
- }
- }
- void OpenMSXComm::sendCommand(const string& command)
- {
- write(fd_out, "<command>", 9);
- write(fd_out, command.c_str(), command.length());
- write(fd_out, "</command>", 10);
- commandStack.push_back(command);
- }
- void OpenMSXComm::start()
- {
- // init XML parser
- state = START;
- unknownLevel = 0;
- memset(&sax_handler, 0, sizeof(sax_handler));
- sax_handler.startElement = (startElementSAXFunc)cb_start_element;
- sax_handler.endElement = (endElementSAXFunc) cb_end_element;
- sax_handler.characters = (charactersSAXFunc) cb_text;
- parser_context = xmlCreatePushParserCtxt(&sax_handler, this, 0, 0, 0);
- // create pipes
- int pipe_to_child[2];
- int pipe_from_child[2];
- pipe(pipe_to_child);
- pipe(pipe_from_child);
- fd_out = pipe_to_child[1];
- // start openmsx sub-process
- pid_t pid = fork();
- if (pid == 0) {
- dup2(pipe_to_child[0], STDIN_FILENO);
- dup2(pipe_from_child[1], STDOUT_FILENO);
- close(pipe_to_child[0]);
- close(pipe_to_child[1]);
- close(pipe_from_child[0]);
- close(pipe_from_child[1]);
- execlp("openmsx", "openmsx", "-control", "stdio:", 0);
- exit(0);
- }
- close(pipe_to_child[0]);
- close(pipe_from_child[1]);
- // send initial commands
- write(pipe_to_child[1], "<openmsx-control>", 17);
- sendCommand("set power on");
- sendCommand("restoredefault renderer");
- // event loop
- string command; // (partial) input from STDIN
- while (true) {
- char buf[4096];
- fd_set rdfs;
- FD_ZERO(&rdfs);
- FD_SET(pipe_from_child[0], &rdfs);
- FD_SET(STDIN_FILENO, &rdfs);
- select(pipe_from_child[0] + 1, &rdfs, NULL, NULL, NULL);
- if (FD_ISSET(pipe_from_child[0], &rdfs)) {
- // data available from openMSX
- ssize_t size = read(pipe_from_child[0], buf, 4096);
- if (size == 0) {
- // openmsx process died
- break;
- }
- xmlParseChunk(parser_context, buf, size, 0);
- }
- if (FD_ISSET(STDIN_FILENO, &rdfs)) {
- // data available from STDIN
- ssize_t size = read(STDIN_FILENO, buf, 4096);
- char* oldpos = buf;
- while (true) {
- char* pos = (char*)memchr(oldpos, '\n', size);
- if (pos) {
- unsigned num = pos - oldpos;
- command.append(oldpos, num);
- sendCommand(command);
- command.clear();
- oldpos = pos + 1;
- size -= num + 1;
- } else {
- command.append(pos, size);
- break;
- }
- }
- }
- }
- // cleanup
- xmlFreeParserCtxt(parser_context);
- }
- int main()
- {
- OpenMSXComm comm;
- comm.start();
- return 0;
- }
|