1 // Written in the D programming language. 2 3 /** 4 * Boilerplate: 5 * $(std_boilerplate.html) 6 * Macros: 7 * WIKI = Phobos/StdOutbuffer 8 * 9 * Copyright: Copyright Digital Mars 2000 - 2009. 10 * License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>. 11 * Authors: $(WEB digitalmars.com, Walter Bright) 12 */ 13 /* Copyright Digital Mars 2000 - 2009. 14 * Distributed under the Boost Software License, Version 1.0. 15 * (See accompanying file LICENSE_1_0.txt or copy at 16 * http://www.boost.org/LICENSE_1_0.txt) 17 */ 18 module dmdscript.outbuffer; 19 20 private 21 { 22 import core.memory; 23 import std..string; 24 import core.stdc.stdio; 25 import core.stdc.stdlib; 26 import core.stdc.stdarg; 27 } 28 29 /********************************************* 30 * OutBuffer provides a way to build up an array of bytes out 31 * of raw data. It is useful for things like preparing an 32 * array of bytes to write out to a file. 33 * OutBuffer's byte order is the format native to the computer. 34 * To control the byte order (endianness), use a class derived 35 * from OutBuffer. 36 */ 37 38 class OutBuffer 39 { 40 void[] data; 41 uint offset; 42 43 invariant() 44 { 45 //printf("this = %p, offset = %x, data.length = %u\n", this, offset, data.length); 46 assert(offset <= data.length); 47 } 48 49 this() 50 { 51 //printf("in OutBuffer constructor\n"); 52 } 53 54 /********************************* 55 * Convert to array 56 */ 57 58 void[] toBytes() { return data[0 .. offset]; } 59 60 /*********************************** 61 * Preallocate nbytes more to the size of the internal buffer. 62 * 63 * This is a 64 * speed optimization, a good guess at the maximum size of the resulting 65 * buffer will improve performance by eliminating reallocations and copying. 66 */ 67 68 69 void reserve(size_t nbytes) 70 in 71 { 72 assert(offset + nbytes >= offset); 73 } 74 out 75 { 76 assert(offset + nbytes <= data.length); 77 } 78 body 79 { 80 //c.stdio.printf("OutBuffer.reserve: length = %d, offset = %d, nbytes = %d\n", data.length, offset, nbytes); 81 if (data.length < offset + nbytes) 82 { 83 data.length = (offset + nbytes) * 2; 84 } 85 } 86 87 /************************************* 88 * Append data to the internal buffer. 89 */ 90 91 void write(const(ubyte)[] bytes) 92 { 93 reserve(bytes.length); 94 (cast(ubyte[])data)[offset .. offset + bytes.length] = bytes[0..$]; 95 offset += bytes.length; 96 } 97 98 void write(in wchar[] chars) 99 { 100 write(cast(ubyte[]) chars); 101 } 102 103 void write(const(dchar)[] chars) 104 { 105 write(cast(ubyte[]) chars); 106 } 107 108 void write(ubyte b) /// ditto 109 { 110 reserve(ubyte.sizeof); 111 *cast(ubyte *)&data[offset] = b; 112 offset += ubyte.sizeof; 113 } 114 115 void write(byte b) { write(cast(ubyte)b); } /// ditto 116 void write(char c) { write(cast(ubyte)c); } /// ditto 117 void write(dchar c) { write(cast(uint)c); } /// ditto 118 119 void write(ushort w) /// ditto 120 { 121 reserve(ushort.sizeof); 122 *cast(ushort *)&data[offset] = w; 123 offset += ushort.sizeof; 124 } 125 126 void write(short s) { write(cast(ushort)s); } /// ditto 127 128 void write(wchar c) /// ditto 129 { 130 reserve(wchar.sizeof); 131 *cast(wchar *)&data[offset] = c; 132 offset += wchar.sizeof; 133 } 134 135 void write(uint w) /// ditto 136 { 137 reserve(uint.sizeof); 138 *cast(uint *)&data[offset] = w; 139 offset += uint.sizeof; 140 } 141 142 void write(int i) { write(cast(uint)i); } /// ditto 143 144 void write(ulong l) /// ditto 145 { 146 reserve(ulong.sizeof); 147 *cast(ulong *)&data[offset] = l; 148 offset += ulong.sizeof; 149 } 150 151 void write(long l) { write(cast(ulong)l); } /// ditto 152 153 void write(float f) /// ditto 154 { 155 reserve(float.sizeof); 156 *cast(float *)&data[offset] = f; 157 offset += float.sizeof; 158 } 159 160 void write(double f) /// ditto 161 { 162 reserve(double.sizeof); 163 *cast(double *)&data[offset] = f; 164 offset += double.sizeof; 165 } 166 167 void write(real f) /// ditto 168 { 169 reserve(real.sizeof); 170 *cast(real *)&data[offset] = f; 171 offset += real.sizeof; 172 } 173 174 void write(in char[] s) /// ditto 175 { 176 write(cast(ubyte[])s); 177 } 178 // void write(immutable(char)[] s) /// ditto 179 // { 180 // write(cast(ubyte[])s); 181 // } 182 183 void write(OutBuffer buf) /// ditto 184 { 185 write(cast(ubyte[])buf.toBytes()); 186 } 187 188 /**************************************** 189 * Append nbytes of 0 to the internal buffer. 190 */ 191 192 void fill0(uint nbytes) 193 { 194 reserve(nbytes); 195 auto range = cast(ubyte[])data[offset .. offset + nbytes]; 196 range[] = 0; 197 offset += nbytes; 198 } 199 200 /********************************** 201 * 0-fill to align on power of 2 boundary. 202 */ 203 204 void alignSize(uint alignsize) 205 in 206 { 207 assert(alignsize && (alignsize & (alignsize - 1)) == 0); 208 } 209 out 210 { 211 assert((offset & (alignsize - 1)) == 0); 212 } 213 body 214 { uint nbytes; 215 216 nbytes = offset & (alignsize - 1); 217 if (nbytes) 218 fill0(alignsize - nbytes); 219 } 220 221 /**************************************** 222 * Optimize common special case alignSize(2) 223 */ 224 225 void align2() 226 { 227 if (offset & 1) 228 write(cast(byte)0); 229 } 230 231 /**************************************** 232 * Optimize common special case alignSize(4) 233 */ 234 235 void align4() 236 { 237 if (offset & 3) 238 { uint nbytes = (4 - offset) & 3; 239 fill0(nbytes); 240 } 241 } 242 243 /************************************** 244 * Convert internal buffer to array of chars. 245 */ 246 247 override string toString() 248 { 249 //printf("OutBuffer.toString()\n"); 250 return cast(string) data[0 .. offset].idup; 251 } 252 253 /***************************************** 254 * Append output of C's vprintf() to internal buffer. 255 */ 256 257 void vprintf(string format, va_list args) 258 { 259 char[128] buffer; 260 char* p; 261 uint psize; 262 int count; 263 264 auto f = toStringz(format); 265 p = buffer.ptr; 266 psize = buffer.length; 267 for (;;) 268 { 269 version(Windows) 270 { 271 count = _vsnprintf(p,psize,f,args); 272 if (count != -1) 273 break; 274 psize *= 2; 275 p = cast(char *) alloca(psize); // buffer too small, try again with larger size 276 } 277 version(Posix) 278 { 279 count = vsnprintf(p,psize,f,args); 280 if (count == -1) 281 psize *= 2; 282 else if (count >= psize) 283 psize = count + 1; 284 else 285 break; 286 /+ 287 if (p != buffer) 288 c.stdlib.free(p); 289 p = (char *) c.stdlib.malloc(psize); // buffer too small, try again with larger size 290 +/ 291 p = cast(char *) alloca(psize); // buffer too small, try again with larger size 292 } 293 } 294 write(cast(ubyte[]) p[0 .. count]); 295 /+ 296 version (Posix) 297 { 298 if (p != buffer) 299 c.stdlib.free(p); 300 } 301 +/ 302 } 303 304 /***************************************** 305 * Append output of C's printf() to internal buffer. 306 */ 307 308 void printf(string format, ...) 309 { 310 va_list ap; 311 ap = cast(va_list)&format; 312 ap += format.sizeof; 313 vprintf(format, ap); 314 } 315 316 /***************************************** 317 * At offset index into buffer, create nbytes of space by shifting upwards 318 * all data past index. 319 */ 320 321 void spread(size_t index, size_t nbytes) 322 in 323 { 324 assert(index <= offset); 325 } 326 body 327 { 328 reserve(nbytes); 329 330 // This is an overlapping copy - should use memmove() 331 for (uint i = offset; i > index; ) 332 { 333 --i; 334 *cast(ubyte*)&data[i + nbytes] = *cast(ubyte*)&data[i]; 335 } 336 offset += nbytes; 337 } 338 } 339 340 unittest 341 { 342 //printf("Starting OutBuffer test\n"); 343 344 OutBuffer buf = new OutBuffer(); 345 346 //printf("buf = %p\n", buf); 347 //printf("buf.offset = %x\n", buf.offset); 348 assert(buf.offset == 0); 349 buf.write("hello"[]); 350 buf.write(cast(byte)0x20); 351 buf.write("world"[]); 352 buf.printf(" %d", 6); 353 //printf("buf = '%.*s'\n", buf.toString()); 354 assert(cmp(buf.toString(), "hello world 6") == 0); 355 }