diff --git a/libphobos/libdruntime/Makefile.am b/libphobos/libdruntime/Makefile.am
index 8df0e1c43b91a145fc7eafb9beec8054e43c6430..efdae4c57861bd030bcd0b2cca78fe42c0585dfa 100644
--- a/libphobos/libdruntime/Makefile.am
+++ b/libphobos/libdruntime/Makefile.am
@@ -167,7 +167,7 @@ install-data-local:
 	done
 
 # Source file definitions. Boring stuff, auto-generated with
-# https://gist.github.com/jpf91/8ad1dbc9902d6ad876313f134c6527d1
+# libphobos/scripts/gen_druntime_sources.d
 # Can't use wildcards here:
 # https://www.gnu.org/software/automake/manual/html_node/Wildcards.html
 
diff --git a/libphobos/libdruntime/Makefile.in b/libphobos/libdruntime/Makefile.in
index 999064e5221c74e9660307fb628e387415eaf17f..1227c59b4854bfc179ea154a0184da8a153fe4f8 100644
--- a/libphobos/libdruntime/Makefile.in
+++ b/libphobos/libdruntime/Makefile.in
@@ -850,7 +850,7 @@ libgdruntime_convenience_la_DEPENDENCIES = $(libgdruntime_la_DEPENDENCIES)
 libgdruntime_convenience_la_LINK = $(libgdruntime_la_LINK)
 
 # Source file definitions. Boring stuff, auto-generated with
-# https://gist.github.com/jpf91/8ad1dbc9902d6ad876313f134c6527d1
+# libphobos/scripts/gen_druntime_sources.d
 # Can't use wildcards here:
 # https://www.gnu.org/software/automake/manual/html_node/Wildcards.html
 DRUNTIME_CSOURCES = core/stdc/errno_.c etc/valgrind/valgrind_.c
diff --git a/libphobos/scripts/.gitignore b/libphobos/scripts/.gitignore
new file mode 100644
index 0000000000000000000000000000000000000000..a5d300b6f6049937b2edcd01242715bb774205bb
--- /dev/null
+++ b/libphobos/scripts/.gitignore
@@ -0,0 +1,3 @@
+# Dub leaves built programs in this directory.
+gen_druntime_sources
+gen_phobos_sources
diff --git a/libphobos/scripts/README b/libphobos/scripts/README
new file mode 100644
index 0000000000000000000000000000000000000000..248324dddac36b9bda6ded0a9524b0409728d264
--- /dev/null
+++ b/libphobos/scripts/README
@@ -0,0 +1,28 @@
+The D language scripts in this directory are used during the syncing of
+libphobos with upstream.  They can be built with the command:
+
+    dub build --single name-of-script.d
+
+Or when dub isn't available:
+
+    gdc -o name-of-script name-of-script.d [any other dependencies].d
+
+Scripts:
+
+gen_druntime_sources.d
+
+    Generates source file definitions for libphobos/libdruntime/Makefile.am.
+    Ran from the libdruntime directory and append output to Makefile.am.
+
+    Example:
+
+	cd libdruntime && ../scripts/gen_druntime_sources >> Makefile.am
+
+gen_phobos_sources.d
+
+    Generates source file definitions for libphobos/src/Makefile.am.  Ran from
+    the libphobos/src directory and append output to Makefile.am.
+
+    Example:
+
+	cd src && ../scripts/gen_phobos_sources >> Makefile.am
diff --git a/libphobos/scripts/gen_druntime_sources.d b/libphobos/scripts/gen_druntime_sources.d
new file mode 100644
index 0000000000000000000000000000000000000000..ee01477a493f5b533f1356857a0e6160c5c29ff2
--- /dev/null
+++ b/libphobos/scripts/gen_druntime_sources.d
@@ -0,0 +1,92 @@
+#!/usr/bin/env dub
+/++dub.sdl:
+name "gen_druntime_sources"
++/
+// Written in the D programming language.
+import std.stdio;
+import std.file;
+import std.path;
+import std.range;
+import std.string;
+import std.algorithm;
+
+string[] filterList = [
+    "./Makefile.in", "./Makefile.am",
+    "./gcc/config.d.in", "./gcc/libbacktrace.d.in", "./gcc/drtstuff.c",
+    "./LICENSE.txt", "./MERGE",
+    "./rt/dylib_fixes.c"
+];
+
+struct Files
+{
+    string[] baseList, cppList;
+    string[][string] sysList;
+}
+
+void main(string[] args)
+{
+    Files[string] fileMap;
+
+    foreach(entry; ".".dirEntries(SpanMode.depth).filter!(a => !filterList.canFind(a)))
+    {
+        if (entry.name.startsWith("./config/"))
+            continue;
+
+        if(entry.isFile)
+        {
+            auto ext = entry.extension.empty ? "" : entry.extension[1 .. $];
+            if(!(ext in fileMap))
+                fileMap[ext] = Files.init;
+
+            string sentry = entry[2 .. $];
+
+            if(entry.name.startsWith("./core/stdcpp/"))
+                fileMap[ext].cppList ~= sentry;
+            else if(entry.name.startsWith("./core/sys/"))
+            {
+                auto components = entry.pathSplitter;
+                components.popFrontN(3);
+                fileMap[ext].sysList[components.front] ~= sentry;
+            }
+            else
+                fileMap[ext].baseList ~= sentry;
+        }
+    }
+
+    foreach(extEntry; fileMap.byKeyValue.array.sort!"a.key < b.key")
+    {
+        auto ext = extEntry.key;
+        auto value = extEntry.value;
+        writeList("DRUNTIME_" ~ ext.toUpper() ~ "SOURCES", value.baseList);
+        writeList("DRUNTIME_" ~ ext.toUpper() ~ "SOURCES_STDCXX", value.cppList);
+        foreach(entry; value.sysList.byKeyValue.array.sort!"a.key < b.key")
+        {
+            writeList("DRUNTIME_" ~ ext.toUpper() ~ "SOURCES_" ~ entry.key.toUpper(), entry.value);
+        }
+    }
+}
+
+void writeList(string name, string[] values, bool force = false)
+{
+    if (!force && values.empty)
+        return;
+
+    values = sort(values).array();
+    writeln();
+    writef("%s =", name);
+    size_t line = name.length + 3;
+    foreach(entry; values)
+    {
+        if(line + entry.length > 70)
+        {
+            line = 0;
+            writeln(` \`);
+            write('\t');
+        }
+        else
+            write(" ");
+        write(entry);
+        line += entry.length + 1;
+    }
+    writeln();
+}
diff --git a/libphobos/scripts/gen_phobos_sources.d b/libphobos/scripts/gen_phobos_sources.d
new file mode 100644
index 0000000000000000000000000000000000000000..661f5a9f76e66820848ff29e233993fa27168a0f
--- /dev/null
+++ b/libphobos/scripts/gen_phobos_sources.d
@@ -0,0 +1,116 @@
+#!/usr/bin/env dub
+/++dub.sdl:
+name "gen_phobos_sources"
++/
+// Written in the D programming language.
+import std.stdio;
+import std.file;
+import std.path;
+import std.range;
+import std.string;
+import std.algorithm;
+
+string[] filterList = [
+    "./Makefile.in", "./Makefile.am",
+    "./index.dd",
+    "./libgphobos.spec.in", "./drtstuff.spec",
+    "./LICENSE_1_0.txt", "./MERGE",
+    "./std/experimental/note.md"
+];
+
+struct Files
+{
+    string[] baseList;
+    string[][string] sysList;
+}
+
+void main(string[] args)
+{
+    Files[string] fileMap;
+
+    foreach (entry; "."
+                    .dirEntries(SpanMode.depth)
+                    .filter!(a => !filterList.canFind(a)))
+    {
+        if (entry.isFile)
+        {
+            auto ext = entry.extension.empty ? "" : entry.extension[1 .. $];
+            if (!(ext in fileMap))
+                fileMap[ext] = Files.init;
+
+            string sentry = entry[2 .. $];
+
+            if (entry.name.startsWith("./std/c/"))
+            {
+                if (entry.dirName == "./std/c")
+                {
+                    fileMap[ext].sysList["stdc"] ~= sentry;
+                }
+                else
+                {
+                    auto components = entry.pathSplitter;
+                    components.popFrontN(3);
+                    fileMap[ext].sysList[components.front] ~= sentry;
+                }
+            }
+            else
+                fileMap[ext].baseList ~= sentry;
+        }
+    }
+
+    writeln("if ENABLE_LIBDRUNTIME_ONLY");
+    foreach (extEntry; fileMap.byKeyValue.array.sort!"a.key < b.key")
+    {
+        auto ext = extEntry.key;
+        auto value = extEntry.value;
+        writeList("PHOBOS_" ~ ext.toUpper() ~ "SOURCES", [],
+                  !value.baseList.empty);
+        foreach (entry; value.sysList.byKeyValue.array.sort!"a.key < b.key")
+        {
+            string name = "PHOBOS_" ~ ext.toUpper() ~ "SOURCES_"
+                ~ entry.key.toUpper();
+            writeList(name, [], !entry.value.empty);
+        }
+    }
+    writeln();
+    writeln("else");
+    foreach (extEntry; fileMap.byKeyValue.array.sort!"a.key < b.key")
+    {
+        auto ext = extEntry.key;
+        auto value = extEntry.value;
+        writeList("PHOBOS_" ~ ext.toUpper() ~ "SOURCES", value.baseList);
+        foreach (entry; value.sysList.byKeyValue.array.sort!"a.key < b.key")
+        {
+            string name = "PHOBOS_" ~ ext.toUpper() ~ "SOURCES_"
+                ~ entry.key.toUpper();
+            writeList(name, entry.value);
+        }
+    }
+    writeln();
+    writeln("endif");
+}
+
+void writeList(string name, string[] values, bool force = false)
+{
+    if (!force && values.empty)
+        return;
+
+    values = sort(values).array();
+    writeln();
+    writef("%s =", name);
+    size_t line = name.length + 3;
+    foreach (entry; values)
+    {
+        if (line + entry.length > 70)
+        {
+            line = 0;
+            writeln(` \`);
+            write('\t');
+        }
+        else
+            write(" ");
+        write(entry);
+        line += entry.length + 1;
+    }
+    writeln();
+}
diff --git a/libphobos/src/Makefile.am b/libphobos/src/Makefile.am
index 763ca3beb59f5c25f176fdb92ae9ed5a63d1c906..ec2bda6833447675f6228176e148e4d03845021c 100644
--- a/libphobos/src/Makefile.am
+++ b/libphobos/src/Makefile.am
@@ -78,7 +78,7 @@ install-data-local:
 	done
 
 # Source file definitions. Boring stuff, auto-generated with
-# https://gist.github.com/jpf91/8744acebc9dcf1e9d1a35cdff20afbb2
+# libphobos/scripts/gen_phobos_sources.d
 # Can't use wildcards here:
 # https://www.gnu.org/software/automake/manual/html_node/Wildcards.html
 if ENABLE_LIBDRUNTIME_ONLY
diff --git a/libphobos/src/Makefile.in b/libphobos/src/Makefile.in
index dcf6425f86f79dde1a6be3314c80e0bdc0f19d05..d1512b9e4c95b69467035727b95822b6b5b7d1af 100644
--- a/libphobos/src/Makefile.in
+++ b/libphobos/src/Makefile.in
@@ -624,7 +624,7 @@ libgphobos_la_LINK = $(LIBTOOL) --tag=D $(libgphobos_la_LIBTOOLFLAGS) \
 
 
 # Source file definitions. Boring stuff, auto-generated with
-# https://gist.github.com/jpf91/8744acebc9dcf1e9d1a35cdff20afbb2
+# libphobos/scripts/gen_phobos_sources.d
 # Can't use wildcards here:
 # https://www.gnu.org/software/automake/manual/html_node/Wildcards.html
 @ENABLE_LIBDRUNTIME_ONLY_TRUE@PHOBOS_DSOURCES =