source: libcfa/src/collections/string_res.hfa@ 5ad6f0d

Last change on this file since 5ad6f0d was 5ad6f0d, checked in by Peter A. Buhr <pabuhr@…>, 5 months ago

more inlining, add strnlen and strncmp for string type

  • Property mode set to 100644
File size: 11.9 KB
Line 
1//
2// Cforall Version 1.0.0 Copyright (C) 2016 University of Waterloo
3//
4// The contents of this file are covered under the licence agreement in the
5// file "LICENCE" distributed with Cforall.
6//
7// string_res -- variable-length, mutable run of text, with resource semantics
8//
9// Author : Michael L. Brooks
10// Created On : Fri Sep 03 11:00:00 2021
11// Last Modified By : Peter A. Buhr
12// Last Modified On : Sun Apr 13 21:03:37 2025
13// Update Count : 79
14//
15
16#pragma once
17
18#include <iostream.hfa>
19#include <string.h> // e.g. strlen
20
21
22//######################### HandleNode #########################
23//private
24
25struct VbyteHeap;
26
27struct HandleNode {
28 HandleNode * flink; // forward link
29 HandleNode * blink; // backward link
30 VbyteHeap * ulink; // upward link
31
32 char * s; // pointer to byte string
33 unsigned int lnth; // length of byte string
34}; // HandleNode
35
36VbyteHeap * DEBUG_string_heap();
37size_t DEBUG_string_bytes_in_heap( VbyteHeap * heap );
38size_t DEBUG_string_bytes_avail_until_gc( VbyteHeap * heap );
39const char * DEBUG_string_heap_start( VbyteHeap * heap );
40
41void TUNING_set_string_heap_liveness_threshold( double val );
42
43//######################### String #########################
44
45// A dynamically-sized string
46struct string_res {
47 HandleNode Handle; // chars, start, end, global neighbours
48 bool shareSet_owns_ulink;
49 string_res * shareSet_prev;
50 string_res * shareSet_next;
51};
52
53
54//######################### charclass_res #########################
55
56struct charclass_res {
57 string_res chars;
58};
59
60void ?{}( charclass_res & ) = void;
61void ?{}( charclass_res &, charclass_res) = void;
62charclass_res ?=?( charclass_res &, charclass_res) = void;
63void ?{}( charclass_res &, const string_res & chars);
64void ?{}( charclass_res &, const char * chars );
65void ?{}( charclass_res &, const char * chars, size_t charssize );
66void ^?{}( charclass_res & );
67
68
69//######################### String #########################
70
71// Getters
72static inline size_t len( const string_res & s ) { return s.Handle.lnth; }
73
74// Constructors, Assignment Operators, Destructor
75void ?{}(string_res & s); // empty string
76void ?{}(string_res & s, const char * buffer, size_t bsize); // copy specific length from buffer
77static inline void ?{}(string_res & s, const char * rhs) { // copy from string literal (NULL-terminated)
78 (s){ rhs, strlen(rhs) };
79}
80static inline void ?{}(string_res & s, char c ) {
81 ?{}( s, &c, 1);
82}
83
84// Deleting the copy constructors makes the compiler reject an attempt to call/return by value
85void ?{}(string_res & s, const string_res & s2) = void;
86void ?{}(string_res & s, string_res & s2) = void;
87
88enum StrResInitMode { COPY_VALUE, SHARE_EDITS };
89void ?{}(string_res & s, const string_res & src, StrResInitMode, size_t start, size_t len );
90static inline void ?{}(string_res & s, const string_res & src, StrResInitMode mode ) {
91 ?{}( s, src, mode, 0, len(src));
92}
93static inline void ?{}(string_res & s, const string_res & src, StrResInitMode mode, size_t maxlen ) {
94 ?{}( s, src, mode, 0, (len(src) > maxlen)?maxlen:len(src) );
95}
96void ?{}( string_res & s, ssize_t rhs );
97void ?{}( string_res & s, size_t rhs );
98void ?{}( string_res & s, double rhs );
99void ?{}( string_res & s, long double rhs );
100void ?{}( string_res & s, double _Complex rhs );
101void ?{}( string_res & s, long double _Complex rhs );
102
103string_res & assign(string_res & s, const string_res & src, size_t maxlen); // copy specific length from other string
104string_res & assign(string_res & s, const char * buffer, size_t bsize); // copy specific length from buffer
105static inline string_res & ?=?(string_res & s, const char * c) { // copy from string literal (NULL-terminated)
106 return assign(s, c, strlen( c));
107}
108string_res & ?=?(string_res & s, const string_res & c);
109string_res & ?=?(string_res & s, string_res & c);
110string_res & ?=?(string_res & s, char c);
111
112string_res & ?=?( string_res & s, ssize_t rhs );
113string_res & ?=?( string_res & s, size_t rhs );
114string_res & ?=?( string_res & s, double rhs );
115string_res & ?=?( string_res & s, long double rhs );
116string_res & ?=?( string_res & s, double _Complex rhs );
117string_res & ?=?( string_res & s, long double _Complex rhs );
118
119void ^?{}(string_res & s);
120
121// IO Operator
122forall( ostype & | basic_ostream( ostype ) ) {
123 ostype & ?|?(ostype & out, const string_res & s);
124 void ?|?(ostype & out, const string_res & s);
125}
126forall( istype & | basic_istream( istype ) )
127istype & ?|?(istype & in, string_res & s);
128
129struct _Istream_Rwidth {
130 string_res * s;
131 inline _Istream_str_base;
132}; // _Istream_Rwidth
133
134struct _Istream_Rquote {
135 // string_res * s;
136 // inline _Istream_str_base;
137 _Istream_Rwidth rstr;
138}; // _Istream_Rquote
139
140struct _Istream_Rstr {
141 string_res * s;
142 inline _Istream_str_base;
143// _Istream_Rwidth rstr;
144}; // _Istream_Rstr
145
146static inline {
147 // read width does not include null terminator
148 _Istream_Rwidth wdi( unsigned int rwd, string_res & s ) { return (_Istream_Rwidth)@{ .s = &s, { {.scanset = 0p}, .wd = rwd, {.flags.rwd = true} } }; }
149 _Istream_Rstr getline( string_res & s, const char delimiter = '\n' ) {
150 return (_Istream_Rstr)@{ .s = &s, { {.delimiters = { delimiter, '\0' } }, .wd = -1, {.flags.delimiter = true} } };
151 }
152 _Istream_Rstr & getline( _Istream_Rwidth & f, const char delimiter = '\n' ) {
153 f.delimiters[0] = delimiter; f.delimiters[1] = '\0'; f.flags.delimiter = true; return (_Istream_Rstr &)f;
154 }
155 _Istream_Rquote quote( string_res & s, const char Ldelimiter = '\"', const char Rdelimiter = '\0' ) {
156 return (_Istream_Rquote)@{ { .s = &s, { {.delimiters = { Ldelimiter, Rdelimiter, '\0' }}, .wd = -1, {.flags.rwd = true} } } };
157 }
158 _Istream_Rquote & quote( _Istream_Rwidth & f, const char Ldelimiter = '"', const char Rdelimiter = '\0' ) {
159 f.delimiters[0] = Ldelimiter; f.delimiters[1] = Rdelimiter; f.delimiters[2] = '\0';
160 return (_Istream_Rquote &)f;
161 }
162 _Istream_Rstr incl( const char scanset[], string_res & s ) { return (_Istream_Rstr)@{ .s = &s, { {.scanset = scanset}, .wd = -1, {.flags.inex = false} } }; }
163 _Istream_Rstr & incl( const char scanset[], _Istream_Rwidth & f ) { f.scanset = scanset; f.flags.inex = false; return (_Istream_Rstr &)f; }
164 _Istream_Rstr excl( const char scanset[], string_res & s ) { return (_Istream_Rstr)@{ .s = &s, { {.scanset = scanset}, .wd = -1, {.flags.inex = true} } }; }
165 _Istream_Rstr & excl( const char scanset[], _Istream_Rwidth & f ) { f.scanset = scanset; f.flags.inex = true; return (_Istream_Rstr &)f; }
166 _Istream_Rstr ignore( string_res & s ) { return (_Istream_Rstr)@{ .s = &s, { {.scanset = 0p}, .wd = -1, {.flags.ignore = true} } }; }
167 _Istream_Rstr & ignore( _Istream_Rwidth & f ) { f.flags.ignore = true; return (_Istream_Rstr &)f; }
168 _Istream_Rquote & ignore( _Istream_Rquote & f ) { f.rstr.flags.ignore = true; return (_Istream_Rquote &)f; }
169 _Istream_Rstr & ignore( _Istream_Rstr & f ) { f.flags.ignore = true; return (_Istream_Rstr &)f; }
170} // distribution
171forall( istype & | basic_istream( istype ) ) {
172 istype & ?|?( istype & is, _Istream_Rquote f );
173 istype & ?|?( istype & is, _Istream_Rstr f );
174 static inline istype & ?|?( istype & is, _Istream_Rwidth f ) { return is | *(_Istream_Rstr *)&f; }
175}
176
177// Concatenation
178void append( string_res & s, const char * buffer, size_t bsize );
179void append( string_res & s, const string_res & s2, size_t maxlen );
180static inline void ?+=?( string_res & s, const string_res & s2 ) { append( s, s2.Handle.s, s2.Handle.lnth ); }
181static inline void ?+=?( string_res & s, char c ) { append( s, & c, 1 ); }
182static inline void ?+=?( string_res & s, const char * c ) { append( s, c, strlen( c ) ); }
183static inline string_res & strcat( string_res & s, const string_res & s2 ) { s += s2; return s; }
184static inline string_res & strcat( string_res & s, const char * c ) { s += c; return s; }
185static inline string_res & strncat( string_res & s, const string_res & s2, size_t maxlen ) { append(s, s2, maxlen); return s; }
186static inline string_res & strncat( string_res & s, const char * buffer, size_t bsize ) { append(s, buffer, bsize); return s; }
187
188// Repetition
189void ?*=?(string_res & s, size_t factor);
190
191// Character access
192void assignAt( const string_res & s, size_t index, char val);
193char ?[?]( const string_res & s, size_t index); // Mike changed to ret by val from Sunjay's ref, to match Peter's
194//char codePointAt( const string_res & s, size_t index); // revisit under Unicode
195
196// Comparisons
197int strcmp$( const char * s1, size_t l1, const char * s2, size_t l2 );
198
199static inline int strcmp( const string_res & s1, const string_res & s2 ) { return strcmp$( s1.Handle.s, s1.Handle.lnth, s2.Handle.s, s2.Handle.lnth ); }
200static inline bool ?==?( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) == 0; }
201static inline bool ?!=?( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) != 0; }
202static inline bool ?>? ( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) > 0; }
203static inline bool ?>=?( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) >= 0; }
204static inline bool ?<=?( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) <= 0; }
205static inline bool ?<? ( const string_res & s1, const string_res & s2 ) { return strcmp( s1, s2 ) < 0; }
206
207static inline int strcmp( const string_res & s1, const char * s2 ) { return strcmp$( s1.Handle.s, s1.Handle.lnth, s2, strlen( s2 ) ); }
208static inline bool ?==?( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) == 0; }
209static inline bool ?!=?( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) != 0; }
210static inline bool ?>? ( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) > 0; }
211static inline bool ?>=?( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) >= 0; }
212static inline bool ?<=?( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) <= 0; }
213static inline bool ?<? ( const string_res & s1, const char * s2 ) { return strcmp( s1, s2 ) < 0; }
214
215static inline int strcmp( const char * s1, const string_res & s2 ) { return strcmp$( s1, strlen( s1 ), s2.Handle.s, s2.Handle.lnth ); }
216static inline bool ?==?( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) == 0; }
217static inline bool ?!=?( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) != 0; }
218static inline bool ?>? ( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) > 0; }
219static inline bool ?>=?( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) >= 0; }
220static inline bool ?<=?( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) <= 0; }
221static inline bool ?<? ( const char * s1, const string_res & s2 ) { return strcmp( s1, s2 ) < 0; }
222
223// String search
224bool contains( const string_res & s, char ch); // single character
225
226int find$( const string_res & s, ssize_t start, ssize_t len, const string_res & key, ssize_t kstart, ssize_t klen );
227
228int find( const string_res & s, char search);
229int find( const string_res & s, const string_res & search);
230int find( const string_res & s, const char * search);
231int find( const string_res & s, const char * search, size_t searchsize);
232
233int findFrom( const string_res & s, size_t fromPos, char search);
234int findFrom( const string_res & s, size_t fromPos, const string_res & search);
235int findFrom( const string_res & s, size_t fromPos, const char * search);
236int findFrom( const string_res & s, size_t fromPos, const char * search, size_t searchsize);
237
238bool includes( const string_res & s, const string_res & search);
239bool includes( const string_res & s, const char * search);
240bool includes( const string_res & s, const char * search, size_t searchsize);
241
242bool startsWith( const string_res & s, const string_res & prefix);
243bool startsWith( const string_res & s, const char * prefix);
244bool startsWith( const string_res & s, const char * prefix, size_t prefixsize);
245
246bool endsWith( const string_res & s, const string_res & suffix);
247bool endsWith( const string_res & s, const char * suffix);
248bool endsWith( const string_res & s, const char * suffix, size_t suffixsize);
249
250int include( const string_res & s, const charclass_res & mask);
251int exclude( const string_res & s, const charclass_res & mask);
252
Note: See TracBrowser for help on using the repository browser.