123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- /*
- htop - RichString.c
- (C) 2004,2011 Hisham H. Muhammad
- Released under the GNU GPLv2+, see the COPYING file
- in the source distribution for its full text.
- */
- #include "config.h" // IWYU pragma: keep
- #include "RichString.h"
- #include <assert.h>
- #include <ctype.h>
- #include <limits.h> // IWYU pragma: keep
- #include <stdlib.h>
- #include <string.h>
- #include "Macros.h"
- #include "XUtils.h"
- #define charBytes(n) (sizeof(CharType) * (n))
- static void RichString_extendLen(RichString* this, int len) {
- if (this->chptr == this->chstr) {
- // String is in internal buffer
- if (len > RICHSTRING_MAXLEN) {
- // Copy from internal buffer to allocated string
- this->chptr = xMalloc(charBytes(len + 1));
- memcpy(this->chptr, this->chstr, charBytes(this->chlen));
- } else {
- // Still fits in internal buffer, do nothing
- assert(this->chlen <= RICHSTRING_MAXLEN);
- }
- } else {
- // String is managed externally
- if (len > RICHSTRING_MAXLEN) {
- // Just reallocate the buffer accordingly
- this->chptr = xRealloc(this->chptr, charBytes(len + 1));
- } else {
- // Move string into internal buffer and free resources
- memcpy(this->chstr, this->chptr, charBytes(len));
- free(this->chptr);
- this->chptr = this->chstr;
- }
- }
- RichString_setChar(this, len, 0);
- this->chlen = len;
- }
- static void RichString_setLen(RichString* this, int len) {
- if (len < RICHSTRING_MAXLEN && this->chlen < RICHSTRING_MAXLEN) {
- RichString_setChar(this, len, 0);
- this->chlen = len;
- } else {
- RichString_extendLen(this, len);
- }
- }
- void RichString_rewind(RichString* this, int count) {
- RichString_setLen(this, this->chlen - count);
- }
- #ifdef HAVE_LIBNCURSESW
- static size_t mbstowcs_nonfatal(wchar_t* restrict dest, const char* restrict src, size_t n) {
- size_t written = 0;
- mbstate_t ps = { 0 };
- bool broken = false;
- while (n > 0) {
- size_t ret = mbrtowc(dest, src, n, &ps);
- if (ret == (size_t)-1 || ret == (size_t)-2) {
- if (!broken) {
- broken = true;
- *dest++ = L'\xFFFD';
- written++;
- }
- src++;
- n--;
- continue;
- }
- broken = false;
- if (ret == 0) {
- break;
- }
- dest++;
- written++;
- src += ret;
- n -= ret;
- }
- return written;
- }
- static inline int RichString_writeFromWide(RichString* this, int attrs, const char* data_c, int from, int len) {
- wchar_t data[len];
- len = mbstowcs_nonfatal(data, data_c, len);
- if (len <= 0)
- return 0;
- int newLen = from + len;
- RichString_setLen(this, newLen);
- for (int i = from, j = 0; i < newLen; i++, j++) {
- this->chptr[i] = (CharType) { .attr = attrs & 0xffffff, .chars = { (iswprint(data[j]) ? data[j] : L'\xFFFD') } };
- }
- return len;
- }
- int RichString_appendnWideColumns(RichString* this, int attrs, const char* data_c, int len, int* columns) {
- wchar_t data[len];
- len = mbstowcs_nonfatal(data, data_c, len);
- if (len <= 0)
- return 0;
- int from = this->chlen;
- int newLen = from + len;
- RichString_setLen(this, newLen);
- int columnsWritten = 0;
- int pos = from;
- for (int j = 0; j < len; j++) {
- wchar_t c = iswprint(data[j]) ? data[j] : L'\xFFFD';
- int cwidth = wcwidth(c);
- if (cwidth > *columns)
- break;
- *columns -= cwidth;
- columnsWritten += cwidth;
- this->chptr[pos] = (CharType) { .attr = attrs & 0xffffff, .chars = { c, '\0' } };
- pos++;
- }
- RichString_setLen(this, pos);
- *columns = columnsWritten;
- return pos - from;
- }
- static inline int RichString_writeFromAscii(RichString* this, int attrs, const char* data, int from, int len) {
- int newLen = from + len;
- RichString_setLen(this, newLen);
- for (int i = from, j = 0; i < newLen; i++, j++) {
- assert((unsigned char)data[j] <= SCHAR_MAX);
- this->chptr[i] = (CharType) { .attr = attrs & 0xffffff, .chars = { (isprint((unsigned char)data[j]) ? data[j] : L'\xFFFD') } };
- }
- return len;
- }
- inline void RichString_setAttrn(RichString* this, int attrs, int start, int charcount) {
- int end = CLAMP(start + charcount, 0, this->chlen);
- for (int i = start; i < end; i++) {
- this->chptr[i].attr = attrs;
- }
- }
- void RichString_appendChr(RichString* this, int attrs, char c, int count) {
- int from = this->chlen;
- int newLen = from + count;
- RichString_setLen(this, newLen);
- for (int i = from; i < newLen; i++) {
- this->chptr[i] = (CharType) { .attr = attrs, .chars = { c, 0 } };
- }
- }
- int RichString_findChar(const RichString* this, char c, int start) {
- const wchar_t wc = btowc(c);
- const cchar_t* ch = this->chptr + start;
- for (int i = start; i < this->chlen; i++) {
- if (ch->chars[0] == wc)
- return i;
- ch++;
- }
- return -1;
- }
- #else /* HAVE_LIBNCURSESW */
- static inline int RichString_writeFromWide(RichString* this, int attrs, const char* data_c, int from, int len) {
- int newLen = from + len;
- RichString_setLen(this, newLen);
- for (int i = from, j = 0; i < newLen; i++, j++) {
- this->chptr[i] = (((unsigned char)data_c[j]) >= 32 ? ((unsigned char)data_c[j]) : '?') | attrs;
- }
- this->chptr[newLen] = 0;
- return len;
- }
- int RichString_appendnWideColumns(RichString* this, int attrs, const char* data_c, int len, int* columns) {
- int written = RichString_writeFromWide(this, attrs, data_c, this->chlen, MINIMUM(len, *columns));
- *columns = written;
- return written;
- }
- static inline int RichString_writeFromAscii(RichString* this, int attrs, const char* data_c, int from, int len) {
- return RichString_writeFromWide(this, attrs, data_c, from, len);
- }
- void RichString_setAttrn(RichString* this, int attrs, int start, int charcount) {
- int end = CLAMP(start + charcount, 0, this->chlen);
- for (int i = start; i < end; i++) {
- this->chptr[i] = (this->chptr[i] & 0xff) | attrs;
- }
- }
- void RichString_appendChr(RichString* this, int attrs, char c, int count) {
- int from = this->chlen;
- int newLen = from + count;
- RichString_setLen(this, newLen);
- for (int i = from; i < newLen; i++) {
- this->chptr[i] = c | attrs;
- }
- }
- int RichString_findChar(const RichString* this, char c, int start) {
- const chtype* ch = this->chptr + start;
- for (int i = start; i < this->chlen; i++) {
- if ((*ch & 0xff) == (chtype) c)
- return i;
- ch++;
- }
- return -1;
- }
- #endif /* HAVE_LIBNCURSESW */
- void RichString_delete(RichString* this) {
- if (this->chlen > RICHSTRING_MAXLEN) {
- free(this->chptr);
- this->chptr = this->chstr;
- }
- }
- void RichString_setAttr(RichString* this, int attrs) {
- RichString_setAttrn(this, attrs, 0, this->chlen);
- }
- int RichString_appendWide(RichString* this, int attrs, const char* data) {
- return RichString_writeFromWide(this, attrs, data, this->chlen, strlen(data));
- }
- int RichString_appendnWide(RichString* this, int attrs, const char* data, int len) {
- return RichString_writeFromWide(this, attrs, data, this->chlen, len);
- }
- int RichString_writeWide(RichString* this, int attrs, const char* data) {
- return RichString_writeFromWide(this, attrs, data, 0, strlen(data));
- }
- int RichString_appendAscii(RichString* this, int attrs, const char* data) {
- return RichString_writeFromAscii(this, attrs, data, this->chlen, strlen(data));
- }
- int RichString_appendnAscii(RichString* this, int attrs, const char* data, int len) {
- return RichString_writeFromAscii(this, attrs, data, this->chlen, len);
- }
- int RichString_writeAscii(RichString* this, int attrs, const char* data) {
- return RichString_writeFromAscii(this, attrs, data, 0, strlen(data));
- }
|