# HG changeset patch # User Chris Cannam # Date 1538477703 -3600 # Node ID 51a8fe0a03cf7e9167467fd64f5e20eec8ab1bce # Parent 2e8a5f665a07154707b5e27033b87cd1845ed7b5 Suspend/resume stdout (as in the Piper simple server) to avoid plugins' own stdout blatherings from interfering with our "return values" diff -r 2e8a5f665a07 -r 51a8fe0a03cf src/helper.cpp --- a/src/helper.cpp Mon Sep 03 16:13:50 2018 +0100 +++ b/src/helper.cpp Tue Oct 02 11:55:03 2018 +0100 @@ -81,8 +81,13 @@ #ifdef _WIN32 #include #include +#include +#else +#include #endif +#include + #include #include @@ -273,6 +278,48 @@ return result; } +// We write our output to stdout, but want to ensure that the plugin +// doesn't write anything itself. To do this we open a null file +// descriptor and dup2() it into place of stdout in the gaps between +// our own output activity. + +static int normalFd = -1; +static int suspendedFd = -1; + +static void initFds() +{ +#ifdef _WIN32 + normalFd = _dup(1); + suspendedFd = _open("NUL", _O_WRONLY); +#else + normalFd = dup(1); + suspendedFd = open("/dev/null", O_WRONLY); +#endif + + if (normalFd < 0 || suspendedFd < 0) { + throw runtime_error("Failed to initialise fds for stdio suspend/resume"); + } +} + +static void suspendOutput() +{ +#ifdef _WIN32 + _dup2(suspendedFd, 1); +#else + dup2(suspendedFd, 1); +#endif +} + +static void resumeOutput() +{ + fflush(stdout); +#ifdef _WIN32 + _dup2(normalFd, 1); +#else + dup2(normalFd, 1); +#endif +} + int main(int argc, char **argv) { bool allGood = true; @@ -313,8 +360,12 @@ SetErrorMode(SEM_FAILCRITICALERRORS); #endif + initFds(); + suspendOutput(); + while (getline(cin, soname)) { Result result = check(soname, descriptor); + resumeOutput(); if (result.code == PluginCheckCode::SUCCESS) { cout << "SUCCESS|" << soname << "|" << endl; } else { @@ -328,6 +379,7 @@ } allGood = false; } + suspendOutput(); } return allGood ? 0 : 1;