Angelscript/samples/asbuild/source/main.cpp

322 lines
8.0 KiB
C++

#include <iostream> // cout
#include <assert.h> // assert()
#include <string.h> // strstr()
#include <angelscript.h>
#include "../../../add_on/scriptbuilder/scriptbuilder.h"
#include "../../../add_on/scripthelper/scripthelper.h"
#include <stdio.h>
#include <stdlib.h>
#include <sstream>
#include <fstream>
#if defined(_MSC_VER) && !defined(_WIN32_WCE)
#include <direct.h>
#include <crtdbg.h>
#endif
#ifdef _WIN32_WCE
#include <windows.h> // For GetModuleFileName
#endif
using namespace std;
// Function prototypes
int ConfigureEngine(asIScriptEngine *engine, const char *configFile);
int CompileScript(asIScriptEngine *engine, const char *scriptFile);
int SaveBytecode(asIScriptEngine *engine, const char *outputFile);
static const char *GetCurrentDir(char *buf, size_t size);
void MessageCallback(const asSMessageInfo *msg, void *param)
{
const char *type = "ERR ";
if( msg->type == asMSGTYPE_WARNING )
type = "WARN";
else if( msg->type == asMSGTYPE_INFORMATION )
type = "INFO";
printf("%s (%d, %d) : %s : %s\n", msg->section, msg->row, msg->col, type, msg->message);
}
int main(int argc, char **argv)
{
#if defined(_MSC_VER)
// Turn on memory leak detection (use _CrtSetBreakAlloc to break at specific allocation)
_CrtSetDbgFlag(_CRTDBG_LEAK_CHECK_DF|_CRTDBG_ALLOC_MEM_DF);
_CrtSetReportMode(_CRT_ASSERT,_CRTDBG_MODE_FILE);
_CrtSetReportFile(_CRT_ASSERT,_CRTDBG_FILE_STDERR);
//_CrtSetBreakAlloc(6150);
#endif
int r;
if( argc < 4 )
{
cout << "Usage: " << endl;
cout << "asbuild <config file> <script file> <output>" << endl;
cout << " <config file> is the file with the application interface" << endl;
cout << " <script file> is the script file that should be compiled" << endl;
cout << " <output> is the name that the compiled script will be saved as" << endl;
return -1;
}
// Create the script engine
asIScriptEngine *engine = asCreateScriptEngine();
if( engine == 0 )
{
cout << "Failed to create script engine." << endl;
return -1;
}
// The script compiler will send any compiler messages to the callback
engine->SetMessageCallback(asFUNCTION(MessageCallback), 0, asCALL_CDECL);
// Configure the script engine with all the functions,
// and variables that the script should be able to use.
r = ConfigureEngine(engine, argv[1]);
if( r < 0 ) return -1;
// Compile the script code
r = CompileScript(engine, argv[2]);
if( r < 0 ) return -1;
// Save the bytecode
r = SaveBytecode(engine, argv[3]);
if( r < 0 ) return -1;
// Shut down the engine
engine->ShutDownAndRelease();
return 0;
}
#ifdef AS_CAN_USE_CPP11
// The string factory doesn't need to keep a specific order in the
// cache, so the unordered_map is faster than the ordinary map
#include <unordered_map> // std::unordered_map
BEGIN_AS_NAMESPACE
typedef unordered_map<string, int> map_t;
END_AS_NAMESPACE
#else
#include <map> // std::map
BEGIN_AS_NAMESPACE
typedef map<string, int> map_t;
END_AS_NAMESPACE
#endif
// Default string factory. Removes duplicate string constants
// This same implementation is provided in the scriptstdstring add-on
class CStdStringFactory : public asIStringFactory
{
public:
CStdStringFactory() {}
~CStdStringFactory()
{
// The script engine must release each string
// constant that it has requested
assert(stringCache.size() == 0);
}
const void *GetStringConstant(const char *data, asUINT length)
{
string str(data, length);
map_t::iterator it = stringCache.find(str);
if (it != stringCache.end())
it->second++;
else
it = stringCache.insert(map_t::value_type(str, 1)).first;
return reinterpret_cast<const void*>(&it->first);
}
int ReleaseStringConstant(const void *str)
{
if (str == 0)
return asERROR;
map_t::iterator it = stringCache.find(*reinterpret_cast<const string*>(str));
if (it == stringCache.end())
return asERROR;
it->second--;
if (it->second == 0)
stringCache.erase(it);
return asSUCCESS;
}
int GetRawStringData(const void *str, char *data, asUINT *length) const
{
if (str == 0)
return asERROR;
if (length)
*length = (asUINT)reinterpret_cast<const string*>(str)->length();
if (data)
memcpy(data, reinterpret_cast<const string*>(str)->c_str(), reinterpret_cast<const string*>(str)->length());
return asSUCCESS;
}
// TODO: Make sure the access to the string cache is thread safe
map_t stringCache;
};
CStdStringFactory stringFactory;
// This function will register the application interface,
// based on information read from a configuration file.
int ConfigureEngine(asIScriptEngine *engine, const char *configFile)
{
int r;
ifstream strm;
strm.open(configFile);
if( strm.fail() )
{
// Write a message to the engine's message callback
char buf[256];
string msg = "Failed to open config file in path: '" + string(GetCurrentDir(buf, 256)) + "'";
engine->WriteMessage(configFile, 0, 0, asMSGTYPE_ERROR, msg.c_str());
return -1;
}
// Configure the engine with the information from the file
r = ConfigEngineFromStream(engine, strm, configFile, &stringFactory);
if( r < 0 )
{
engine->WriteMessage(configFile, 0, 0, asMSGTYPE_ERROR, "Configuration failed");
return -1;
}
engine->WriteMessage(configFile, 0, 0, asMSGTYPE_INFORMATION, "Configuration successfully registered");
return 0;
}
int CompileScript(asIScriptEngine *engine, const char *scriptFile)
{
int r;
CScriptBuilder builder;
r = builder.StartNewModule(engine, "build");
if( r < 0 ) return -1;
r = builder.AddSectionFromFile(scriptFile);
if( r < 0 ) return -1;
r = builder.BuildModule();
if( r < 0 )
{
engine->WriteMessage(scriptFile, 0, 0, asMSGTYPE_ERROR, "Script failed to build");
return -1;
}
engine->WriteMessage(scriptFile, 0, 0, asMSGTYPE_INFORMATION, "Script successfully built");
return 0;
}
class CBytecodeStream : public asIBinaryStream
{
public:
CBytecodeStream() {f = 0;}
~CBytecodeStream() { if( f ) fclose(f); }
int Open(const char *filename)
{
if( f ) return -1;
#if _MSC_VER >= 1500
fopen_s(&f, filename, "wb");
#else
f = fopen(filename, "wb");
#endif
if( f == 0 ) return -1;
return 0;
}
int Write(const void *ptr, asUINT size)
{
if( size == 0 || f == 0 ) return 0;
fwrite(ptr, size, 1, f);
return 0;
}
int Read(void *, asUINT) { return -1; }
protected:
FILE *f;
};
int SaveBytecode(asIScriptEngine *engine, const char *outputFile)
{
CBytecodeStream stream;
int r = stream.Open(outputFile);
if( r < 0 )
{
engine->WriteMessage(outputFile, 0, 0, asMSGTYPE_ERROR, "Failed to open output file for writing");
return -1;
}
asIScriptModule *mod = engine->GetModule("build");
if( mod == 0 )
{
engine->WriteMessage(outputFile, 0, 0, asMSGTYPE_ERROR, "Failed to retrieve the compiled bytecode");
return -1;
}
r = mod->SaveByteCode(&stream);
if( r < 0 )
{
engine->WriteMessage(outputFile, 0, 0, asMSGTYPE_ERROR, "Failed to write the bytecode");
return -1;
}
engine->WriteMessage(outputFile, 0, 0, asMSGTYPE_INFORMATION, "Bytecode successfully saved");
return 0;
}
static const char *GetCurrentDir(char *buf, size_t size)
{
#ifdef _MSC_VER
#ifdef _WIN32_WCE
static TCHAR apppath[MAX_PATH] = TEXT("");
if (!apppath[0])
{
GetModuleFileName(NULL, apppath, MAX_PATH);
int appLen = _tcslen(apppath);
// Look for the last backslash in the path, which would be the end
// of the path itself and the start of the filename. We only want
// the path part of the exe's full-path filename
// Safety is that we make sure not to walk off the front of the
// array (in case the path is nothing more than a filename)
while (appLen > 1)
{
if (apppath[appLen-1] == TEXT('\\'))
break;
appLen--;
}
// Terminate the string after the trailing backslash
apppath[appLen] = TEXT('\0');
}
#ifdef _UNICODE
wcstombs(buf, apppath, min(size, wcslen(apppath)*sizeof(wchar_t)));
#else
memcpy(buf, apppath, min(size, strlen(apppath)));
#endif
return buf;
#else
return _getcwd(buf, (int)size);
#endif
#elif defined(__APPLE__)
return getcwd(buf, size);
#else
return "";
#endif
}