BuffSock.cpp
上传用户:woshihumen
上传日期:2013-07-18
资源大小:484k
文件大小:7k
源码类别:

Email服务器

开发平台:

Visual C++

  1. /*
  2.  *  XMail by Davide Libenzi ( Intranet and Internet mail server )
  3.  *  Copyright (C) 1999,..,2004  Davide Libenzi
  4.  *
  5.  *  This program is free software; you can redistribute it and/or modify
  6.  *  it under the terms of the GNU General Public License as published by
  7.  *  the Free Software Foundation; either version 2 of the License, or
  8.  *  (at your option) any later version.
  9.  *
  10.  *  This program is distributed in the hope that it will be useful,
  11.  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  12.  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  13.  *  GNU General Public License for more details.
  14.  *
  15.  *  You should have received a copy of the GNU General Public License
  16.  *  along with this program; if not, write to the Free Software
  17.  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  18.  *
  19.  *  Davide Libenzi <davidel@xmailserver.org>
  20.  *
  21.  */
  22. #include "SysInclude.h"
  23. #include "SysDep.h"
  24. #include "SvrDefines.h"
  25. #include "StrUtils.h"
  26. #include "BuffSock.h"
  27. #define BSOCK_EOF                   INT_MIN
  28. struct BuffSocketData {
  29. SYS_SOCKET SockFD;
  30. int iBufferSize;
  31. char *pszBuffer;
  32. int iBytesInBuffer;
  33. int iReadIndex;
  34. };
  35. static int BSckFetchData(BuffSocketData * pBSD, int iTimeout);
  36. BSOCK_HANDLE BSckAttach(SYS_SOCKET SockFD, int iBufferSize)
  37. {
  38. BuffSocketData *pBSD = (BuffSocketData *) SysAlloc(sizeof(BuffSocketData));
  39. if (pBSD == NULL)
  40. return (INVALID_BSOCK_HANDLE);
  41. char *pszBuffer = (char *) SysAlloc(iBufferSize);
  42. if (pszBuffer == NULL) {
  43. SysFree(pBSD);
  44. return (INVALID_BSOCK_HANDLE);
  45. }
  46. pBSD->SockFD = SockFD;
  47. pBSD->iBufferSize = iBufferSize;
  48. pBSD->pszBuffer = pszBuffer;
  49. pBSD->iBytesInBuffer = 0;
  50. pBSD->iReadIndex = 0;
  51. return ((BSOCK_HANDLE) pBSD);
  52. }
  53. SYS_SOCKET BSckDetach(BSOCK_HANDLE hBSock, int iCloseSocket)
  54. {
  55. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  56. SYS_SOCKET SockFD = pBSD->SockFD;
  57. SysFree(pBSD->pszBuffer);
  58. SysFree(pBSD);
  59. if (iCloseSocket) {
  60. SysCloseSocket(SockFD);
  61. return (SYS_INVALID_SOCKET);
  62. }
  63. return (SockFD);
  64. }
  65. static int BSckFetchData(BuffSocketData * pBSD, int iTimeout)
  66. {
  67. int iReadedBytes;
  68. pBSD->iReadIndex = 0;
  69. if ((iReadedBytes = SysRecvData(pBSD->SockFD, pBSD->pszBuffer,
  70. pBSD->iBufferSize, iTimeout)) <= 0) {
  71. ErrSetErrorCode(ERR_SOCK_NOMORE_DATA);
  72. return (iReadedBytes);
  73. }
  74. pBSD->iBytesInBuffer = iReadedBytes;
  75. return (iReadedBytes);
  76. }
  77. int BSckGetChar(BSOCK_HANDLE hBSock, int iTimeout)
  78. {
  79. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  80. if ((pBSD->iBytesInBuffer == 0) && (BSckFetchData(pBSD, iTimeout) <= 0))
  81. return (BSOCK_EOF);
  82. int iChar = (int) pBSD->pszBuffer[pBSD->iReadIndex];
  83. pBSD->iReadIndex = INext(pBSD->iReadIndex, pBSD->iBufferSize);
  84. --pBSD->iBytesInBuffer;
  85. return (iChar);
  86. }
  87. char *BSckChGetString(BSOCK_HANDLE hBSock, char *pszBuffer, int iMaxChars, int iTimeout,
  88.       int *pLineLength, int *piGotNL)
  89. {
  90. int ii;
  91. for (ii = 0, iMaxChars--; ii < iMaxChars; ii++) {
  92. int iChar = BSckGetChar(hBSock, iTimeout);
  93. if (iChar == BSOCK_EOF)
  94. return (NULL);
  95. if (iChar == 'n') {
  96. for (; (ii > 0) && (pszBuffer[ii - 1] == 'r'); ii--);
  97. pszBuffer[ii] = '';
  98. if (pLineLength != NULL)
  99. *pLineLength = ii;
  100. if (piGotNL != NULL)
  101. *piGotNL = 1;
  102. return (pszBuffer);
  103. } else
  104. pszBuffer[ii] = (char) iChar;
  105. }
  106. pszBuffer[ii] = '';
  107. if (pLineLength != NULL)
  108. *pLineLength = ii;
  109. if (piGotNL != NULL) {
  110. *piGotNL = 0;
  111. return (pszBuffer);
  112. }
  113. ErrSetErrorCode(ERR_LINE_TOO_LONG);
  114. return (NULL);
  115. }
  116. char *BSckGetString(BSOCK_HANDLE hBSock, char *pszBuffer, int iMaxChars, int iTimeout,
  117.     int *pLineLength, int *piGotNL)
  118. {
  119. int ii;
  120. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  121. for (ii = 0, iMaxChars--; ii < iMaxChars;) {
  122. ///////////////////////////////////////////////////////////////////////////////
  123. //  Verify to have something to read
  124. ///////////////////////////////////////////////////////////////////////////////
  125. if ((pBSD->iBytesInBuffer == 0) && (BSckFetchData(pBSD, iTimeout) <= 0))
  126. return (NULL);
  127. int iBytesLookup = Min(pBSD->iBytesInBuffer, iMaxChars - ii);
  128. if (iBytesLookup > 0) {
  129. char *pszNL = (char *) memchr(pBSD->pszBuffer + pBSD->iReadIndex, 'n',
  130.       iBytesLookup);
  131. if (pszNL != NULL) {
  132. int iCopySize =
  133.     (int) (pszNL - (pBSD->pszBuffer + pBSD->iReadIndex));
  134. memcpy(pszBuffer + ii, pBSD->pszBuffer + pBSD->iReadIndex,
  135.        iCopySize);
  136. ii += iCopySize;
  137. pBSD->iReadIndex += iCopySize + 1;
  138. pBSD->iBytesInBuffer -= iCopySize + 1;
  139. ///////////////////////////////////////////////////////////////////////////////
  140. //  Line cleanup
  141. ///////////////////////////////////////////////////////////////////////////////
  142. for (; (ii > 0) && (pszBuffer[ii - 1] == 'r'); ii--);
  143. pszBuffer[ii] = '';
  144. if (pLineLength != NULL)
  145. *pLineLength = ii;
  146. if (piGotNL != NULL)
  147. *piGotNL = 1;
  148. return (pszBuffer);
  149. } else {
  150. memcpy(pszBuffer + ii, pBSD->pszBuffer + pBSD->iReadIndex,
  151.        iBytesLookup);
  152. ii += iBytesLookup;
  153. pBSD->iReadIndex += iBytesLookup;
  154. pBSD->iBytesInBuffer -= iBytesLookup;
  155. }
  156. }
  157. }
  158. pszBuffer[ii] = '';
  159. if (pLineLength != NULL)
  160. *pLineLength = ii;
  161. if (piGotNL != NULL) {
  162. *piGotNL = 0;
  163. return (pszBuffer);
  164. }
  165. ErrSetErrorCode(ERR_LINE_TOO_LONG);
  166. return (NULL);
  167. }
  168. int BSckSendString(BSOCK_HANDLE hBSock, char const *pszBuffer, int iTimeout)
  169. {
  170. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  171. char *pszSendBuffer = (char *) SysAlloc(strlen(pszBuffer) + 3);
  172. if (pszSendBuffer == NULL)
  173. return (ErrGetErrorCode());
  174. sprintf(pszSendBuffer, "%srn", pszBuffer);
  175. int iSendLength = strlen(pszSendBuffer);
  176. if (SysSend(pBSD->SockFD, pszSendBuffer, iSendLength, iTimeout) != iSendLength) {
  177. SysFree(pszSendBuffer);
  178. return (ErrGetErrorCode());
  179. }
  180. SysFree(pszSendBuffer);
  181. return (iSendLength);
  182. }
  183. int BSckVSendString(BSOCK_HANDLE hBSock, int iTimeout, char const *pszFormat, ...)
  184. {
  185. char *pszBuffer = NULL;
  186. STRSPRINTF(pszBuffer, pszFormat, pszFormat);
  187. if (pszBuffer == NULL)
  188. return (ErrGetErrorCode());
  189. if (BSckSendString(hBSock, pszBuffer, iTimeout) < 0) {
  190. ErrorPush();
  191. SysFree(pszBuffer);
  192. return (ErrorPop());
  193. }
  194. SysFree(pszBuffer);
  195. return (0);
  196. }
  197. int BSckSendData(BSOCK_HANDLE hBSock, char const *pszBuffer, int iSize, int iTimeout)
  198. {
  199. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  200. if (SysSend(pBSD->SockFD, pszBuffer, iSize, iTimeout) != iSize)
  201. return (ErrGetErrorCode());
  202. return (iSize);
  203. }
  204. int BSckReadData(BSOCK_HANDLE hBSock, char *pszBuffer, int iSize, int iTimeout)
  205. {
  206. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  207. int iReadedBytes = 0;
  208. int iReadFromBuffer = Min(iSize, pBSD->iBytesInBuffer);
  209. if (iReadFromBuffer > 0) {
  210. memcpy(pszBuffer, pBSD->pszBuffer + pBSD->iReadIndex, iReadFromBuffer);
  211. pBSD->iReadIndex += iReadFromBuffer;
  212. pBSD->iBytesInBuffer -= iReadFromBuffer;
  213. iReadedBytes = iReadFromBuffer;
  214. }
  215. if ((iReadedBytes < iSize) &&
  216.     (SysRecv(pBSD->SockFD, pszBuffer + iReadedBytes, iSize - iReadedBytes,
  217.      iTimeout) != (iSize - iReadedBytes)))
  218. return (ErrGetErrorCode());
  219. return (iSize);
  220. }
  221. SYS_SOCKET BSckGetAttachedSocket(BSOCK_HANDLE hBSock)
  222. {
  223. BuffSocketData *pBSD = (BuffSocketData *) hBSock;
  224. return (pBSD->SockFD);
  225. }