ref: 7d7e8ae31ac01434c7017afce4eb0a033281a757
dir: /sys/src/cmd/fax/fax2send.c/
#include <u.h> #include <libc.h> #include <bio.h> #include "modem.h" int faxsend(Modem *m, int argc, char *argv[]) { int c, count, r, flow; char buf[128]; verbose("faxsend"); if((r = initfaxmodem(m)) != Eok) return r; /* telco just does the dialing */ r = response(m, 120); switch(r){ case Rok: break; default: r = seterror(m, Enoanswer); return r; } xonoff(m, 1); verbose("sending"); m->pageno = 1; while(argc--){ if(m->pageno != 1) sleep(1000); /* let the paper catch up */ m->valid &= ~(Vfhng|Vfet|Vfpts|Vftsi|Vfdcs); if((r = openfaxfile(m, *argv)) != Eok) return r; verbose("sending geometry"); sprint(buf, "AT+FDT=%ld,%ld,%ld,%ld", m->df, m->vr, m->wd, m->ln); if(command(m, buf) != Eok) goto buggery; if(response(m, 20) != Rconnect){ r = seterror(m, Eincompatible); goto buggery; } /* * Write the data, stuffing DLE's. * After each bufferfull check if the remote * sent us anything, e.g. CAN to abort. * This also flushes out the ^S/^Q characters * which the driver insists on sending us. * (Could fix the driver, of course...). */ verbose("sending data"); for(;;){ flow = 0; count = 0; c = 0; while(count < sizeof(buf)-1){ if((c = Bgetc(m->bp)) < 0) break; buf[count++] = c; if(c == '\020') buf[count++] = c; } verbose("sending %d bytes", count); if(count && write(m->fd, buf, count) < 0){ verbose("write failed: %r"); r = seterror(m, Esys); goto buggery; } /* * this does really rough flow control since the * avanstar is even worse */ verbose("flow control"); while((r = rawmchar(m, buf)) == Eok || flow){ if(r != Eok){ if(flow-- == 0) break; sleep(250); continue; } switch(buf[0]){ case '\030': verbose("%c", buf[0]); if(write(m->fd, "\020\003", 2) < 0){ r = seterror(m, Esys); goto buggery; } goto okexit; case '\021': flow = 0; break; case '\023': flow = 4; break; case '\n': break; default: verbose("%c", buf[0]); r = seterror(m, Eproto); goto buggery; } } if(c < 0) break; } /* * End of page, send DLE+ETX, * get OK in response. */ verbose("sending end of page"); if(write(m->fd, "\020\003", 2) < 0){ r = seterror(m, Esys); goto buggery; } verbose("waiting for OK"); if(response(m, 120) != Rok){ r = seterror(m, Enoresponse); goto buggery; } /* * Did you hear me? - IT'S THE END OF THE PAGE. * Argument is 0 if more pages to follow. * Should get back an FPTS with an indication * as to whether the page was successfully * transmitted or not. */ sprint(buf, "AT+FET=%d", argc == 0? 2: 0); if(command(m, buf) != Eok) goto buggery; switch(response(m, 20)){ case Rok: break; case Rhangup: if(m->fhng == 0 && argc == 0) break; r = seterror(m, Eproto); goto buggery; default: r = seterror(m, Enoresponse); goto buggery; } if((m->valid & Vfpts) == 0 || m->fpts[0] != 1){ r = seterror(m, Eproto); goto buggery; } Bterm(m->bp); m->pageno++; argv++; } okexit: xonoff(m, 0); return Eok; buggery: xonoff(m, 0); Bterm(m->bp); command(m, "AT+FK"); response(m, 5); return r; }