Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rsp support #32

Open
wants to merge 12 commits into
base: master
Choose a base branch
from
9 changes: 5 additions & 4 deletions source/dpp/runtime/app.d
Original file line number Diff line number Diff line change
Expand Up @@ -11,22 +11,23 @@ import dpp.from;
void run(in from!"dpp.runtime.options".Options options) @safe {
import std.stdio: File;
import std.exception: enforce;
import std.process: execute;
import std.process: spawnProcess, wait;
import std.array: join;
import std.file: remove;
import std.conv : text;

foreach(dppFileName; options.dppFileNames)
preprocess!File(options, dppFileName, options.toDFileName(dppFileName));

if(options.preprocessOnly) return;

const args = options.dlangCompiler ~ options.dlangCompilerArgs;
const res = execute(args);
enforce(res.status == 0, "Could not execute `" ~ args.join(" ") ~ "`:\n" ~ res.output);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no need to have this, if the target compiler can handle what's been given then we just pass it through.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's needed or else the compiler failing would cause d++ to return 0.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sorry, I commented in the wrong place. This line is still there, just moved down a bit.

const status = spawnProcess(args).wait();
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why spawnProcess.wait instad of execute?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So that the compiler output gets out regardless of what d++ thinks. pragma(msg,...) needs to get out even on success, as do warnings. Also, you want to see whatever output you might have got before (or during) an infinite loop in the compiler or ctfe, so we can't wait for the compiler to finish before printing, might as well let it do its own printing to std{out/err}.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Makes sense.

if(!options.keepDlangFiles) {
foreach(fileName; options.dFileNames)
remove(fileName);
}
enforce(status == 0, "Executing `" ~ args.join(" ") ~ "` failed with exit code\n" ~ status.text);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

moved this to the end so files get cleaned up even if things fail (user can obviously specify to keep the files if they want to inspect them)

}


Expand Down Expand Up @@ -108,7 +109,7 @@ private string preamble() @safe pure {
import core.stdc.config;
import core.stdc.stdarg: va_list;
struct __locale_data { int dummy; } // FIXME
#define __gnuc_va_list va_list
} ~ "#define __gnuc_va_list va_list\n" ~ q{
alias _Bool = bool;

struct dpp {
Expand Down
19 changes: 6 additions & 13 deletions source/dpp/runtime/options.d
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ struct Options {
import std.algorithm: map, filter, canFind, startsWith;
import std.array: array;
import std.conv: text;
import dpp.runtime.response : response_expand;

parseArgs(args);
if(earlyExit) return;
Expand All @@ -46,23 +47,15 @@ struct Options {
else
enforce(args.length >= 2, "Not enough arguments\n" ~ usage);

args = response_expand(args);

dppFileNames = args.filter!(a => a.extension == ".dpp").array;
enforce(dppFileNames.length != 0, "No .dpp input file specified\n" ~ usage);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is where I meant to write this:

no need to have this, if the target compiler can handle what's been given then we just pass it through.


// Remove the name of this binary and the name of the .dpp input file from args
// so that a D compiler can use the remaining entries.
dlangCompilerArgs =
args[1..$].filter!(a => a.extension != ".dpp").array ~
dFileNames;

// if no -of option is given, default to the name of the .dpp file
if(!dlangCompilerArgs.canFind!(a => a.startsWith("-of")) && !dlangCompilerArgs.canFind("-c"))
dlangCompilerArgs ~= "-of" ~
args.
filter!(a => a.extension == ".dpp" || a.extension == ".d")
.front
.stripExtension
~ exeExtension;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's no need to handle this here, if we don't mess with file ordering then we get dmd's behaviour for free.

dlangCompilerArgs = args[1..$]
.map!(a => a.extension == ".dpp" ? toDFileName(a) : a)
.array;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

avoiding re-ordering files on the command line.


includePaths = systemPaths ~ includePaths;
}
Expand Down
206 changes: 206 additions & 0 deletions source/dpp/runtime/response.d
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
module dpp.runtime.response;
import core.stdc.stdio;
import core.stdc.stdlib;
import core.stdc.string;

/// wrapper to make response_expand usable
/// with strings. Yes, it allocates a lot.
string[] response_expand(string[] args) @trusted
{
import std.algorithm : map;
import std.array : array;
import std.string : fromStringz;
import std.exception : enforce;

auto cargs = args
.map!toConstStringz
.array;
enforce(!response_expand(cargs), "expanding args failed");
return cargs
.map!(s => s.fromStringz.idup)
.array;
}

const(char)* toConstStringz(string s) @safe
{
auto r = new char[](s.length + 1);
size_t i = 0;
while (i < s.length)
{
r[i] = s[i];
if (r[i] == '\0')
break;
++i;
}
if (i == s.length)
r[i] = '\0';
return &r[0];
}

// ported from dmd's function of the same name.
// only modifications are to use builtin arrays
// and phobos in place of dmd's bespoke types
bool response_expand(ref const(char)*[] args)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was gonna port some dmd tests for this function as well, but..... I couldn't find them. I suspect they don't exist

{
import std.algorithm : remove;
import std.file : readText;
import std.array : insertInPlace;

const(char)* cp;
int recurse = 0;
for (size_t i = 0; i < args.length;)
{
cp = args[i];
if (*cp != '@')
{
++i;
continue;
}
args = args.remove(i);
char* buffer;
char* bufend;
cp++;
if (auto p = getenv(cp))
{
buffer = strdup(p);
if (!buffer)
goto noexpand;
bufend = buffer + strlen(buffer);
}
else
{
auto s = cp[0 .. strlen(cp)].readText!(char[]);
buffer = s.ptr;
bufend = buffer + s.length;
}
// The logic of this should match that in setargv()
int comment = 0;
for (auto p = buffer; p < bufend; p++)
{
char* d;
char c, lastc;
ubyte instring;
int num_slashes, non_slashes;
switch (*p)
{
case 26:
/* ^Z marks end of file */
goto L2;
case 0xD:
case '\n':
if (comment)
{
comment = 0;
}
goto case;
case 0:
case ' ':
case '\t':
continue;
// scan to start of argument
case '#':
comment = 1;
continue;
case '@':
if (comment)
{
continue;
}
recurse = 1;
goto default;
default:
/* start of new argument */
if (comment)
{
continue;
}
args.insertInPlace(i, p);
++i;
instring = 0;
c = 0;
num_slashes = 0;
for (d = p; 1; p++)
{
lastc = c;
if (p >= bufend)
{
*d = 0;
goto L2;
}
c = *p;
switch (c)
{
case '"':
/*
Yes this looks strange,but this is so that we are
MS Compatible, tests have shown that:
\\\\"foo bar" gets passed as \\foo bar
\\\\foo gets passed as \\\\foo
\\\"foo gets passed as \"foo
and \"foo gets passed as "foo in VC!
*/
non_slashes = num_slashes % 2;
num_slashes = num_slashes / 2;
for (; num_slashes > 0; num_slashes--)
{
d--;
*d = '\0';
}
if (non_slashes)
{
*(d - 1) = c;
}
else
{
instring ^= 1;
}
break;
case 26:
*d = 0; // terminate argument
goto L2;
case 0xD:
// CR
c = lastc;
continue;
// ignore
case '@':
recurse = 1;
goto Ladd;
case ' ':
case '\t':
if (!instring)
{
case '\n':
case 0:
*d = 0; // terminate argument
goto Lnextarg;
}
goto default;
default:
Ladd:
if (c == '\\')
num_slashes++;
else
num_slashes = 0;
*d++ = c;
break;
}
}
break;
}
Lnextarg:
}
L2:
}
if (recurse)
{
/* Recursively expand @filename */
if (response_expand(args))
goto noexpand;
}
return false; /* success */
noexpand:
/* error */
/* BUG: any file buffers are not free'd */
return true;
}
16 changes: 0 additions & 16 deletions tests/it/issues.d
Original file line number Diff line number Diff line change
Expand Up @@ -165,22 +165,6 @@ import it;
);
}

@Tags("issue")
@("14")
@safe unittest {
import dpp.runtime.options: Options;
with(immutable IncludeSandbox()) {

writeFile("foo.h",
q{
typedef int foo;
});

runPreprocessOnly("foo.h").shouldThrowWithMessage(
"No .dpp input file specified\n" ~ Options.usage);
}
}

@Tags("issue", "preprocessor")
@("22.0")
@safe unittest {
Expand Down
1 change: 1 addition & 0 deletions tests/test_main.d
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ int main(string[] args) {

// unit tests
"ut.type",
"ut.response",

"it.issues",

Expand Down
20 changes: 20 additions & 0 deletions tests/ut/response.d
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
module ut.response;

import dpp.test;
import dpp.runtime.response;

@("toConstStringz")
@safe unittest
{
import std.algorithm : until;
foreach (s; [null, "", "a", "áåâà", "fd\0fds"])
{
// make sure we're not being saved by string
// literals being null-terminated
auto cstr = (s ~ 'a')[0 .. $ - 1]
.toConstStringz;
static assert(is(typeof(cstr) == const(char)*));
(() @trusted => cstr[0 .. s.length + 1])().until('\0')
.shouldEqual(s.until('\0'));
}
}