Merge branch 'master' of github.com:adamdruppe/arsd

This commit is contained in:
Adam D. Ruppe 2019-08-23 19:44:07 -04:00
commit b615f51ac8
6 changed files with 87 additions and 58 deletions

21
cgi.d
View File

@ -1708,7 +1708,8 @@ class Cgi {
remoteAddress = value;
}
else if (name == "x-forwarded-host" || name == "host") {
host = value;
if(name != "host" || host is null)
host = value;
}
else if (name == "accept-encoding") {
if(value.indexOf("gzip") != -1)
@ -5662,7 +5663,13 @@ final class BasicDataServerImplementation : BasicDataServer, EventIoServer {
}
/++
See [schedule] to make one of these. You then call one of the methods here to set it up:
---
schedule!fn(args).at(DateTime(2019, 8, 7, 12, 00, 00)); // run the function at August 7, 2019, 12 noon UTC
schedule!fn(args).delay(6.seconds); // run it after waiting 6 seconds
schedule!fn(args).asap(); // run it in the background as soon as the event loop gets around to it
---
+/
struct ScheduledJobHelper {
private string func;
@ -5684,6 +5691,10 @@ struct ScheduledJobHelper {
void at(DateTime when, immutable TimeZone timezone = UTC()) {
consumed = true;
auto conn = ScheduledJobServerConnection.connection;
import std.file;
auto st = SysTime(when, timezone);
auto jobId = conn.scheduleJob(1, cast(int) st.toUnixTime(), thisExePath, func, args);
}
/++
@ -5692,6 +5703,9 @@ struct ScheduledJobHelper {
void delay(Duration delay) {
consumed = true;
auto conn = ScheduledJobServerConnection.connection;
import std.file;
auto jobId = conn.scheduleJob(0, cast(int) delay.total!"seconds", thisExePath, func, args);
}
/++
@ -5701,19 +5715,20 @@ struct ScheduledJobHelper {
+/
void asap() {
consumed = true;
//delay(0);
auto conn = ScheduledJobServerConnection.connection;
import std.file;
auto jobId = conn.scheduleJob(0, 1, thisExePath, func, args);
}
/+
/++
Schedules the job to recur on the given pattern.
+/
version(none)
void recur(string spec) {
}
+/
}
private immutable void delegate(string[])[string] scheduledJobHandlers;

23
color.d
View File

@ -923,7 +923,7 @@ class IndexedImage : MemoryImage {
override Color getPixel(int x, int y) const pure nothrow @trusted @nogc {
if (x >= 0 && y >= 0 && x < _width && y < _height) {
uint pos = y*_width+x;
size_t pos = cast(size_t)y*_width+x;
if (pos >= data.length) return Color(0, 0, 0, 0);
ubyte b = data.ptr[pos];
if (b >= palette.length) return Color(0, 0, 0, 0);
@ -935,7 +935,7 @@ class IndexedImage : MemoryImage {
override void setPixel(int x, int y, in Color clr) nothrow @trusted {
if (x >= 0 && y >= 0 && x < _width && y < _height) {
uint pos = y*_width+x;
size_t pos = cast(size_t)y*_width+x;
if (pos >= data.length) return;
ubyte pidx = findNearestColor(palette, clr);
if (palette.length < 255 &&
@ -954,7 +954,11 @@ class IndexedImage : MemoryImage {
this(int w, int h) pure nothrow @safe {
_width = w;
_height = h;
data = new ubyte[w*h];
// ensure that the computed size does not exceed basic address space limits
assert(cast(ulong)w * h <= size_t.max);
// upcast to avoid overflow for images larger than 536 Mpix
data = new ubyte[cast(size_t)w*h];
}
/*
@ -1061,7 +1065,7 @@ class TrueColorImage : MemoryImage {
override Color getPixel(int x, int y) const pure nothrow @trusted @nogc {
if (x >= 0 && y >= 0 && x < _width && y < _height) {
uint pos = y*_width+x;
size_t pos = cast(size_t)y*_width+x;
return imageData.colors.ptr[pos];
} else {
return Color(0, 0, 0, 0);
@ -1070,7 +1074,7 @@ class TrueColorImage : MemoryImage {
override void setPixel(int x, int y, in Color clr) nothrow @trusted {
if (x >= 0 && y >= 0 && x < _width && y < _height) {
uint pos = y*_width+x;
size_t pos = cast(size_t)y*_width+x;
if (pos < imageData.bytes.length/4) imageData.colors.ptr[pos] = clr;
}
}
@ -1079,14 +1083,19 @@ class TrueColorImage : MemoryImage {
this(int w, int h) pure nothrow @safe {
_width = w;
_height = h;
imageData.bytes = new ubyte[w*h*4];
// ensure that the computed size does not exceed basic address space limits
assert(cast(ulong)w * h * 4 <= size_t.max);
// upcast to avoid overflow for images larger than 536 Mpix
imageData.bytes = new ubyte[cast(size_t)w * h * 4];
}
/// Creates with existing data. The data pointer is stored here.
this(int w, int h, ubyte[] data) pure nothrow @safe {
_width = w;
_height = h;
assert(data.length == w * h * 4);
assert(cast(ulong)w * h * 4 <= size_t.max);
assert(data.length == cast(size_t)w * h * 4);
imageData.bytes = data;
}

View File

@ -14,13 +14,15 @@
{
"name": "simpledisplay",
"description": "Window creation and basic drawing",
"targetType": "sourceLibrary",
"libs-posix": ["X11", "Xext", "GL", "GLU"],
"libs-windows": ["gdi32", "opengl32", "glu32"],
"targetType": "library",
"importPaths": ["."],
"dflags": ["-mv=arsd.simpledisplay=simpledisplay.d"],
"dependencies": {"arsd-official:color_base":"*"},
"configurations": [
{
"name": "normal"
"name": "normal",
"libs-posix": ["X11", "Xext", "GL", "GLU"],
"libs-windows": ["gdi32", "opengl32", "glu32"],
},
{
"name": "without-opengl",
@ -34,19 +36,10 @@
{
"name": "minigui",
"description": "Small GUI widget library for Windows and Linux",
"targetType": "sourceLibrary",
"targetType": "library",
"importPaths": ["."],
"dflags": ["-mv=arsd.minigui=minigui.d"],
"dependencies": {"arsd-official:simpledisplay":"*"},
"configurations": [
{
"name": "normal"
},
{
"name": "without-opengl",
"versions": ["without_opengl"],
"libs-windows": ["gdi32"],
"libs-posix": ["X11", "Xext"]
}
],
"sourceFiles": ["minigui.d"]
},
{
@ -202,7 +195,9 @@
{
"name": "color_base",
"description": "Base color, point, image interface definitions",
"targetType": "sourceLibrary",
"targetType": "library",
"importPaths": ["."],
"dflags": ["-mv=arsd.color=color.d"],
"sourceFiles": ["color.d"]
},
{

View File

@ -12198,6 +12198,7 @@ version(bindbc){
alias GLchar = char;
alias GLsizei = int;
alias GLfloat = float;
alias GLintptr = size_t;
alias GLsizeiptr = ptrdiff_t;
enum uint GL_STENCIL_BUFFER_BIT = 0x00000400;
@ -12312,6 +12313,8 @@ version(bindbc){
__gshared glbfn_glDeleteVertexArrays glDeleteVertexArrays_NVGLZ; alias glDeleteVertexArrays = glDeleteVertexArrays_NVGLZ;
alias glbfn_glGenerateMipmap = void function(GLenum);
__gshared glbfn_glGenerateMipmap glGenerateMipmap_NVGLZ; alias glGenerateMipmap = glGenerateMipmap_NVGLZ;
alias glbfn_glBufferSubData = void function(GLenum, GLintptr, GLsizeiptr, const(GLvoid)*);
__gshared glbfn_glBufferSubData glBufferSubData_NVGLZ; alias glBufferSubData = glBufferSubData_NVGLZ;
alias glbfn_glStencilMask = void function(GLuint);
__gshared glbfn_glStencilMask glStencilMask_NVGLZ; alias glStencilMask = glStencilMask_NVGLZ;
@ -12413,6 +12416,8 @@ version(bindbc){
if (glDeleteVertexArrays_NVGLZ is null) assert(0, `OpenGL function 'glDeleteVertexArrays' not found!`);
glGenerateMipmap_NVGLZ = cast(glbfn_glGenerateMipmap)glbindGetProcAddress(`glGenerateMipmap`);
if (glGenerateMipmap_NVGLZ is null) assert(0, `OpenGL function 'glGenerateMipmap' not found!`);
glBufferSubData_NVGLZ = cast(glbfn_glBufferSubData)glbindGetProcAddress(`glBufferSubData`);
if (glBufferSubData_NVGLZ is null) assert(0, `OpenGL function 'glBufferSubData' not found!`);
glStencilMask_NVGLZ = cast(glbfn_glStencilMask)glbindGetProcAddress(`glStencilMask`);
if (glStencilMask_NVGLZ is null) assert(0, `OpenGL function 'glStencilMask' not found!`);

45
png.d
View File

@ -130,7 +130,7 @@ MemoryImage imageFromPng(PNG* png) {
assert(0, "invalid png");
}
auto idataIdx = 0;
size_t idataIdx = 0;
auto file = LazyPngFile!(Chunk[])(png.chunks);
immutable(ubyte)[] previousLine;
@ -150,7 +150,7 @@ MemoryImage imageFromPng(PNG* png) {
}
// idata needs to be already sized for the image! width * height if indexed, width*height*4 if not.
void convertPngData(ubyte type, ubyte depth, const(ubyte)[] data, int width, ubyte[] idata, ref int idataIdx) {
void convertPngData(ubyte type, ubyte depth, const(ubyte)[] data, int width, ubyte[] idata, ref size_t idataIdx) {
ubyte consumeOne() {
ubyte ret = data[0];
data = data[1 .. $];
@ -300,7 +300,7 @@ PNG* pngFromImage(IndexedImage i) {
Chunk alpha;
if(i.hasAlpha) {
alpha.type = ['t', 'R', 'N', 'S'];
alpha.size = cast(int) i.palette.length;
alpha.size = cast(uint) i.palette.length;
alpha.payload.length = alpha.size;
}
@ -324,7 +324,7 @@ PNG* pngFromImage(IndexedImage i) {
addImageDatastreamToPng(i.data, png);
} else {
// gotta convert it
ubyte[] datastream = new ubyte[i.width * i.height * h.depth / 8]; // FIXME?
ubyte[] datastream = new ubyte[cast(size_t)i.width * i.height * h.depth / 8]; // FIXME?
int shift = 0;
switch(h.depth) {
@ -334,8 +334,8 @@ PNG* pngFromImage(IndexedImage i) {
case 4: shift = 4; break;
case 8: shift = 0; break;
}
int dsp = 0;
int dpos = 0;
size_t dsp = 0;
size_t dpos = 0;
bool justAdvanced;
for(int y = 0; y < i.height; y++) {
for(int x = 0; x < i.width; x++) {
@ -467,12 +467,12 @@ void writeImageToPngFile(in char[] filename, TrueColorImage image) {
h.height = image.height;
png = blankPNG(h);
auto bytesPerLine = h.width * 4;
size_t bytesPerLine = cast(size_t)h.width * 4;
if(h.type == 3)
bytesPerLine = h.width * 8 / h.depth;
bytesPerLine = cast(size_t)h.width * 8 / h.depth;
Chunk dat;
dat.type = ['I', 'D', 'A', 'T'];
int pos = 0;
size_t pos = 0;
auto compressor = new Compress();
@ -489,7 +489,8 @@ void writeImageToPngFile(in char[] filename, TrueColorImage image) {
com ~= cast(ubyte[]) compressor.flush();
dat.size = cast(int) com.length;
assert(com.length <= uint.max);
dat.size = cast(uint) com.length;
dat.payload = com;
dat.checksum = crc("IDAT", dat.payload);
@ -541,7 +542,7 @@ ubyte[] writePng(PNG* p) {
foreach(c; p.chunks)
a.length += c.size + 12;
}
uint pos;
size_t pos;
a[0..8] = p.header[0..8];
pos = 8;
@ -576,8 +577,8 @@ PngHeader getHeaderFromFile(string filename) {
if(data[0..8] != [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
throw new Exception("file " ~ filename ~ " is not a png");
auto pos = 8;
uint size;
size_t pos = 8;
size_t size;
size |= data[pos++] << 24;
size |= data[pos++] << 16;
size |= data[pos++] << 8;
@ -609,7 +610,7 @@ PNG* readPng(in ubyte[] data) {
if(p.header != [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
throw new Exception("not a png, header wrong");
uint pos = 8;
size_t pos = 8;
while(pos < data.length) {
Chunk n;
@ -648,7 +649,7 @@ PNG* blankPNG(PngHeader h) {
c.type = ['I', 'H', 'D', 'R'];
c.payload.length = 13;
int pos = 0;
size_t pos = 0;
c.payload[pos++] = h.width >> 24;
c.payload[pos++] = (h.width >> 16) & 0xff;
@ -684,31 +685,31 @@ void addImageDatastreamToPng(const(ubyte)[] data, PNG* png) {
PngHeader h = getHeader(png);
int bytesPerLine;
size_t bytesPerLine;
switch(h.type) {
case 0:
// FIXME: < 8 depth not supported here but should be
bytesPerLine = h.width * 1 * h.depth / 8;
bytesPerLine = cast(size_t)h.width * 1 * h.depth / 8;
break;
case 2:
bytesPerLine = h.width * 3 * h.depth / 8;
bytesPerLine = cast(size_t)h.width * 3 * h.depth / 8;
break;
case 3:
bytesPerLine = h.width * 1 * h.depth / 8;
bytesPerLine = cast(size_t)h.width * 1 * h.depth / 8;
break;
case 4:
// FIXME: < 8 depth not supported here but should be
bytesPerLine = h.width * 2 * h.depth / 8;
bytesPerLine = cast(size_t)h.width * 2 * h.depth / 8;
break;
case 6:
bytesPerLine = h.width * 4 * h.depth / 8;
bytesPerLine = cast(size_t)h.width * 4 * h.depth / 8;
break;
default: assert(0);
}
Chunk dat;
dat.type = ['I', 'D', 'A', 'T'];
int pos = 0;
size_t pos = 0;
const(ubyte)[] output;
while(pos+bytesPerLine <= data.length) {

View File

@ -1708,6 +1708,19 @@ struct RealTimeConsoleInput {
/// Check for input, waiting no longer than the number of milliseconds
bool timedCheckForInput(int milliseconds) {
if(inputQueue.length || timedCheckForInput_bypassingBuffer(timeout))
return true;
version(Posix)
if(interrupted || windowSizeChanged || hangedUp)
return true;
return false;
}
/* private */ bool anyInput_internal(int timeout = 0) {
return timedCheckForInput(timeout);
}
bool timedCheckForInput_bypassingBuffer(int milliseconds) {
version(Windows) {
auto response = WaitForSingleObject(terminal.hConsole, milliseconds);
if(response == 0)
@ -1733,15 +1746,6 @@ struct RealTimeConsoleInput {
}
}
/* private */ bool anyInput_internal() {
if(inputQueue.length || timedCheckForInput(0))
return true;
version(Posix)
if(interrupted || windowSizeChanged || hangedUp)
return true;
return false;
}
private dchar getchBuffer;
/// Get one key press from the terminal, discarding other