forked from facebookarchive/warp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtextbuf.d
177 lines (155 loc) · 3.86 KB
/
textbuf.d
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
/**
* C preprocessor
* Copyright: 2013 by Digital Mars
* License: $(LINK2 http://boost.org/LICENSE_1_0.txt, Boost License 1.0).
* Authors: Walter Bright
*/
module textbuf;
import core.stdc.stdlib;
import core.stdc.string;
import std.stdio;
/**************************************
* Textbuf encapsulates using a local array as a buffer.
* It is initialized with the local array that should be large enough for
* most uses. If the need exceeds the size, it will resize it
* using malloc() and friends.
*/
//debug=Textbuf;
struct Textbuf(T, string id = null)
{
this(T[] buf)
{
assert(!(buf.length & RESIZED));
this.buf = buf.ptr;
this.buflen = cast(uint)buf.length - 2; // 1 for storing past end, 1 for RESIZED
}
void put(T c)
{
const j = i;
buf[j] = c;
i = j + 1;
if (j == buflen)
{
resize(j * 2 + 16);
}
}
static if (T.sizeof == 1)
{
void put(dchar c)
{
put(cast(T)c);
}
void put(const(T)[] s)
{
const newlen = i + s.length;
const len = buflen;
if (newlen > len)
resize(newlen <= len * 2 ? len * 2 : newlen);
buf[i .. newlen] = s[];
i = cast(uint)newlen;
}
}
/******
* Use this to retrieve the result.
*/
T[] opSlice(size_t lwr, size_t upr) pure
{
assert(lwr <= buflen);
assert(upr <= buflen);
assert(lwr <= upr);
return buf[lwr .. upr];
}
T[] opSlice() pure
{
assert(i <= buflen);
return buf[0 .. i];
}
T opIndex(size_t i) const pure
{
assert(i < buflen);
return buf[i];
}
void initialize() pure { i = 0; }
T last() const pure
{
assert(i - 1 < buflen);
return buf[i - 1];
}
T pop() pure
{
assert(i - 1 < buflen);
return buf[--i];
}
@property size_t length() const pure
{
return i;
}
void setLength(size_t i) pure
{
assert(i < buflen);
this.i = cast(uint)i;
}
/**************************
* Release any malloc'd data.
*/
void free()
{
debug(Textbuf) buf[0 .. buflen] = 0;
if (buflen & RESIZED)
.free(buf);
this = this.init;
}
private:
T* buf;
uint buflen;
enum RESIZED = 1; // this bit is set in buflen if we control the memory
uint i;
void resize(size_t newsize)
{
//writefln("%s: oldsize %s newsize %s %s", id, buflen, newsize, &this);
void* p;
if (buflen & RESIZED)
{
/* Prefer realloc when possible
*/
p = realloc(buf, (newsize + 2) * T.sizeof);
assert(p);
}
else
{
assert(i < newsize + 2);
p = malloc((newsize + 2) * T.sizeof);
assert(p);
memcpy(p, buf, i * T.sizeof);
debug(Textbuf) buf[0 .. buflen] = 0;
}
buf = cast(T*)p;
buflen = cast(uint)newsize | RESIZED;
/* Fake loop to prevent inlining. This function is called only rarely,
* inlining results in poorer register allocation.
*/
while (1) { break; }
}
}
unittest
{
char[4] buf = void;
auto textbuf = Textbuf!char(buf);
textbuf.put('a');
textbuf.put('x');
textbuf.put("abc");
assert(textbuf.length == 5);
assert(textbuf[1..3] == "xa");
assert(textbuf[3] == 'b');
textbuf.pop();
assert(textbuf[0..textbuf.length] == "axab");
textbuf.setLength(3);
assert(textbuf[0..textbuf.length] == "axa");
assert(textbuf.last() == 'a');
assert(textbuf[1..3] == "xa");
textbuf.put(cast(dchar)'z');
assert(textbuf[] == "axaz");
textbuf.initialize();
assert(textbuf.length == 0);
textbuf.free();
}