jbuff.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362
  1. /*##############################################################################
  2. HPCC SYSTEMS software Copyright (C) 2012 HPCC Systems®.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. ############################################################################## */
  13. #include "platform.h"
  14. #include <stdio.h>
  15. #include <stdlib.h>
  16. #include <stdarg.h>
  17. #include <assert.h>
  18. #include <string.h>
  19. #include <ctype.h>
  20. #include <time.h>
  21. #include <math.h>
  22. #ifndef WIN32
  23. #include <sys/mman.h>
  24. #define LARGEMEM_USE_MMAP_SIZE 0x10000 // in largemem use mmap for chunks bigger than 64K
  25. #endif
  26. #include "jbuff.hpp"
  27. #include "jexcept.hpp"
  28. #include "jmisc.hpp"
  29. #include "jutil.hpp"
  30. #ifdef _DEBUG
  31. #define KILL_CLEARS_MEMORY
  32. //#define TRACE_LARGEMEM
  33. //#define TRACE_LARGEMEM_ALLOC
  34. #define TRACE_LARGEMEM_OOM
  35. #endif
  36. #define VMPAGESIZE (0x1000)
  37. #define VMPAGEMASK (VMPAGESIZE-1)
  38. #define VMPAGEROUND(s) (((s)+VMPAGEMASK)&~VMPAGEMASK)
  39. #if 1
  40. #define ChunkSize 0x10000
  41. #define DOUBLE_LIMIT 0x7fffffff // avoid doubling hitting 0 and infinite loop
  42. #else
  43. #define ChunkSize 2048
  44. #define DOUBLE_LIMIT 4096
  45. #endif
  46. #define FIRST_CHUNK_SIZE 8
  47. #define DETACH_GRANULARITY 16
  48. #ifdef _DEBUG
  49. #define CHECKREADPOS(len) assertex(readPos+(len)<=length())
  50. #else
  51. #define CHECKREADPOS(len)
  52. #endif
  53. //-----------------------------------------------------------------------
  54. static inline size32_t checkMemoryBufferOverflow(size32_t curLen, size32_t inc)
  55. {
  56. size_t newLen = (size_t)curLen + (size_t)inc;
  57. if (newLen > MEMBUFFER_MAXLEN)
  58. RaiseOutOfMemException(-10, curLen, inc, false, "Exceeded maximum size");
  59. return (size32_t)newLen;
  60. }
  61. jlib_decl void *checked_realloc(void *orig, size_t newlen, size_t origlen,int errcode)
  62. {
  63. if (newlen==0) {
  64. free(orig);
  65. return NULL;
  66. }
  67. if (orig==NULL)
  68. return checked_malloc(newlen,errcode);
  69. void *ret = realloc(orig, newlen);
  70. if (!ret)
  71. RaiseOutOfMemException(errcode, newlen, origlen);
  72. return ret;
  73. }
  74. class jlib_thrown_decl COutOfMemException: implements IOutOfMemException, public CInterface
  75. {
  76. int errcode;
  77. size_t wanted;
  78. size_t got;
  79. static int recursion;
  80. bool expected;
  81. StringBuffer errorMsg;
  82. public:
  83. IMPLEMENT_IINTERFACE;
  84. COutOfMemException(int _errcode, size_t _wanted, size_t _got, bool _expected, const char *errMsg)
  85. {
  86. errcode = _errcode;
  87. wanted = _wanted;
  88. expected = _expected;
  89. got = _got;
  90. if (nullptr == errMsg)
  91. errorMsg.append("Out of Memory");
  92. else
  93. errorMsg.append(errMsg);
  94. // DebugBreak();
  95. if ((recursion++==0)&&!expected) {
  96. // Bit risky if *very* out of memory so protect against recursion and catch exceptions
  97. try {
  98. // try to log
  99. PROGLOG("Jbuff: %s (%d,%" I64F "d,%" I64F "dk)",errorMsg.str(),_errcode,(unsigned __int64)wanted,(unsigned __int64) (got/1024));
  100. PrintStackReport();
  101. PrintMemoryReport();
  102. }
  103. catch (...) {
  104. }
  105. }
  106. recursion--;
  107. };
  108. int errorCode() const { return errcode; }
  109. StringBuffer & errorMessage(StringBuffer &str) const
  110. {
  111. str.append("Jbuff: ").append(errorMsg.str()).append(" (").append((unsigned __int64)wanted);
  112. if (got)
  113. str.append(',').append((unsigned __int64)got);
  114. return str.append(")");
  115. }
  116. MessageAudience errorAudience() const { return MSGAUD_user; }
  117. };
  118. int COutOfMemException::recursion=0;
  119. IOutOfMemException *createOutOfMemException(int errcode, size_t wanted, size_t got, bool expected, const char *errMsg)
  120. {
  121. return new COutOfMemException(errcode, wanted, got, expected, errMsg);
  122. }
  123. void RaiseOutOfMemException(int errcode, size_t wanted, size_t got, bool expected, const char *errMsg)
  124. {
  125. throw createOutOfMemException(errcode, wanted, got, expected, errMsg);
  126. }
  127. MemoryAttr::MemoryAttr(size_t _len)
  128. {
  129. ptr = checked_malloc(_len,-1);
  130. len = _len;
  131. }
  132. MemoryAttr::MemoryAttr(size_t _len, const void * _ptr)
  133. {
  134. len = 0;
  135. ptr = NULL;
  136. set(_len, _ptr);
  137. }
  138. MemoryAttr::MemoryAttr(const MemoryAttr & src)
  139. {
  140. len = 0;
  141. ptr = NULL;
  142. set(src.length(), src.get());
  143. }
  144. void MemoryAttr::set(size_t _len, const void * _ptr)
  145. {
  146. memcpy(allocate(_len), _ptr, _len);
  147. }
  148. void MemoryAttr::setOwn(size_t _len, void * _ptr)
  149. {
  150. free(ptr);
  151. len = _len;
  152. ptr = _ptr;
  153. }
  154. void MemoryAttr::clear()
  155. {
  156. free(ptr);
  157. ptr = NULL;
  158. len = 0;
  159. }
  160. void * MemoryAttr::detach()
  161. {
  162. void * ret=ptr;
  163. ptr = NULL;
  164. len = 0;
  165. return ret;
  166. }
  167. int MemoryAttr::compare(const MemoryAttr & m1, const MemoryAttr & m2)
  168. {
  169. size_t len1 = m1.length();
  170. size_t len2 = m2.length();
  171. size_t len = len1;
  172. if (len1 > len2)
  173. len = len2;
  174. int compare = memcmp(m1.get(), m2.get(), len);
  175. if (compare == 0)
  176. compare = (len1 > len2) ? +1 : (len1 < len2) ? -1 : 0;
  177. return compare;
  178. }
  179. void * MemoryAttr::allocate(size_t _len)
  180. {
  181. if (_len==len)
  182. return ptr;
  183. clear();
  184. ptr = checked_malloc(_len,-2);
  185. len = _len;
  186. return ptr;
  187. }
  188. void * MemoryAttr::ensure(size_t _len)
  189. {
  190. if (_len <=len)
  191. return ptr;
  192. return reallocate(_len);
  193. }
  194. void * MemoryAttr::reallocate(size_t _len)
  195. {
  196. if (_len==len)
  197. return ptr;
  198. ptr = checked_realloc(ptr, _len, len, -9);
  199. len = _len;
  200. return ptr;
  201. }
  202. //===========================================================================
  203. void MemoryBuffer::_realloc(size32_t newLen)
  204. {
  205. if (newLen > maxLen)
  206. {
  207. assertex(ownBuffer);
  208. size32_t newMax = maxLen;
  209. //double up to a certain size, otherwise go up in chunks.
  210. if (newLen < DOUBLE_LIMIT)
  211. {
  212. if (newMax == 0)
  213. newMax = FIRST_CHUNK_SIZE;
  214. while (newLen > newMax)
  215. {
  216. size32_t newMaxTmp = checkMemoryBufferOverflow(newMax, newMax);
  217. newMax = newMaxTmp;
  218. }
  219. }
  220. else
  221. {
  222. size32_t newMaxTmp = checkMemoryBufferOverflow((newLen & ~(ChunkSize-1)), ChunkSize);
  223. /*** ((Size + 1) + (ChunkSize - 1)) & ~(ChunkSize-1) ***/
  224. newMax = newMaxTmp;
  225. }
  226. buffer =(char *)checked_realloc(buffer, newMax, maxLen, -7);
  227. maxLen = newMax;
  228. }
  229. }
  230. void MemoryBuffer::_reallocExact(size32_t newLen)
  231. {
  232. if (newLen > maxLen)
  233. {
  234. assertex(ownBuffer);
  235. buffer =(char *)checked_realloc(buffer, newLen, maxLen, -8);
  236. maxLen = newLen;
  237. }
  238. }
  239. void MemoryBuffer::init()
  240. {
  241. buffer = NULL;
  242. curLen = 0;
  243. maxLen = 0;
  244. ownBuffer = true;
  245. readPos = 0;
  246. swapEndian = false;
  247. }
  248. void *MemoryBuffer::insertDirect(unsigned offset, size32_t insertLen)
  249. {
  250. assertex(offset<=curLen);
  251. unsigned newLen = checkMemoryBufferOverflow(curLen, insertLen);
  252. _realloc(newLen);
  253. memmove(buffer + offset + insertLen, buffer + offset, curLen - offset);
  254. curLen += insertLen;
  255. return buffer+offset;
  256. }
  257. void * MemoryBuffer::ensureCapacity(unsigned max)
  258. {
  259. if (maxLen - curLen < max)
  260. {
  261. unsigned newLen = checkMemoryBufferOverflow(curLen, max);
  262. _realloc(newLen);
  263. }
  264. return buffer + curLen;
  265. }
  266. void MemoryBuffer::kill()
  267. {
  268. if (ownBuffer)
  269. free(buffer);
  270. }
  271. MemoryBuffer & MemoryBuffer::_remove(unsigned start, unsigned len)
  272. {
  273. if (start > curLen) start = curLen;
  274. if (start + len > curLen) len = curLen - start;
  275. unsigned start2 = start + len;
  276. memmove(buffer + start, buffer + start2, curLen - start2);
  277. setLength(curLen - len);
  278. return *this;
  279. }
  280. void * MemoryBuffer::reserve(unsigned size)
  281. {
  282. unsigned newLen = checkMemoryBufferOverflow(curLen, size);
  283. _realloc(newLen);
  284. void * ret = buffer + curLen;
  285. curLen += size;
  286. return ret;
  287. }
  288. void * MemoryBuffer::reserveTruncate(unsigned size)
  289. {
  290. unsigned newLen = checkMemoryBufferOverflow(curLen, size);
  291. curLen += size;
  292. _reallocExact(newLen);
  293. truncate();
  294. return buffer + curLen - size;
  295. }
  296. void MemoryBuffer::truncate()
  297. {
  298. if (maxLen>curLen) {
  299. if (curLen==0) {
  300. free(buffer);
  301. buffer = NULL;
  302. }
  303. else
  304. buffer = (char *)realloc(buffer, curLen);
  305. maxLen = curLen;
  306. }
  307. }
  308. void MemoryBuffer::resetBuffer()
  309. {
  310. kill();
  311. init();
  312. }
  313. MemoryBuffer & MemoryBuffer::_reverse()
  314. {
  315. unsigned max = curLen/2;
  316. char * end = buffer + curLen;
  317. unsigned idx;
  318. for (idx = 0; idx < max; idx++)
  319. {
  320. char temp = buffer[idx];
  321. end--;
  322. buffer[idx] = *end;
  323. *end = temp;
  324. }
  325. return *this;
  326. }
  327. void MemoryBuffer::setBuffer(size_t len, void * _buffer, bool takeOwnership)
  328. {
  329. assertex((size32_t)len == len);
  330. kill();
  331. buffer = (char *) _buffer;
  332. if (len) assertex(buffer);
  333. curLen = maxLen = (size32_t)len;
  334. ownBuffer = takeOwnership;
  335. readPos = 0;
  336. }
  337. void *MemoryBuffer::detach()
  338. {
  339. void *ret;
  340. if (ownBuffer) {
  341. if (maxLen>curLen+DETACH_GRANULARITY)
  342. buffer = (char *)realloc(buffer,curLen);
  343. ret = buffer;
  344. }
  345. else {
  346. ret = memcpy(checked_malloc(curLen,-3), buffer, curLen);
  347. }
  348. init();
  349. return ret;
  350. }
  351. void *MemoryBuffer::detachOwn()
  352. {
  353. assertex(ownBuffer);
  354. void *ret = buffer;
  355. init();
  356. return ret;
  357. }
  358. void MemoryBuffer::setLength(unsigned len)
  359. {
  360. if (len > curLen)
  361. {
  362. _realloc(len);
  363. memset(buffer + curLen, 0, len-curLen);
  364. }
  365. else
  366. {
  367. #ifdef KILL_CLEARS_MEMORY
  368. if (curLen)
  369. memset(buffer + len, 'x', curLen-len);
  370. #endif
  371. }
  372. curLen = len;
  373. }
  374. void MemoryBuffer::setWritePos(unsigned len)
  375. {
  376. if (len > curLen)
  377. _realloc(len);
  378. curLen = len;
  379. }
  380. #define SWAP(x, y, t) { t t_##x = x; x = y; y = t_##x; }
  381. void MemoryBuffer::swapWith(MemoryBuffer & other)
  382. {
  383. //swap two string buffers. Used for efficiently moving a string on in a pipeline etc.
  384. SWAP(buffer, other.buffer, char *);
  385. SWAP(curLen, other.curLen, size32_t);
  386. SWAP(maxLen, other.maxLen, size32_t);
  387. SWAP(readPos, other.readPos, size32_t);
  388. SWAP(swapEndian, other.swapEndian, bool);
  389. }
  390. //-----------------------------------------------------------------------
  391. MemoryBuffer::MemoryBuffer(size_t initial)
  392. {
  393. assertex((size32_t)initial == initial);
  394. init();
  395. _realloc((size32_t)initial);
  396. }
  397. MemoryBuffer::MemoryBuffer(MemoryBuffer & value __attribute__((unused)))
  398. {
  399. assertex(!"This should never be used");
  400. }
  401. MemoryBuffer::MemoryBuffer(size_t len, const void * newBuffer)
  402. {
  403. init();
  404. assertex((size32_t)len == len);
  405. append((size32_t)len, newBuffer);
  406. }
  407. MemoryBuffer & MemoryBuffer::append(char value)
  408. {
  409. unsigned newLen = checkMemoryBufferOverflow(curLen, 1);
  410. _realloc(newLen);
  411. buffer[curLen] = value;
  412. ++curLen;
  413. return *this;
  414. }
  415. MemoryBuffer & MemoryBuffer::append(unsigned char value)
  416. {
  417. unsigned newLen = checkMemoryBufferOverflow(curLen, 1);
  418. _realloc(newLen);
  419. buffer[curLen] = value;
  420. ++curLen;
  421. return *this;
  422. }
  423. MemoryBuffer & MemoryBuffer::append(bool value)
  424. {
  425. unsigned newLen = checkMemoryBufferOverflow(curLen, 1);
  426. _realloc(newLen);
  427. buffer[curLen] = (value==0)?0:1;
  428. ++curLen;
  429. return *this;
  430. }
  431. MemoryBuffer & MemoryBuffer::append(const char * value)
  432. {
  433. if (value)
  434. return append((size32_t)strlen(value)+1,value);
  435. else
  436. return append((char)0);
  437. }
  438. MemoryBuffer & MemoryBuffer::append(const unsigned char * value)
  439. {
  440. return append((const char *) value);
  441. }
  442. MemoryBuffer & MemoryBuffer::append(unsigned len, const void * value)
  443. {
  444. unsigned newLen = checkMemoryBufferOverflow(curLen, len);
  445. _realloc(newLen);
  446. memcpy(buffer + curLen, value, len);
  447. curLen += len;
  448. return *this;
  449. }
  450. MemoryBuffer & MemoryBuffer::append(double value)
  451. {
  452. return appendEndian(sizeof(value), &value);
  453. }
  454. MemoryBuffer & MemoryBuffer::append(float value)
  455. {
  456. return appendEndian(sizeof(value), &value);
  457. }
  458. MemoryBuffer & MemoryBuffer::append(short value)
  459. {
  460. return appendEndian(sizeof(value), &value);
  461. }
  462. MemoryBuffer & MemoryBuffer::append(unsigned short value)
  463. {
  464. return appendEndian(sizeof(value), &value);
  465. }
  466. MemoryBuffer & MemoryBuffer::append(int value)
  467. {
  468. return appendEndian(sizeof(value), &value);
  469. }
  470. MemoryBuffer & MemoryBuffer::append(unsigned value)
  471. {
  472. return appendEndian(sizeof(value), &value);
  473. }
  474. #if 0
  475. MemoryBuffer & MemoryBuffer::append(long value)
  476. {
  477. return appendEndian(sizeof(value), &value);
  478. }
  479. MemoryBuffer & MemoryBuffer::append(unsigned long value)
  480. {
  481. return appendEndian(sizeof(value), &value);
  482. }
  483. #endif
  484. MemoryBuffer & MemoryBuffer::append(__int64 value)
  485. {
  486. return appendEndian(sizeof(value), &value);
  487. }
  488. MemoryBuffer & MemoryBuffer::append(unsigned __int64 value)
  489. {
  490. return appendEndian(sizeof(value), &value);
  491. }
  492. MemoryBuffer & MemoryBuffer::appendPacked(unsigned __int64 value)
  493. {
  494. //Append bytes with the top bit set until the value is less than 0x80
  495. while (value >= 0x80)
  496. {
  497. byte next = ((byte)value) | 0x80;
  498. append(next);
  499. value >>= 7;
  500. }
  501. return append((byte)value);
  502. }
  503. MemoryBuffer & MemoryBuffer::append(const MemoryBuffer & value)
  504. {
  505. size32_t SourceLen = value.length();
  506. unsigned newLen = checkMemoryBufferOverflow(curLen, SourceLen);
  507. _realloc(newLen);
  508. memcpy(buffer + curLen, value.toByteArray(), SourceLen);
  509. curLen += SourceLen;
  510. return *this;
  511. }
  512. MemoryBuffer & MemoryBuffer::appendBytes(unsigned char value, unsigned count)
  513. {
  514. unsigned newLen = checkMemoryBufferOverflow(curLen, count);
  515. _realloc(newLen);
  516. memset(buffer+curLen, value, count);
  517. curLen+=count;
  518. return *this;
  519. }
  520. MemoryBuffer & MemoryBuffer::appendEndian(size32_t len, const void * value)
  521. {
  522. unsigned newLen = checkMemoryBufferOverflow(curLen, len);
  523. _realloc(newLen);
  524. if (swapEndian)
  525. _cpyrevn(buffer + curLen, value, len);
  526. else
  527. memcpy(buffer + curLen, value, len);
  528. curLen += len;
  529. return *this;
  530. }
  531. MemoryBuffer & MemoryBuffer::appendSwap(size32_t len, const void * value)
  532. {
  533. unsigned newLen = checkMemoryBufferOverflow(curLen, len);
  534. _realloc(newLen);
  535. _cpyrevn(buffer + curLen, value, len);
  536. curLen += len;
  537. return *this;
  538. }
  539. MemoryBuffer &MemoryBuffer::appendFile(const char *fileName)
  540. {
  541. char buf[1024];
  542. int h = _open(fileName, _O_BINARY | _O_RDONLY | _O_SEQUENTIAL);
  543. if (h == HFILE_ERROR)
  544. throw MakeStringException(0, "MemoryBuffer: Error reading file : %s", fileName);
  545. append(fileName);
  546. unsigned fileSize = _lseek(h, 0, FILE_END);
  547. _lseek(h, 0, FILE_BEGIN);
  548. append(fileSize);
  549. int r;
  550. while ((r = _read(h, buf, 1024)) != 0)
  551. {
  552. if (-1==r) throw makeErrnoException("MemoryBuffer::appendFile");
  553. append(r, buf);
  554. }
  555. _close(h);
  556. return *this;
  557. }
  558. MemoryBuffer & MemoryBuffer::read(char & value)
  559. {
  560. CHECKREADPOS(sizeof(value));
  561. value = buffer[readPos++];
  562. return *this;
  563. }
  564. MemoryBuffer & MemoryBuffer::read(unsigned char & value)
  565. {
  566. CHECKREADPOS(sizeof(value));
  567. value = buffer[readPos++];
  568. return *this;
  569. }
  570. MemoryBuffer & MemoryBuffer::read(bool & value)
  571. {
  572. CHECKREADPOS(sizeof(value));
  573. char _value = buffer[readPos++];
  574. value = (_value==0 ? false : true);
  575. return *this;
  576. }
  577. MemoryBuffer & MemoryBuffer::read(StringAttr & value)
  578. {
  579. char * src = buffer + readPos;
  580. size32_t len = (size32_t)strlen(src);
  581. CHECKREADPOS(len+1);
  582. value.set(src, len);
  583. readPos += (len+1);
  584. return *this;
  585. }
  586. MemoryBuffer & MemoryBuffer::read(StringBuffer & value)
  587. {
  588. char * src = buffer + readPos;
  589. size32_t len = (size32_t)strlen(src);
  590. CHECKREADPOS(len+1);
  591. value.append(len, src);
  592. readPos += (len+1);
  593. return *this;
  594. }
  595. MemoryBuffer & MemoryBuffer::read(const char * &value)
  596. {
  597. value = buffer+readPos;
  598. size32_t len = (size32_t)strlen(value);
  599. CHECKREADPOS(len+1);
  600. readPos += (len+1);
  601. return *this;
  602. }
  603. MemoryBuffer & MemoryBuffer::read(size32_t len, void * value)
  604. {
  605. CHECKREADPOS(len);
  606. memcpy(value, buffer + readPos, len);
  607. readPos += len;
  608. return *this;
  609. }
  610. MemoryBuffer & MemoryBuffer::read(double & value)
  611. {
  612. return readEndian(sizeof(value), &value);
  613. }
  614. MemoryBuffer & MemoryBuffer::read(float & value)
  615. {
  616. return readEndian(sizeof(value), &value);
  617. }
  618. MemoryBuffer & MemoryBuffer::read(short & value)
  619. {
  620. return readEndian(sizeof(value), &value);
  621. }
  622. MemoryBuffer & MemoryBuffer::read(unsigned short & value)
  623. {
  624. return readEndian(sizeof(value), &value);
  625. }
  626. MemoryBuffer & MemoryBuffer::read(int & value)
  627. {
  628. return readEndian(sizeof(value), &value);
  629. }
  630. MemoryBuffer & MemoryBuffer::read(unsigned & value)
  631. {
  632. return readEndian(sizeof(value), &value);
  633. }
  634. #if 0
  635. MemoryBuffer & MemoryBuffer::read(unsigned long & value)
  636. {
  637. return readEndian(sizeof(value), &value);
  638. }
  639. MemoryBuffer & MemoryBuffer::read(long & value)
  640. {
  641. return readEndian(sizeof(value), &value);
  642. }
  643. #endif
  644. MemoryBuffer & MemoryBuffer::read(unsigned __int64 & value)
  645. {
  646. return readEndian(sizeof(value), &value);
  647. }
  648. MemoryBuffer & MemoryBuffer::read(__int64 & value)
  649. {
  650. return readEndian(sizeof(value), &value);
  651. }
  652. const byte * MemoryBuffer::readDirect(size32_t len)
  653. {
  654. CHECKREADPOS(len);
  655. const byte * ret = (const byte *)buffer + readPos;
  656. readPos += len;
  657. return ret;
  658. }
  659. unsigned __int64 MemoryBuffer::readPacked()
  660. {
  661. unsigned __int64 value = 0;
  662. unsigned shift = 0;
  663. for (;;)
  664. {
  665. byte next;
  666. read(next);
  667. value = value | (((unsigned __int64)(next & 0x7f)) << shift);
  668. if (!(next & 0x80))
  669. break;
  670. shift += 7;
  671. }
  672. return value;
  673. }
  674. MemoryBuffer & MemoryBuffer::readPacked(unsigned & value)
  675. {
  676. unsigned __int64 serializedValue = readPacked();
  677. dbgassertex((unsigned)serializedValue == serializedValue);
  678. value = (unsigned)serializedValue;
  679. return *this;
  680. }
  681. MemoryBuffer & MemoryBuffer::readPacked(unsigned __int64 & value)
  682. {
  683. value = readPacked();
  684. return *this;
  685. }
  686. MemoryBuffer & MemoryBuffer::skip(unsigned len)
  687. {
  688. CHECKREADPOS(len);
  689. readPos += len;
  690. return *this;
  691. }
  692. void MemoryBuffer::writeDirect(size32_t pos,size32_t len,const void *buf)
  693. {
  694. assertex(pos+len<=curLen); // does not extend
  695. memcpy(buffer+pos,buf,len);
  696. }
  697. void MemoryBuffer::writeEndianDirect(size32_t pos,size32_t len,const void *buf)
  698. {
  699. assertex(pos+len<=curLen); // does not extend
  700. if (swapEndian)
  701. _cpyrevn(buffer+pos,buf,len);
  702. else
  703. memcpy(buffer+pos,buf,len);
  704. }
  705. MemoryBuffer & MemoryBuffer::readEndian(size32_t len, void * value)
  706. {
  707. CHECKREADPOS(len);
  708. if (swapEndian)
  709. _cpyrevn(value, buffer + readPos, len);
  710. else
  711. memcpy(value, buffer + readPos, len);
  712. readPos += len;
  713. return *this;
  714. }
  715. MemoryBuffer & MemoryBuffer::readSwap(size32_t len, void * value)
  716. {
  717. CHECKREADPOS(len);
  718. _cpyrevn(value, buffer + readPos, len);
  719. readPos += len;
  720. return *this;
  721. }
  722. MemoryBuffer &MemoryBuffer::readFile(StringAttr &fileName)
  723. {
  724. read(fileName);
  725. unsigned fileSize;
  726. read(fileSize);
  727. int h = _open(fileName.get(), _O_WRONLY|_O_CREAT|_O_TRUNC|_O_BINARY|_O_SEQUENTIAL, _S_IREAD | _S_IWRITE);
  728. if (h == HFILE_ERROR)
  729. throw MakeStringException(0, "MemoryBuffer: Unable to create file : %s, error=%d", fileName.get(), GetLastError());
  730. CHECKREADPOS(fileSize);
  731. int w;
  732. while (fileSize) {
  733. w = _write(h, buffer+readPos, fileSize);
  734. if (w == 0) {
  735. _close(h);
  736. throw MakeStringException(0, "MemoryBuffer: Disk full writing %d to file : %s", fileSize, fileName.get());
  737. }
  738. if (w == -1)
  739. {
  740. _close(h);
  741. throw MakeStringException(0, "MemoryBuffer: Error writing to file : %s, error=%d", fileName.get(), GetLastError());
  742. }
  743. readPos += (size32_t)w;
  744. fileSize -= (size32_t)w;
  745. }
  746. _close(h);
  747. return *this;
  748. }
  749. MemoryBuffer & MemoryBuffer::rewrite(size32_t pos)
  750. {
  751. assertex(pos<=maxLen);
  752. curLen = pos;
  753. if (readPos>pos)
  754. readPos = pos;
  755. return *this;
  756. }
  757. MemoryBuffer & MemoryBuffer::reset(size32_t pos)
  758. {
  759. CHECKREADPOS(pos-readPos);
  760. readPos = pos;
  761. return *this;
  762. }
  763. #if 0
  764. void MemoryBuffer::getBytes(int srcBegin, int srcEnd, char * target)
  765. {
  766. memcpy(target, buffer + srcBegin, srcEnd - srcBegin);
  767. }
  768. MemoryBuffer & MemoryBuffer::remove(unsigned start, unsigned len)
  769. {
  770. return (MemoryBuffer &)_remove(start, len);
  771. }
  772. #endif
  773. int MemoryBuffer::setEndian(int endian)
  774. {
  775. assertex((endian == __LITTLE_ENDIAN) || (endian == __BIG_ENDIAN));
  776. bool wasSwapped = setSwapEndian(endian != __BYTE_ORDER);
  777. return wasSwapped ? (__BYTE_ORDER ^ __LITTLE_ENDIAN ^ __BIG_ENDIAN) : __BYTE_ORDER;
  778. }
  779. bool MemoryBuffer::setSwapEndian(bool swap)
  780. {
  781. bool saved = swapEndian;
  782. swapEndian = swap;
  783. return saved;
  784. }
  785. MemoryBuffer & serialize(MemoryBuffer & buffer, const MemoryAttr & value)
  786. {
  787. size32_t length = (size32_t)value.length();
  788. buffer.append(length).append(length, value.get());
  789. return buffer;
  790. }
  791. MemoryBuffer & deserialize(MemoryBuffer & buffer, MemoryAttr & value)
  792. {
  793. unsigned length;
  794. buffer.read(length);
  795. void * target = value.allocate(length);
  796. buffer.read(length, target);
  797. return buffer;
  798. }
  799. MemoryBuffer & serialize(MemoryBuffer & buffer, const char * value)
  800. {
  801. if (value)
  802. {
  803. unsigned length = (size32_t)strlen(value);
  804. buffer.append(length).append(length, value);
  805. }
  806. else
  807. buffer.append((unsigned)-1);
  808. return buffer;
  809. }
  810. MemoryBuffer & deserialize(MemoryBuffer & buffer, StringAttr & value)
  811. {
  812. unsigned length;
  813. buffer.read(length);
  814. if (length == (unsigned)-1)
  815. value.clear();
  816. else
  817. {
  818. char * target = (char *)checked_malloc(length+1,-4);
  819. buffer.read(length, target);
  820. target[length] = 0;
  821. value.setown(target);
  822. }
  823. return buffer;
  824. }
  825. // =====================================================================================================
  826. const char * MemoryAttr2IStringVal::str() const
  827. {
  828. UNIMPLEMENTED;
  829. }
  830. // =====================================================================================================
  831. static memsize_t LMsemlimit=0;
  832. static memsize_t LMtotal=0;
  833. static CriticalSection LMsemsect;
  834. static Owned<ILargeMemLimitNotify> LMnotify;
  835. static bool LMlocked = false;
  836. void setLargeMemLimitNotify(memsize_t size,ILargeMemLimitNotify *notify)
  837. {
  838. CriticalBlock block(LMsemsect);
  839. LMsemlimit = size;
  840. LMnotify.set(notify);
  841. if (LMlocked&&(LMtotal<LMsemlimit))
  842. LMlocked = false;
  843. }
  844. inline void incLargeMemTotal(memsize_t sz)
  845. {
  846. if (sz) {
  847. CriticalBlock block(LMsemsect);
  848. LMtotal += sz;
  849. #ifdef TRACE_LARGEMEM
  850. if ((LMtotal/0x100000)!=((LMtotal-sz)/0x100000))
  851. PROGLOG("LARGEMEM(+): %" I64F "d",(offset_t)LMtotal);
  852. #endif
  853. if (!LMlocked&&LMnotify.get()&&(LMtotal>=LMsemlimit)) {
  854. LMlocked = true;
  855. DBGLOG("LargeMemTotal limit exceeded: %" I64F "d",(offset_t)LMtotal);
  856. if (!LMnotify->take(LMtotal)) {
  857. LMtotal -= sz;
  858. LMlocked = false;
  859. throw createOutOfMemException(-9,sz, LMtotal);
  860. }
  861. DBGLOG("LargeMem taken");
  862. }
  863. }
  864. }
  865. inline void decLargeMemTotal(memsize_t sz)
  866. {
  867. if (sz) {
  868. CriticalBlock block(LMsemsect);
  869. LMtotal -= sz;
  870. #ifdef TRACE_LARGEMEM
  871. if ((LMtotal/0x100000)!=((LMtotal+sz)/0x100000))
  872. PROGLOG("LARGEMEM(-): %" I64F "d",(offset_t)LMtotal);
  873. #endif
  874. if (LMlocked) {
  875. if (LMtotal<LMsemlimit) {
  876. DBGLOG("LargeMemTotal limit reduced to %" I64F "d",(offset_t)LMtotal);
  877. LMlocked = false;
  878. if (LMnotify.get())
  879. LMnotify->give(LMtotal);
  880. }
  881. }
  882. }
  883. }
  884. void CLargeMemoryAllocator::allocchunkmem()
  885. {
  886. #ifdef LARGEMEM_USE_MMAP_SIZE
  887. size32_t masize = VMPAGEROUND(chunk.max);
  888. if (masize>=LARGEMEM_USE_MMAP_SIZE) { // use mmap
  889. chunk.base = (byte *) mmap(NULL,masize,PROT_READ|PROT_WRITE,MAP_PRIVATE|MAP_NORESERVE|MAP_ANONYMOUS,-1,0);
  890. if (chunk.base == (byte *)MAP_FAILED)
  891. chunk.base = NULL;
  892. #ifdef TRACE_LARGEMEM_ALLOC
  893. PROGLOG("CLargeMemoryAllocator::allocchunkmem mmaped %d at %p",masize,chunk.base);
  894. #endif
  895. return;
  896. }
  897. #endif
  898. chunk.base = (byte *)malloc(chunk.max);
  899. #ifdef TRACE_LARGEMEM_ALLOC
  900. PROGLOG("CLargeMemoryAllocator::allocchunkmem malloced %d at %p",chunk.max,chunk.base);
  901. #endif
  902. }
  903. void CLargeMemoryAllocator::disposechunkmem()
  904. {
  905. #ifdef LARGEMEM_USE_MMAP_SIZE
  906. size32_t masize = VMPAGEROUND(chunk.max);
  907. if (masize>=LARGEMEM_USE_MMAP_SIZE) { // use mmap
  908. munmap(chunk.base,masize);
  909. return;
  910. }
  911. #endif
  912. free(chunk.base);
  913. }
  914. bool CLargeMemoryAllocator::newchunk(size32_t sz,size32_t extra,bool exceptionwanted)
  915. {
  916. size32_t newchunksz = (sz>chunkmin)?sz:chunkmin;
  917. if (maxallocated()+newchunksz+extra>totalmax) {
  918. #ifdef TRACE_LARGEMEM_OOM
  919. PrintStackReport();
  920. PROGLOG("OOM.1 wanted sz=%d, extra = %d, maxallocated=%" I64F "d, newchunksz=%u, totalmax=%" I64F "d",sz,extra,(offset_t)maxallocated(),newchunksz,(offset_t)totalmax);
  921. #endif
  922. if (exceptionwanted) {
  923. throw createOutOfMemException(-5,sz, maxallocated(),true);
  924. }
  925. return false;
  926. }
  927. if (chunk.size) {
  928. Chunk *p = new Chunk;
  929. *p = chunk;
  930. chunk.prev = p;
  931. atot += chunk.size;
  932. }
  933. else if (chunk.max) {
  934. decLargeMemTotal(chunk.max);
  935. amax -= chunk.max;
  936. disposechunkmem();
  937. }
  938. chunk.max = newchunksz;
  939. allocchunkmem();
  940. chunk.size = 0;
  941. if (!chunk.base) {
  942. // restore prev
  943. if (chunk.prev) {
  944. Chunk *p = chunk.prev;
  945. chunk = *p;
  946. atot -= chunk.size;
  947. delete p;
  948. }
  949. else
  950. chunk.max = 0;
  951. #ifdef TRACE_LARGEMEM_OOM
  952. PrintStackReport();
  953. PROGLOG("OOM.2 wanted sz=%d, extra = %d, maxallocated=%" I64F "d, newchunksz=%u, totalmax=%" I64F "d",sz,extra,(offset_t)maxallocated(),newchunksz,(offset_t)totalmax);
  954. #endif
  955. if (throwexception) {
  956. throw createOutOfMemException(-6,sz, maxallocated(),true);
  957. }
  958. return false;
  959. }
  960. amax += chunk.max;
  961. incLargeMemTotal(newchunksz);
  962. return true;
  963. }
  964. void CLargeMemoryAllocator::reset()
  965. {
  966. decLargeMemTotal(maxallocated());
  967. disposechunkmem();
  968. while (chunk.prev) {
  969. Chunk *p = chunk.prev;
  970. chunk = *chunk.prev;
  971. delete p;
  972. disposechunkmem();
  973. }
  974. chunk.max = 0;
  975. chunk.base = NULL;
  976. chunk.size = 0;
  977. atot = 0;
  978. amax = 0;
  979. }
  980. void CLargeMemoryAllocator::reduceSize(memsize_t amount)
  981. {
  982. if (amount<=chunk.size) {
  983. chunk.size-=amount;
  984. return;
  985. }
  986. memsize_t reduced = 0;
  987. do {
  988. amount -= chunk.size;
  989. reduced += chunk.max;
  990. disposechunkmem();
  991. amax -= chunk.max;
  992. Chunk *p = chunk.prev;
  993. chunk = *p;
  994. atot -= chunk.size;
  995. delete p;
  996. } while (amount>chunk.size);
  997. chunk.size-=amount;
  998. decLargeMemTotal(reduced);
  999. }
  1000. void CLargeMemoryAllocator::setSize(memsize_t pos)
  1001. {
  1002. memsize_t sz = allocated();
  1003. assertex(sz>=pos);
  1004. reduceSize(sz-pos);
  1005. }
  1006. byte *CLargeMemoryAllocator::next(memsize_t pos,size32_t &size) // this should not be used for small jumps as it is slow
  1007. {
  1008. memsize_t sz = allocated();
  1009. if (sz<=pos) {
  1010. size = 0;
  1011. return NULL;
  1012. }
  1013. memsize_t dif = sz-pos; // how much to go back
  1014. Chunk *p = &chunk;
  1015. while (dif>p->size) {
  1016. dif -= p->size;
  1017. p = p->prev;
  1018. }
  1019. size = (size32_t)dif; // must be smaller than chunk
  1020. return p->base+p->size-dif;
  1021. }
  1022. CLargeMemoryAllocator::CLargeMemoryAllocator()
  1023. {
  1024. // values overwritten by init
  1025. throwexception = true;
  1026. totalmax = 0;
  1027. chunkmin = 0x1000;
  1028. chunk.prev = NULL;
  1029. chunk.max = 0;
  1030. chunk.base = NULL;
  1031. chunk.size = 0;
  1032. atot = 0;
  1033. amax = 0;
  1034. }
  1035. void CLargeMemoryAllocator::init(memsize_t _totalmax,size32_t _chunkmin,bool _throwexception)
  1036. {
  1037. throwexception = _throwexception;
  1038. totalmax = _totalmax;
  1039. chunkmin = _chunkmin;
  1040. chunk.prev = NULL;
  1041. chunk.max = 0;
  1042. chunk.base = NULL;
  1043. chunk.size = 0;
  1044. atot = 0;
  1045. amax = 0;
  1046. }
  1047. MemoryBuffer &CLargeMemoryAllocator::serialize(MemoryBuffer &mb)
  1048. {
  1049. memsize_t al = allocated();
  1050. size32_t sz = (size32_t)al;
  1051. if (sz!=al)
  1052. throw MakeStringException(-1,"CLargeMemoryAllocator::serialize overflow");
  1053. byte *d = (byte *)mb.reserveTruncate(sz)+sz;
  1054. Chunk *p = &chunk;
  1055. while (sz&&p) {
  1056. size32_t s = p->size;
  1057. d -= s;
  1058. memcpy(d,p->base,s);
  1059. p = p->prev;
  1060. sz -= s;
  1061. }
  1062. return mb;
  1063. }
  1064. MemoryBuffer &CLargeMemoryAllocator::deserialize(MemoryBuffer &mb,size32_t sz, size32_t extra)
  1065. {
  1066. mb.read(sz,alloc(sz,extra));
  1067. return mb;
  1068. }
  1069. void *CLargeMemoryAllocator::nextBuffer(void *prev,size32_t &sz)
  1070. { // not fast
  1071. Chunk *p = NULL;
  1072. Chunk *n = &chunk;
  1073. while (n&&(n->base!=prev)) {
  1074. p = n;
  1075. n = n->prev;
  1076. }
  1077. if (!p) {
  1078. sz = 0;
  1079. return NULL;
  1080. }
  1081. sz = p->size;
  1082. return p->base;
  1083. }
  1084. CFixedSizeAllocator::CFixedSizeAllocator()
  1085. {
  1086. chunklist = NULL;
  1087. }
  1088. CFixedSizeAllocator::CFixedSizeAllocator(size32_t _allocsize,size32_t _chunksize)
  1089. {
  1090. chunklist = NULL;
  1091. init(_allocsize,_chunksize);
  1092. }
  1093. void CFixedSizeAllocator::init(size32_t _allocsize,size32_t _chunksize)
  1094. {
  1095. kill();
  1096. allocsize = _allocsize;
  1097. assertex(allocsize);
  1098. if (allocsize<sizeof(void *))
  1099. allocsize = sizeof(void *);
  1100. chunksize = _chunksize;
  1101. if (chunksize<allocsize*16)
  1102. chunksize = allocsize+sizeof(void *); // give up on sublety
  1103. }
  1104. void CFixedSizeAllocator::kill()
  1105. {
  1106. while (chunklist) {
  1107. void *p = chunklist;
  1108. chunklist = *(void **)p;
  1109. freeChunk(p);
  1110. }
  1111. freelist = NULL;
  1112. numalloc = 0;
  1113. numfree = 0;
  1114. chunklist = NULL;
  1115. }
  1116. CFixedSizeAllocator::~CFixedSizeAllocator()
  1117. {
  1118. kill();
  1119. }
  1120. void *CFixedSizeAllocator::allocChunk()
  1121. {
  1122. return checked_malloc(chunksize,-5); // don't try to be clever and allocate less (fragmentation)
  1123. }
  1124. void CFixedSizeAllocator::freeChunk(void *p)
  1125. {
  1126. free(p);
  1127. }
  1128. void *CFixedSizeAllocator::alloc()
  1129. {
  1130. NonReentrantSpinBlock block(lock);
  1131. void *ret;
  1132. if (numfree) {
  1133. numfree--;
  1134. ret = freelist;
  1135. freelist = *(void **)freelist;
  1136. }
  1137. else {
  1138. void **newchunk = (void **)allocChunk();
  1139. unsigned num = (chunksize-sizeof(void *))/allocsize;
  1140. assertex(num);
  1141. *newchunk = chunklist;
  1142. chunklist = (void *)newchunk;
  1143. newchunk++;
  1144. ret = (void *)newchunk;
  1145. numfree+=num-1;
  1146. while (--num) { // we could do this on the fly but I think this marginally better
  1147. newchunk = (void **)(((byte *)newchunk)+allocsize);
  1148. *newchunk = freelist;
  1149. freelist = (void *)newchunk;
  1150. }
  1151. }
  1152. numalloc++;
  1153. return ret;
  1154. }
  1155. void CFixedSizeAllocator::dealloc(void *blk)
  1156. {
  1157. if (blk) {
  1158. NonReentrantSpinBlock block(lock);
  1159. *(void **)blk = freelist;
  1160. freelist = blk;
  1161. numfree++;
  1162. numalloc--;
  1163. }
  1164. }
  1165. void CFixedSizeAllocator::stats(size32_t &sizealloc, size32_t &sizeunused)
  1166. {
  1167. NonReentrantSpinBlock block(lock);
  1168. sizealloc = numalloc*allocsize;
  1169. sizeunused = numfree*allocsize;
  1170. }