https://github.com/shader-slang/slang
Tip revision: 21a14cb4e0d578bc4f8a460016269a1199cac0da authored by Tim Foley on 06 July 2017, 16:52:53 UTC
Merge pull request #53 from tfoleyNV/cross-compilation
Merge pull request #53 from tfoleyNV/cross-compilation
Tip revision: 21a14cb
slang.cpp
#include "../../slang.h"
#include "../core/slang-io.h"
#include "../slang/slang-stdlib.h"
#include "parameter-binding.h"
#include "../slang/parser.h"
#include "../slang/preprocessor.h"
#include "../slang/reflection.h"
#include "syntax-visitors.h"
#include "../slang/type-layout.h"
#ifdef _WIN32
#define WIN32_LEAN_AND_MEAN
#define NOMINMAX
#include <Windows.h>
#undef WIN32_LEAN_AND_MEAN
#undef NOMINMAX
#endif
namespace Slang {
static void stdlibDiagnosticCallback(
char const* message,
void* userData)
{
fputs(message, stderr);
fflush(stderr);
#ifdef WIN32
OutputDebugStringA(message);
#endif
}
class Session
{
public:
bool useCache = false;
String cacheDir;
RefPtr<Scope> slangLanguageScope;
RefPtr<Scope> hlslLanguageScope;
RefPtr<Scope> glslLanguageScope;
List<RefPtr<ProgramSyntaxNode>> loadedModuleCode;
Session(bool /*pUseCache*/, String /*pCacheDir*/)
{
// Initialize global state
// TODO: move this into the session instead
BasicExpressionType::Init();
// Create scopes for various language builtins.
//
// TODO: load these on-demand to avoid parsing
// stdlib code for languages the user won't use.
slangLanguageScope = new Scope();
hlslLanguageScope = new Scope();
hlslLanguageScope->parent = slangLanguageScope;
glslLanguageScope = new Scope();
glslLanguageScope->parent = slangLanguageScope;
addBuiltinSource(slangLanguageScope, "stdlib", SlangStdLib::GetCode());
addBuiltinSource(glslLanguageScope, "glsl", getGLSLLibraryCode());
}
~Session()
{
// We need to clean up the strings for the standard library
// code that we might have allocated and loaded into static
// variables (TODO: don't use `static` variables for this stuff)
SlangStdLib::Finalize();
// Ditto for our type represnetation stuff
ExpressionType::Finalize();
}
void addBuiltinSource(
RefPtr<Scope> const& scope,
String const& path,
String const& source);
};
struct IncludeHandlerImpl : IncludeHandler
{
CompileRequest* request;
virtual IncludeResult TryToFindIncludeFile(
String const& pathToInclude,
String const& pathIncludedFrom,
String* outFoundPath,
String* outFoundSource) override
{
String path = Path::Combine(Path::GetDirectoryName(pathIncludedFrom), pathToInclude);
if (File::Exists(path))
{
*outFoundPath = path;
*outFoundSource = File::ReadAllText(path);
request->mDependencyFilePaths.Add(path);
return IncludeResult::Found;
}
for (auto & dir : request->searchDirectories)
{
path = Path::Combine(dir.path, pathToInclude);
if (File::Exists(path))
{
*outFoundPath = path;
*outFoundSource = File::ReadAllText(path);
request->mDependencyFilePaths.Add(path);
return IncludeResult::Found;
}
}
return IncludeResult::NotFound;
}
};
void CompileRequest::parseTranslationUnit(
TranslationUnitRequest* translationUnit)
{
IncludeHandlerImpl includeHandler;
includeHandler.request = this;
RefPtr<Scope> languageScope;
switch (translationUnit->sourceLanguage)
{
case SourceLanguage::HLSL:
languageScope = mSession->hlslLanguageScope;
break;
case SourceLanguage::GLSL:
languageScope = mSession->glslLanguageScope;
break;
case SourceLanguage::Slang:
default:
languageScope = mSession->slangLanguageScope;
break;
}
Dictionary<String, String> combinedPreprocessorDefinitions;
for(auto& def : preprocessorDefinitions)
combinedPreprocessorDefinitions.Add(def.Key, def.Value);
for(auto& def : translationUnit->preprocessorDefinitions)
combinedPreprocessorDefinitions.Add(def.Key, def.Value);
RefPtr<ProgramSyntaxNode> translationUnitSyntax = new ProgramSyntaxNode();
translationUnit->SyntaxNode = translationUnitSyntax;
for (auto sourceFile : translationUnit->sourceFiles)
{
auto sourceFilePath = sourceFile->path;
String source = sourceFile->content;
auto tokens = preprocessSource(
source,
sourceFilePath,
&mSink,
&includeHandler,
combinedPreprocessorDefinitions,
translationUnit);
parseSourceFile(
translationUnit,
tokens,
&mSink,
sourceFilePath,
languageScope);
}
}
void CompileRequest::checkAllTranslationUnits()
{
for( auto& translationUnit : translationUnits )
{
checkTranslationUnit(translationUnit.Ptr());
}
}
int CompileRequest::executeActionsInner()
{
// Do some cleanup on settings specified by user.
// In particular, we want to propagate flags from the overall request down to
// each translation unit.
for( auto& translationUnit : translationUnits )
{
translationUnit->compileFlags |= compileFlags;
// However, the "no checking" flag shouldn't be applied to
// any translation unit that is native Slang code.
if( translationUnit->sourceLanguage == SourceLanguage::Slang )
{
translationUnit->compileFlags &= ~SLANG_COMPILE_FLAG_NO_CHECKING;
}
}
#if 0
// If we are being asked to do pass-through, then we need to do that here...
if (passThrough != PassThroughMode::None)
{
for (auto& translationUnitOptions : Options.translationUnits)
{
switch (translationUnitOptions.sourceLanguage)
{
// We can pass-through code written in a native shading language
case SourceLanguage::GLSL:
case SourceLanguage::HLSL:
break;
// All other translation units need to be skipped
default:
continue;
}
auto sourceFile = translationUnitOptions.sourceFiles[0];
auto sourceFilePath = sourceFile->path;
String source = sourceFile->content;
auto translationUnitResult = passThrough(
source,
sourceFilePath,
Options,
translationUnitOptions);
mResult.translationUnits.Add(translationUnitResult);
}
return 0;
}
#endif
// We only do parsing and semantic checking if we *aren't* doing
// a pass-through compilation.
//
// Note that we *do* perform output generation as normal in pass-through mode.
if( passThrough == PassThroughMode::None )
{
// Parse everything from the input files requested
for (auto& translationUnit : translationUnits)
{
parseTranslationUnit(translationUnit.Ptr());
}
if (mSink.GetErrorCount() != 0)
return 1;
// Perform semantic checking on the whole collection
checkAllTranslationUnits();
if (mSink.GetErrorCount() != 0)
return 1;
// Now do shader parameter binding generation, which
// needs to be performed globally.
generateParameterBindings(this);
if (mSink.GetErrorCount() != 0)
return 1;
}
// Generate output code, in whatever format was requested
generateOutput(this);
if (mSink.GetErrorCount() != 0)
return 1;
return 0;
}
// Act as expected of the API-based compiler
int CompileRequest::executeActions()
{
int err = executeActionsInner();
mDiagnosticOutput = mSink.outputBuffer.ProduceString();
return err;
}
int CompileRequest::addTranslationUnit(SourceLanguage language, String const& name)
{
int result = translationUnits.Count();
RefPtr<TranslationUnitRequest> translationUnit = new TranslationUnitRequest();
translationUnit->compileRequest = this;
translationUnit->sourceLanguage = SourceLanguage(language);
translationUnits.Add(translationUnit);
return result;
}
void CompileRequest::addTranslationUnitSourceString(
int translationUnitIndex,
String const& path,
String const& source)
{
RefPtr<SourceFile> sourceFile = new SourceFile();
sourceFile->path = path;
sourceFile->content = source;
translationUnits[translationUnitIndex]->sourceFiles.Add(sourceFile);
}
void CompileRequest::addTranslationUnitSourceFile(
int translationUnitIndex,
String const& path)
{
String source;
try
{
source = File::ReadAllText(path);
}
catch (...)
{
// Emit a diagnostic!
mSink.diagnose(
CodePosition(0, 0, 0, path),
Diagnostics::cannotOpenFile,
path);
return;
}
addTranslationUnitSourceString(
translationUnitIndex,
path,
source);
mDependencyFilePaths.Add(path);
}
int CompileRequest::addEntryPoint(
int translationUnitIndex,
String const& name,
Profile profile)
{
RefPtr<EntryPointRequest> entryPoint = new EntryPointRequest();
entryPoint->compileRequest = this;
entryPoint->name = name;
entryPoint->profile = profile;
entryPoint->translationUnitIndex = translationUnitIndex;
auto translationUnit = translationUnits[translationUnitIndex].Ptr();
translationUnit->entryPoints.Add(entryPoint);
int result = entryPoints.Count();
entryPoints.Add(entryPoint);
return result;
}
RefPtr<ProgramSyntaxNode> CompileRequest::loadModule(
String const& name,
String const& path,
String const& source,
CodePosition const& loc)
{
RefPtr<TranslationUnitRequest> translationUnit = new TranslationUnitRequest();
translationUnit->compileRequest = this;
// We don't want to use the same options that the user specified
// for loading modules on-demand. In particular, we always want
// semantic checking to be enabled.
//
// TODO: decide which options, if any, should be inherited.
RefPtr<SourceFile> sourceFile = new SourceFile();
sourceFile->path = path;
sourceFile->content = source;
translationUnit->sourceFiles.Add(sourceFile);
parseTranslationUnit(translationUnit.Ptr());
// TODO: handle errors
checkTranslationUnit(translationUnit.Ptr());
// Skip code generation
//
RefPtr<ProgramSyntaxNode> moduleDecl = translationUnit->SyntaxNode;
mapPathToLoadedModule.Add(path, moduleDecl);
mapNameToLoadedModules.Add(name, moduleDecl);
loadedModulesList.Add(moduleDecl);
return moduleDecl;
}
void CompileRequest::handlePoundImport(
String const& path,
TokenList const& tokens)
{
RefPtr<TranslationUnitRequest> translationUnit = new TranslationUnitRequest();
translationUnit->compileRequest = this;
// Imported code is always native Slang code
RefPtr<Scope> languageScope = mSession->slangLanguageScope;
RefPtr<ProgramSyntaxNode> translationUnitSyntax = new ProgramSyntaxNode();
translationUnit->SyntaxNode = translationUnitSyntax;
parseSourceFile(
translationUnit.Ptr(),
tokens,
&mSink,
path,
languageScope);
// TODO: handle errors
checkTranslationUnit(translationUnit.Ptr());
// Skip code generation
//
RefPtr<ProgramSyntaxNode> moduleDecl = translationUnit->SyntaxNode;
// TODO: It is a bit broken here that we use the module path,
// as the "name" when registering things, but this saves
// us the trouble of trying to special-case things when
// checking an `import` down the road.
mapNameToLoadedModules.Add(path, moduleDecl);
mapPathToLoadedModule.Add(path, moduleDecl);
loadedModulesList.Add(moduleDecl);
}
RefPtr<ProgramSyntaxNode> CompileRequest::findOrImportModule(
String const& name,
CodePosition const& loc)
{
// Have we already loaded a module matching this name?
// If so, return it.
RefPtr<ProgramSyntaxNode> moduleDecl;
if (mapNameToLoadedModules.TryGetValue(name, moduleDecl))
return moduleDecl;
// Derive a file name for the module, by taking the given
// identifier, replacing all occurences of `_` with `-`,
// and then appending `.slang`.
//
// For example, `foo_bar` becomes `foo-bar.slang`.
StringBuilder sb;
for (auto c : name)
{
if (c == '_')
c = '-';
sb.Append(c);
}
sb.Append(".slang");
String fileName = sb.ProduceString();
// Next, try to find the file of the given name,
// using our ordinary include-handling logic.
IncludeHandlerImpl includeHandler;
includeHandler.request = this;
String pathIncludedFrom = loc.FileName;
String foundPath;
String foundSource;
IncludeResult includeResult = includeHandler.TryToFindIncludeFile(fileName, pathIncludedFrom, &foundPath, &foundSource);
switch( includeResult )
{
case IncludeResult::NotFound:
case IncludeResult::Error:
{
this->mSink.diagnose(loc, Diagnostics::cannotFindFile, fileName);
mapNameToLoadedModules[name] = nullptr;
return nullptr;
}
break;
default:
break;
}
// Maybe this was loaded previously via `#import`
if (mapPathToLoadedModule.TryGetValue(foundPath, moduleDecl))
return moduleDecl;
// We've found a file that we can load for the given module, so
// go ahead and perform the module-load action
return loadModule(
name,
foundPath,
foundSource,
loc);
}
RefPtr<ProgramSyntaxNode> findOrImportModule(
CompileRequest* request,
String const& name,
CodePosition const& loc)
{
return request->findOrImportModule(name, loc);
}
void Session::addBuiltinSource(
RefPtr<Scope> const& scope,
String const& path,
String const& source)
{
RefPtr<CompileRequest> compileRequest = new CompileRequest(this);
auto translationUnitIndex = compileRequest->addTranslationUnit(SourceLanguage::Slang, path);
compileRequest->addTranslationUnitSourceString(
translationUnitIndex,
path,
source);
int err = compileRequest->executeActions();
if (err)
{
fprintf(stderr, "%s", compileRequest->mDiagnosticOutput.Buffer());
#ifdef _WIN32
OutputDebugStringA(compileRequest->mDiagnosticOutput.Buffer());
#endif
assert(!"error in stdlib");
}
// Extract the AST for the code we just parsed
auto syntax = compileRequest->translationUnits[translationUnitIndex]->SyntaxNode;
// HACK(tfoley): mark all declarations in the "stdlib" so
// that we can detect them later (e.g., so we don't emit them)
for (auto m : syntax->Members)
{
auto fromStdLibModifier = new FromStdLibModifier();
fromStdLibModifier->next = m->modifiers.first;
m->modifiers.first = fromStdLibModifier;
}
// Add the resulting code to the appropriate scope
if (!scope->containerDecl)
{
// We are the first chunk of code to be loaded for this scope
scope->containerDecl = syntax.Ptr();
}
else
{
// We need to create a new scope to link into the whole thing
auto subScope = new Scope();
subScope->containerDecl = syntax.Ptr();
subScope->nextSibling = scope->nextSibling;
scope->nextSibling = subScope;
}
// We need to retain this AST so that we can use it in other code
// (Note that the `Scope` type does not retain the AST it points to)
loadedModuleCode.Add(syntax);
}
}
// implementation of C interface
#define SESSION(x) reinterpret_cast<Slang::Session *>(x)
#define REQ(x) reinterpret_cast<Slang::CompileRequest*>(x)
SLANG_API SlangSession* spCreateSession(const char * cacheDir)
{
return reinterpret_cast<SlangSession *>(new Slang::Session((cacheDir ? true : false), cacheDir));
}
SLANG_API void spDestroySession(
SlangSession* session)
{
if(!session) return;
delete SESSION(session);
}
SLANG_API void spAddBuiltins(
SlangSession* session,
char const* sourcePath,
char const* sourceString)
{
auto s = SESSION(session);
s->addBuiltinSource(
// TODO(tfoley): Add ability to directly new builtins to the approriate scope
s->slangLanguageScope,
sourcePath,
sourceString);
}
SLANG_API SlangCompileRequest* spCreateCompileRequest(
SlangSession* session)
{
auto s = SESSION(session);
auto req = new Slang::CompileRequest(s);
return reinterpret_cast<SlangCompileRequest*>(req);
}
/*!
@brief Destroy a compile request.
*/
SLANG_API void spDestroyCompileRequest(
SlangCompileRequest* request)
{
if(!request) return;
auto req = REQ(request);
delete req;
}
SLANG_API void spSetCompileFlags(
SlangCompileRequest* request,
SlangCompileFlags flags)
{
REQ(request)->compileFlags = flags;
}
SLANG_API void spSetCodeGenTarget(
SlangCompileRequest* request,
int target)
{
REQ(request)->Target = (Slang::CodeGenTarget)target;
}
SLANG_API void spSetPassThrough(
SlangCompileRequest* request,
SlangPassThrough passThrough)
{
REQ(request)->passThrough = Slang::PassThroughMode(passThrough);
}
SLANG_API void spSetDiagnosticCallback(
SlangCompileRequest* request,
SlangDiagnosticCallback callback,
void const* userData)
{
if(!request) return;
auto req = REQ(request);
req->mSink.callback = callback;
req->mSink.callbackUserData = (void*) userData;
}
SLANG_API void spAddSearchPath(
SlangCompileRequest* request,
const char* path)
{
REQ(request)->searchDirectories.Add(Slang::SearchDirectory(path));
}
SLANG_API void spAddPreprocessorDefine(
SlangCompileRequest* request,
const char* key,
const char* value)
{
REQ(request)->preprocessorDefinitions[key] = value;
}
SLANG_API char const* spGetDiagnosticOutput(
SlangCompileRequest* request)
{
if(!request) return 0;
auto req = REQ(request);
return req->mDiagnosticOutput.begin();
}
// New-fangled compilation API
SLANG_API int spAddTranslationUnit(
SlangCompileRequest* request,
SlangSourceLanguage language,
char const* name)
{
auto req = REQ(request);
return req->addTranslationUnit(
Slang::SourceLanguage(language),
name ? name : "");
}
SLANG_API void spTranslationUnit_addPreprocessorDefine(
SlangCompileRequest* request,
int translationUnitIndex,
const char* key,
const char* value)
{
auto req = REQ(request);
req->translationUnits[translationUnitIndex]->preprocessorDefinitions[key] = value;
}
SLANG_API void spAddTranslationUnitSourceFile(
SlangCompileRequest* request,
int translationUnitIndex,
char const* path)
{
if(!request) return;
auto req = REQ(request);
if(!path) return;
if(translationUnitIndex < 0) return;
if(translationUnitIndex >= req->translationUnits.Count()) return;
req->addTranslationUnitSourceFile(
translationUnitIndex,
path);
}
// Add a source string to the given translation unit
SLANG_API void spAddTranslationUnitSourceString(
SlangCompileRequest* request,
int translationUnitIndex,
char const* path,
char const* source)
{
if(!request) return;
auto req = REQ(request);
if(!source) return;
if(translationUnitIndex < 0) return;
if(translationUnitIndex >= req->translationUnits.Count()) return;
if(!path) path = "";
req->addTranslationUnitSourceString(
translationUnitIndex,
path,
source);
}
SLANG_API SlangProfileID spFindProfile(
SlangSession* session,
char const* name)
{
return Slang::Profile::LookUp(name).raw;
}
SLANG_API int spAddEntryPoint(
SlangCompileRequest* request,
int translationUnitIndex,
char const* name,
SlangProfileID profile)
{
if(!request) return -1;
auto req = REQ(request);
if(!name) return -1;
if(translationUnitIndex < 0) return -1;
if(translationUnitIndex >= req->translationUnits.Count()) return -1;
return req->addEntryPoint(
translationUnitIndex,
name,
Slang::Profile(Slang::Profile::RawVal(profile)));
}
// Compile in a context that already has its translation units specified
SLANG_API int spCompile(
SlangCompileRequest* request)
{
auto req = REQ(request);
int anyErrors = req->executeActions();
return anyErrors;
}
SLANG_API int
spGetDependencyFileCount(
SlangCompileRequest* request)
{
if(!request) return 0;
auto req = REQ(request);
return req->mDependencyFilePaths.Count();
}
/** Get the path to a file this compilation dependend on.
*/
SLANG_API char const*
spGetDependencyFilePath(
SlangCompileRequest* request,
int index)
{
if(!request) return 0;
auto req = REQ(request);
return req->mDependencyFilePaths[index].begin();
}
SLANG_API int
spGetTranslationUnitCount(
SlangCompileRequest* request)
{
auto req = REQ(request);
return req->translationUnits.Count();
}
// Get the output code associated with a specific translation unit
SLANG_API char const* spGetTranslationUnitSource(
SlangCompileRequest* request,
int translationUnitIndex)
{
auto req = REQ(request);
return req->translationUnits[translationUnitIndex]->result.outputSource.Buffer();
}
SLANG_API char const* spGetEntryPointSource(
SlangCompileRequest* request,
int entryPointIndex)
{
auto req = REQ(request);
return req->entryPoints[entryPointIndex]->result.outputSource.Buffer();
}
// Reflection API
SLANG_API SlangReflection* spGetReflection(
SlangCompileRequest* request)
{
if( !request ) return 0;
auto req = REQ(request);
return (SlangReflection*) req->layout.Ptr();
}
// ... rest of reflection API implementation is in `Reflection.cpp`