Simplification through adversity, or, POSIX is not as portable as it would like :D It still works, though!

This commit is contained in:
Pietro Gagliardi 2018-05-24 16:11:58 -04:00
parent a5e07be2f5
commit 8b35ab973d
1 changed files with 49 additions and 32 deletions

View File

@ -1,38 +1,59 @@
// 21 may 2018 // 21 may 2018
#ifdef _WIN32
#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1
#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES_COUNT 1
#include <io.h>
#include <sys/stat.h>
#define openfunc _open
#define openflags (_O_RDONLY | _O_BINARY)
#define openmode (_S_IREAD)
#define readfunc _read
#define readtype int
#define closefunc _close
#else
#include <fcntl.h>
#include <unistd.h>
#define openfunc open
#define openflags (O_RDONLY)
#define openmode 0644
#define readfunc read
#define readtype ssize_t
#define closefunc close
#endif
#include <vector> #include <vector>
#include <stdio.h> #include <stdio.h>
#include <string.h> #include <string.h>
#include <errno.h>
class Scanner { class Scanner {
FILE *fin; int fd;
char *buf; char *buf;
const char *p; const char *p;
size_t n; size_t n;
std::vector<char> *line; std::vector<char> *line;
bool eof; bool eof;
bool eofNextTime; int error;
bool error;
public: public:
Scanner(FILE *fin); Scanner(int fd);
~Scanner(void); ~Scanner(void);
bool Scan(void); bool Scan(void);
const char *Bytes(void) const; const char *Bytes(void) const;
size_t Len(void) const; size_t Len(void) const;
bool Error(void) const; int Error(void) const;
}; };
#define nbuf 1024 #define nbuf 1024
Scanner::Scanner(FILE *fin) Scanner::Scanner(int fd)
{ {
this->fin = fin; this->fd = fd;
this->buf = new char[nbuf]; this->buf = new char[nbuf];
this->p = this->buf; this->p = this->buf;
this->n = 0; this->n = 0;
this->line = new std::vector<char>; this->line = new std::vector<char>;
this->eof = false; this->eof = false;
this->error = false; this->error = 0;
} }
Scanner::~Scanner(void) Scanner::~Scanner(void)
@ -43,7 +64,9 @@ Scanner::~Scanner(void)
bool Scanner::Scan(void) bool Scanner::Scan(void)
{ {
if (this->eof || this->error) readtype n;
if (this->eof || this->error != 0)
return false; return false;
this->line->clear(); this->line->clear();
for (;;) { for (;;) {
@ -69,24 +92,17 @@ bool Scanner::Scan(void)
// otherwise, the buffer was exhausted in the middle of a line, so fall through // otherwise, the buffer was exhausted in the middle of a line, so fall through
} }
// need to refill the buffer // need to refill the buffer
if (this->eofNextTime) { n = readfunc(this->fd, this->buf, nbuf * sizeof (char));
if (n < 0) {
this->error = errno;
return false;
}
if (n == 0) {
this->eof = true; this->eof = true;
return false; return false;
} }
this->n = fread(this->buf, sizeof (char), nbuf, this->fin);
if (this->n < nbuf) {
// TODO what if this->eofNextTime && this->error? the C standard does not expressly disallow this
this->eofNextTime = feof(this->fin) != 0;
this->error = ferror(this->fin) != 0;
// according to various people in irc.freenode.net/##c, feof() followed by fread() can result in ferror(), so we must be sure not to read twice on a feof()
// however, because a partial (nonzero) fread() may or may not set feof(), we have to do this whole delayed check acrobatics
if (this->eofNextTime && this->n == 0)
this->eof = true;
if (this->eof || this->error)
return false;
// otherwise process this last chunk of the file
}
this->p = this->buf; this->p = this->buf;
this->n = n;
} }
} }
@ -100,7 +116,7 @@ size_t Scanner::Len(void) const
return this->line->size(); return this->line->size();
} }
bool Scanner::Error(void) const int Scanner::Error(void) const
{ {
return this->error; return this->error;
} }
@ -135,7 +151,8 @@ bool process(const char *line, size_t n, FILE *fout)
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
FILE *fin = NULL, *fout = NULL; int fin = -1;
FILE *fout = NULL;
Scanner *s = NULL; Scanner *s = NULL;
int ret = 1; int ret = 1;
@ -144,9 +161,9 @@ int main(int argc, char *argv[])
return 1; return 1;
} }
fin = fopen(argv[1], "rb"); fin = openfunc(argv[1], openflags, openmode);
if (fin == NULL) { if (fin < 0) {
fprintf(stderr, "error opening %s\n", argv[1]); fprintf(stderr, "error opening %s: %s\n", argv[1], strerror(errno));
goto done; goto done;
} }
fout = fopen(argv[2], "wb"); fout = fopen(argv[2], "wb");
@ -167,8 +184,8 @@ int main(int argc, char *argv[])
goto done; goto done;
} }
} }
if (s->Error()) { if (s->Error() != 0) {
fprintf(stderr, "error reading from %s\n", argv[1]); fprintf(stderr, "error reading from %s: %s\n", argv[1], strerror(s->Error()));
goto done; goto done;
} }
@ -178,7 +195,7 @@ done:
delete s; delete s;
if (fout != NULL) if (fout != NULL)
fclose(fout); fclose(fout);
if (fin != NULL) if (fin >= 0)
fclose(fin); closefunc(fin);
return ret; return ret;
} }