PageRenderTime 52ms CodeModel.GetById 21ms RepoModel.GetById 1ms app.codeStats 0ms

/Bootloaders/Mighty1284p/avr/cores/standard/WString.cpp

https://gitlab.com/vapkse/arduino
C++ | 645 lines | 509 code | 86 blank | 50 comment | 139 complexity | e96f0c16c068977c288c7b082fdf4c38 MD5 | raw file
  1. /*
  2. WString.cpp - String library for Wiring & Arduino
  3. ...mostly rewritten by Paul Stoffregen...
  4. Copyright (c) 2009-10 Hernando Barragan. All rights reserved.
  5. Copyright 2011, Paul Stoffregen, paul@pjrc.com
  6. This library is free software; you can redistribute it and/or
  7. modify it under the terms of the GNU Lesser General Public
  8. License as published by the Free Software Foundation; either
  9. version 2.1 of the License, or (at your option) any later version.
  10. This library 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 GNU
  13. Lesser General Public License for more details.
  14. You should have received a copy of the GNU Lesser General Public
  15. License along with this library; if not, write to the Free Software
  16. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  17. */
  18. #include "WString.h"
  19. /*********************************************/
  20. /* Constructors */
  21. /*********************************************/
  22. String::String(const char *cstr)
  23. {
  24. init();
  25. if (cstr) copy(cstr, strlen(cstr));
  26. }
  27. String::String(const String &value)
  28. {
  29. init();
  30. *this = value;
  31. }
  32. #ifdef __GXX_EXPERIMENTAL_CXX0X__
  33. String::String(String &&rval)
  34. {
  35. init();
  36. move(rval);
  37. }
  38. String::String(StringSumHelper &&rval)
  39. {
  40. init();
  41. move(rval);
  42. }
  43. #endif
  44. String::String(char c)
  45. {
  46. init();
  47. char buf[2];
  48. buf[0] = c;
  49. buf[1] = 0;
  50. *this = buf;
  51. }
  52. String::String(unsigned char value, unsigned char base)
  53. {
  54. init();
  55. char buf[9];
  56. utoa(value, buf, base);
  57. *this = buf;
  58. }
  59. String::String(int value, unsigned char base)
  60. {
  61. init();
  62. char buf[18];
  63. itoa(value, buf, base);
  64. *this = buf;
  65. }
  66. String::String(unsigned int value, unsigned char base)
  67. {
  68. init();
  69. char buf[17];
  70. utoa(value, buf, base);
  71. *this = buf;
  72. }
  73. String::String(long value, unsigned char base)
  74. {
  75. init();
  76. char buf[34];
  77. ltoa(value, buf, base);
  78. *this = buf;
  79. }
  80. String::String(unsigned long value, unsigned char base)
  81. {
  82. init();
  83. char buf[33];
  84. ultoa(value, buf, base);
  85. *this = buf;
  86. }
  87. String::~String()
  88. {
  89. free(buffer);
  90. }
  91. /*********************************************/
  92. /* Memory Management */
  93. /*********************************************/
  94. inline void String::init(void)
  95. {
  96. buffer = NULL;
  97. capacity = 0;
  98. len = 0;
  99. flags = 0;
  100. }
  101. void String::invalidate(void)
  102. {
  103. if (buffer) free(buffer);
  104. buffer = NULL;
  105. capacity = len = 0;
  106. }
  107. unsigned char String::reserve(unsigned int size)
  108. {
  109. if (buffer && capacity >= size) return 1;
  110. if (changeBuffer(size)) {
  111. if (len == 0) buffer[0] = 0;
  112. return 1;
  113. }
  114. return 0;
  115. }
  116. unsigned char String::changeBuffer(unsigned int maxStrLen)
  117. {
  118. char *newbuffer = (char *)realloc(buffer, maxStrLen + 1);
  119. if (newbuffer) {
  120. buffer = newbuffer;
  121. capacity = maxStrLen;
  122. return 1;
  123. }
  124. return 0;
  125. }
  126. /*********************************************/
  127. /* Copy and Move */
  128. /*********************************************/
  129. String & String::copy(const char *cstr, unsigned int length)
  130. {
  131. if (!reserve(length)) {
  132. invalidate();
  133. return *this;
  134. }
  135. len = length;
  136. strcpy(buffer, cstr);
  137. return *this;
  138. }
  139. #ifdef __GXX_EXPERIMENTAL_CXX0X__
  140. void String::move(String &rhs)
  141. {
  142. if (buffer) {
  143. if (capacity >= rhs.len) {
  144. strcpy(buffer, rhs.buffer);
  145. len = rhs.len;
  146. rhs.len = 0;
  147. return;
  148. } else {
  149. free(buffer);
  150. }
  151. }
  152. buffer = rhs.buffer;
  153. capacity = rhs.capacity;
  154. len = rhs.len;
  155. rhs.buffer = NULL;
  156. rhs.capacity = 0;
  157. rhs.len = 0;
  158. }
  159. #endif
  160. String & String::operator = (const String &rhs)
  161. {
  162. if (this == &rhs) return *this;
  163. if (rhs.buffer) copy(rhs.buffer, rhs.len);
  164. else invalidate();
  165. return *this;
  166. }
  167. #ifdef __GXX_EXPERIMENTAL_CXX0X__
  168. String & String::operator = (String &&rval)
  169. {
  170. if (this != &rval) move(rval);
  171. return *this;
  172. }
  173. String & String::operator = (StringSumHelper &&rval)
  174. {
  175. if (this != &rval) move(rval);
  176. return *this;
  177. }
  178. #endif
  179. String & String::operator = (const char *cstr)
  180. {
  181. if (cstr) copy(cstr, strlen(cstr));
  182. else invalidate();
  183. return *this;
  184. }
  185. /*********************************************/
  186. /* concat */
  187. /*********************************************/
  188. unsigned char String::concat(const String &s)
  189. {
  190. return concat(s.buffer, s.len);
  191. }
  192. unsigned char String::concat(const char *cstr, unsigned int length)
  193. {
  194. unsigned int newlen = len + length;
  195. if (!cstr) return 0;
  196. if (length == 0) return 1;
  197. if (!reserve(newlen)) return 0;
  198. strcpy(buffer + len, cstr);
  199. len = newlen;
  200. return 1;
  201. }
  202. unsigned char String::concat(const char *cstr)
  203. {
  204. if (!cstr) return 0;
  205. return concat(cstr, strlen(cstr));
  206. }
  207. unsigned char String::concat(char c)
  208. {
  209. char buf[2];
  210. buf[0] = c;
  211. buf[1] = 0;
  212. return concat(buf, 1);
  213. }
  214. unsigned char String::concat(unsigned char num)
  215. {
  216. char buf[4];
  217. itoa(num, buf, 10);
  218. return concat(buf, strlen(buf));
  219. }
  220. unsigned char String::concat(int num)
  221. {
  222. char buf[7];
  223. itoa(num, buf, 10);
  224. return concat(buf, strlen(buf));
  225. }
  226. unsigned char String::concat(unsigned int num)
  227. {
  228. char buf[6];
  229. utoa(num, buf, 10);
  230. return concat(buf, strlen(buf));
  231. }
  232. unsigned char String::concat(long num)
  233. {
  234. char buf[12];
  235. ltoa(num, buf, 10);
  236. return concat(buf, strlen(buf));
  237. }
  238. unsigned char String::concat(unsigned long num)
  239. {
  240. char buf[11];
  241. ultoa(num, buf, 10);
  242. return concat(buf, strlen(buf));
  243. }
  244. /*********************************************/
  245. /* Concatenate */
  246. /*********************************************/
  247. StringSumHelper & operator + (const StringSumHelper &lhs, const String &rhs)
  248. {
  249. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  250. if (!a.concat(rhs.buffer, rhs.len)) a.invalidate();
  251. return a;
  252. }
  253. StringSumHelper & operator + (const StringSumHelper &lhs, const char *cstr)
  254. {
  255. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  256. if (!cstr || !a.concat(cstr, strlen(cstr))) a.invalidate();
  257. return a;
  258. }
  259. StringSumHelper & operator + (const StringSumHelper &lhs, char c)
  260. {
  261. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  262. if (!a.concat(c)) a.invalidate();
  263. return a;
  264. }
  265. StringSumHelper & operator + (const StringSumHelper &lhs, unsigned char num)
  266. {
  267. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  268. if (!a.concat(num)) a.invalidate();
  269. return a;
  270. }
  271. StringSumHelper & operator + (const StringSumHelper &lhs, int num)
  272. {
  273. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  274. if (!a.concat(num)) a.invalidate();
  275. return a;
  276. }
  277. StringSumHelper & operator + (const StringSumHelper &lhs, unsigned int num)
  278. {
  279. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  280. if (!a.concat(num)) a.invalidate();
  281. return a;
  282. }
  283. StringSumHelper & operator + (const StringSumHelper &lhs, long num)
  284. {
  285. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  286. if (!a.concat(num)) a.invalidate();
  287. return a;
  288. }
  289. StringSumHelper & operator + (const StringSumHelper &lhs, unsigned long num)
  290. {
  291. StringSumHelper &a = const_cast<StringSumHelper&>(lhs);
  292. if (!a.concat(num)) a.invalidate();
  293. return a;
  294. }
  295. /*********************************************/
  296. /* Comparison */
  297. /*********************************************/
  298. int String::compareTo(const String &s) const
  299. {
  300. if (!buffer || !s.buffer) {
  301. if (s.buffer && s.len > 0) return 0 - *(unsigned char *)s.buffer;
  302. if (buffer && len > 0) return *(unsigned char *)buffer;
  303. return 0;
  304. }
  305. return strcmp(buffer, s.buffer);
  306. }
  307. unsigned char String::equals(const String &s2) const
  308. {
  309. return (len == s2.len && compareTo(s2) == 0);
  310. }
  311. unsigned char String::equals(const char *cstr) const
  312. {
  313. if (len == 0) return (cstr == NULL || *cstr == 0);
  314. if (cstr == NULL) return buffer[0] == 0;
  315. return strcmp(buffer, cstr) == 0;
  316. }
  317. unsigned char String::operator<(const String &rhs) const
  318. {
  319. return compareTo(rhs) < 0;
  320. }
  321. unsigned char String::operator>(const String &rhs) const
  322. {
  323. return compareTo(rhs) > 0;
  324. }
  325. unsigned char String::operator<=(const String &rhs) const
  326. {
  327. return compareTo(rhs) <= 0;
  328. }
  329. unsigned char String::operator>=(const String &rhs) const
  330. {
  331. return compareTo(rhs) >= 0;
  332. }
  333. unsigned char String::equalsIgnoreCase( const String &s2 ) const
  334. {
  335. if (this == &s2) return 1;
  336. if (len != s2.len) return 0;
  337. if (len == 0) return 1;
  338. const char *p1 = buffer;
  339. const char *p2 = s2.buffer;
  340. while (*p1) {
  341. if (tolower(*p1++) != tolower(*p2++)) return 0;
  342. }
  343. return 1;
  344. }
  345. unsigned char String::startsWith( const String &s2 ) const
  346. {
  347. if (len < s2.len) return 0;
  348. return startsWith(s2, 0);
  349. }
  350. unsigned char String::startsWith( const String &s2, unsigned int offset ) const
  351. {
  352. if (offset > len - s2.len || !buffer || !s2.buffer) return 0;
  353. return strncmp( &buffer[offset], s2.buffer, s2.len ) == 0;
  354. }
  355. unsigned char String::endsWith( const String &s2 ) const
  356. {
  357. if ( len < s2.len || !buffer || !s2.buffer) return 0;
  358. return strcmp(&buffer[len - s2.len], s2.buffer) == 0;
  359. }
  360. /*********************************************/
  361. /* Character Access */
  362. /*********************************************/
  363. char String::charAt(unsigned int loc) const
  364. {
  365. return operator[](loc);
  366. }
  367. void String::setCharAt(unsigned int loc, char c)
  368. {
  369. if (loc < len) buffer[loc] = c;
  370. }
  371. char & String::operator[](unsigned int index)
  372. {
  373. static char dummy_writable_char;
  374. if (index >= len || !buffer) {
  375. dummy_writable_char = 0;
  376. return dummy_writable_char;
  377. }
  378. return buffer[index];
  379. }
  380. char String::operator[]( unsigned int index ) const
  381. {
  382. if (index >= len || !buffer) return 0;
  383. return buffer[index];
  384. }
  385. void String::getBytes(unsigned char *buf, unsigned int bufsize, unsigned int index) const
  386. {
  387. if (!bufsize || !buf) return;
  388. if (index >= len) {
  389. buf[0] = 0;
  390. return;
  391. }
  392. unsigned int n = bufsize - 1;
  393. if (n > len - index) n = len - index;
  394. strncpy((char *)buf, buffer + index, n);
  395. buf[n] = 0;
  396. }
  397. /*********************************************/
  398. /* Search */
  399. /*********************************************/
  400. int String::indexOf(char c) const
  401. {
  402. return indexOf(c, 0);
  403. }
  404. int String::indexOf( char ch, unsigned int fromIndex ) const
  405. {
  406. if (fromIndex >= len) return -1;
  407. const char* temp = strchr(buffer + fromIndex, ch);
  408. if (temp == NULL) return -1;
  409. return temp - buffer;
  410. }
  411. int String::indexOf(const String &s2) const
  412. {
  413. return indexOf(s2, 0);
  414. }
  415. int String::indexOf(const String &s2, unsigned int fromIndex) const
  416. {
  417. if (fromIndex >= len) return -1;
  418. const char *found = strstr(buffer + fromIndex, s2.buffer);
  419. if (found == NULL) return -1;
  420. return found - buffer;
  421. }
  422. int String::lastIndexOf( char theChar ) const
  423. {
  424. return lastIndexOf(theChar, len - 1);
  425. }
  426. int String::lastIndexOf(char ch, unsigned int fromIndex) const
  427. {
  428. if (fromIndex >= len) return -1;
  429. char tempchar = buffer[fromIndex + 1];
  430. buffer[fromIndex + 1] = '\0';
  431. char* temp = strrchr( buffer, ch );
  432. buffer[fromIndex + 1] = tempchar;
  433. if (temp == NULL) return -1;
  434. return temp - buffer;
  435. }
  436. int String::lastIndexOf(const String &s2) const
  437. {
  438. return lastIndexOf(s2, len - s2.len);
  439. }
  440. int String::lastIndexOf(const String &s2, unsigned int fromIndex) const
  441. {
  442. if (s2.len == 0 || len == 0 || s2.len > len) return -1;
  443. if (fromIndex >= len) fromIndex = len - 1;
  444. int found = -1;
  445. for (char *p = buffer; p <= buffer + fromIndex; p++) {
  446. p = strstr(p, s2.buffer);
  447. if (!p) break;
  448. if ((unsigned int)(p - buffer) <= fromIndex) found = p - buffer;
  449. }
  450. return found;
  451. }
  452. String String::substring( unsigned int left ) const
  453. {
  454. return substring(left, len);
  455. }
  456. String String::substring(unsigned int left, unsigned int right) const
  457. {
  458. if (left > right) {
  459. unsigned int temp = right;
  460. right = left;
  461. left = temp;
  462. }
  463. String out;
  464. if (left > len) return out;
  465. if (right > len) right = len;
  466. char temp = buffer[right]; // save the replaced character
  467. buffer[right] = '\0';
  468. out = buffer + left; // pointer arithmetic
  469. buffer[right] = temp; //restore character
  470. return out;
  471. }
  472. /*********************************************/
  473. /* Modification */
  474. /*********************************************/
  475. void String::replace(char find, char replace)
  476. {
  477. if (!buffer) return;
  478. for (char *p = buffer; *p; p++) {
  479. if (*p == find) *p = replace;
  480. }
  481. }
  482. void String::replace(const String& find, const String& replace)
  483. {
  484. if (len == 0 || find.len == 0) return;
  485. int diff = replace.len - find.len;
  486. char *readFrom = buffer;
  487. char *foundAt;
  488. if (diff == 0) {
  489. while ((foundAt = strstr(readFrom, find.buffer)) != NULL) {
  490. memcpy(foundAt, replace.buffer, replace.len);
  491. readFrom = foundAt + replace.len;
  492. }
  493. } else if (diff < 0) {
  494. char *writeTo = buffer;
  495. while ((foundAt = strstr(readFrom, find.buffer)) != NULL) {
  496. unsigned int n = foundAt - readFrom;
  497. memcpy(writeTo, readFrom, n);
  498. writeTo += n;
  499. memcpy(writeTo, replace.buffer, replace.len);
  500. writeTo += replace.len;
  501. readFrom = foundAt + find.len;
  502. len += diff;
  503. }
  504. strcpy(writeTo, readFrom);
  505. } else {
  506. unsigned int size = len; // compute size needed for result
  507. while ((foundAt = strstr(readFrom, find.buffer)) != NULL) {
  508. readFrom = foundAt + find.len;
  509. size += diff;
  510. }
  511. if (size == len) return;
  512. if (size > capacity && !changeBuffer(size)) return; // XXX: tell user!
  513. int index = len - 1;
  514. while (index >= 0 && (index = lastIndexOf(find, index)) >= 0) {
  515. readFrom = buffer + index + find.len;
  516. memmove(readFrom + diff, readFrom, len - (readFrom - buffer));
  517. len += diff;
  518. buffer[len] = 0;
  519. memcpy(buffer + index, replace.buffer, replace.len);
  520. index--;
  521. }
  522. }
  523. }
  524. void String::toLowerCase(void)
  525. {
  526. if (!buffer) return;
  527. for (char *p = buffer; *p; p++) {
  528. *p = tolower(*p);
  529. }
  530. }
  531. void String::toUpperCase(void)
  532. {
  533. if (!buffer) return;
  534. for (char *p = buffer; *p; p++) {
  535. *p = toupper(*p);
  536. }
  537. }
  538. void String::trim(void)
  539. {
  540. if (!buffer || len == 0) return;
  541. char *begin = buffer;
  542. while (isspace(*begin)) begin++;
  543. char *end = buffer + len - 1;
  544. while (isspace(*end) && end >= begin) end--;
  545. len = end + 1 - begin;
  546. if (begin > buffer) memcpy(buffer, begin, len);
  547. buffer[len] = 0;
  548. }
  549. /*********************************************/
  550. /* Parsing / Conversion */
  551. /*********************************************/
  552. long String::toInt(void) const
  553. {
  554. if (buffer) return atol(buffer);
  555. return 0;
  556. }