1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
|
//#define LOCAL_DEBUG
#include "debug.h"
#include "config.h"
#include "acbuf.h"
#include "fileio.h"
#include <unistd.h>
#include "sockio.h"
namespace acng
{
bool acbuf::setsize(unsigned int c) {
if(m_nCapacity==c)
return true;
char *p = (char*) realloc(m_buf, c+1);
if(!p)
return false;
m_buf=p;
m_nCapacity=c;
m_buf[c]=0; // terminate to make string operations safe
return true;
}
bool acbuf::initFromFile(const char *szPath)
{
struct stat statbuf;
if (0!=stat(szPath, &statbuf))
return false;
int fd=::open(szPath, O_RDONLY);
if (fd<0)
return false;
clear();
if(!setsize(statbuf.st_size))
return false;
while (freecapa()>0)
{
if (sysread(fd) < 0)
{
forceclose(fd);
return false;
}
}
forceclose(fd);
return true;
}
int acbuf::syswrite(int fd, unsigned int maxlen) {
size_t todo(std::min(maxlen, size()));
int n;
do
{
n=::write(fd, rptr(), todo);
} while(n<0 && errno==EINTR);
if(n<0 && errno==EAGAIN)
n=0;
if(n<0)
return -errno;
drop(n);
return n;
}
int acbuf::sysread(int fd, unsigned int maxlen)
{
size_t todo(std::min(maxlen, freecapa()));
int n;
do {
n=::read(fd, m_buf+w, todo);
} while( (n<0 && EINTR == errno) /* || (EAGAIN == errno && n<=0) */ ); // cannot handle EAGAIN here, let the caller check errno
if(n<0)
return -errno;
if(n>0)
w+=n;
return(n);
}
bool tSS::send(int nConFd, mstring* sErrorStatus)
{
while (!empty())
{
auto n = ::send(nConFd, rptr(), size(), 0);
if (n > 0)
{
drop(n);
continue;
}
if (n <= 0)
{
if (EINTR == errno || EAGAIN == errno)
{
struct timeval tv{cfg::nettimeout, 0};
fd_set wfds;
FD_ZERO(&wfds);
FD_SET(nConFd, &wfds);
auto r=::select(nConFd + 1, nullptr, &wfds, nullptr, &tv);
if(!r && errno != EINTR)
{
if(sErrorStatus)
*sErrorStatus = "502 Socket timeout";
return false;
}
continue;
}
#ifdef MINIBUILD
if(sErrorStatus)
*sErrorStatus = "502 Socket error";
#else
if(sErrorStatus)
*sErrorStatus = tErrnoFmter("502 Socket error, ");
#endif
return false;
}
}
return true;
}
bool tSS::recv(int nConFd, mstring* sErrorStatus)
{
struct timeval tv
{ cfg::nettimeout, 0 };
fd_set rfds;
FD_ZERO(&rfds);
FD_SET(nConFd, &rfds);
auto r = ::select(nConFd + 1, &rfds, nullptr, nullptr, &tv);
if (!r)
{
if(errno == EINTR)
return true;
if(sErrorStatus)
*sErrorStatus = "502 Socket timeout";
return false;
}
// must be readable
r = ::recv(nConFd, wptr(), freecapa(), 0);
if(r<=0)
{
#ifdef MINIBUILD
if(sErrorStatus)
*sErrorStatus = "502 Socket error";
#else
if(sErrorStatus)
*sErrorStatus = tErrnoFmter("502 Socket error, ");
#endif
return false;
}
got(r);
return true;
}
/*
tSS & tSS::addEscaped(const char *fmt)
{
if(!fmt || !*fmt)
return *this;
int nl=strlen(fmt);
reserve(length()+nl);
char *p=wptr();
for(;*fmt;fmt++)
{
if(*fmt=='\\')
*(p++)=unEscape(*(++fmt));
else
*(p++)=*fmt;
}
got(p-wptr());
return *this;
}
*/
}
|