Commit 13c0c5e2 authored by twanvl's avatar twanvl

Fixed a nasty order of destruction bug, where the memory pool for ScriptInts...

Fixed a nasty order of destruction bug, where the memory pool for ScriptInts was destroyed before the PackageManager
parent 98fcbbdd
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
#include <data/field.hpp> #include <data/field.hpp>
#include <data/field/text.hpp> #include <data/field/text.hpp>
#include <util/error.hpp>
// ----------------------------------------------------------------------------- : Field // ----------------------------------------------------------------------------- : Field
...@@ -50,7 +51,8 @@ shared_ptr<Field> read_new<Field>(Reader& reader) { ...@@ -50,7 +51,8 @@ shared_ptr<Field> read_new<Field>(Reader& reader) {
if (type == _("text")) { if (type == _("text")) {
return new_shared<TextField>(); return new_shared<TextField>();
} else { } else {
throw "TODO"; //return new_shared<TextField>();
throw ParseError(_("Unsupported field type: '") + type + _("'"));
} }
} }
......
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
// ----------------------------------------------------------------------------- : Includes // ----------------------------------------------------------------------------- : Includes
#include <util/prec.hpp> #include <util/prec.hpp>
#include <util/io/package_manager.hpp>
#include <data/game.hpp> #include <data/game.hpp>
#include <data/set.hpp> #include <data/set.hpp>
#include <data/settings.hpp> #include <data/settings.hpp>
...@@ -51,6 +52,7 @@ bool MSE::OnInit() { ...@@ -51,6 +52,7 @@ bool MSE::OnInit() {
} catch (...) { } catch (...) {
handle_error(InternalError(_("An unexpected exception occurred, this is a bug!\nPlease save your work (use 'save as' to so you don't overwrite things),\n and restart Magic Set Editor.\n\nYou can leave a bug report on http://magicseteditor.sourceforge.net/")), false); handle_error(InternalError(_("An unexpected exception occurred, this is a bug!\nPlease save your work (use 'save as' to so you don't overwrite things),\n and restart Magic Set Editor.\n\nYou can leave a bug report on http://magicseteditor.sourceforge.net/")), false);
} }
packages.destroy();
return false; return false;
} }
...@@ -58,6 +60,7 @@ bool MSE::OnInit() { ...@@ -58,6 +60,7 @@ bool MSE::OnInit() {
int MSE::OnExit() { int MSE::OnExit() {
settings.write(); settings.write();
packages.destroy();
return 0; return 0;
} }
......
//+----------------------------------------------------------------------------+
//| Description: Magic Set Editor - Program to make Magic (tm) cards |
//| Copyright: (C) 2001 - 2006 Twan van Laarhoven |
//| License: GNU General Public License 2 or later (see file COPYING) |
//+----------------------------------------------------------------------------+
// ----------------------------------------------------------------------------- : Includes
#include <script/scriptable.hpp>
#include <script/context.hpp>
#include <script/parser.hpp>
#include <script/script.hpp>
#include <script/value.hpp>
// ----------------------------------------------------------------------------- : Store
void store(const ScriptValueP& val, String& var) { var = static_cast<String>(*val); }
void store(const ScriptValueP& val, int& var) { var = *val; }
void store(const ScriptValueP& val, double& var) { var = *val; }
void store(const ScriptValueP& val, bool& var) { var = static_cast<int>(*val); }
void store(const ScriptValueP& val, Defaultable<String>& var) { var.assign(*val); }
// ----------------------------------------------------------------------------- : OptionalScript
OptionalScript::~OptionalScript() {}
ScriptValueP OptionalScript::invoke(Context& ctx) {
if (script) {
return ctx.eval(*script);
} else {
return script_nil;
}
}
// custom reflection, different for each type
template <> void Reader::handle(OptionalScript& os) {
handle(os.unparsed);
// read the script
os.script = parse(os.unparsed);
}
template <> void Writer::handle(const OptionalScript& os) {
handle(os.unparsed);
}
template <> void GetMember::handle(const OptionalScript& os) {
// no members
}
template <> void GetMember::store(const OptionalScript& os) {
// reflect as the script itself
if (os.script) {
store(os.script);
} else {
store(script_nil);
}
}
//+----------------------------------------------------------------------------+
//| Description: Magic Set Editor - Program to make Magic (tm) cards |
//| Copyright: (C) 2001 - 2006 Twan van Laarhoven |
//| License: GNU General Public License 2 or later (see file COPYING) |
//+----------------------------------------------------------------------------+
#ifndef HEADER_SCRIPT_SCRIPTABLE
#define HEADER_SCRIPT_SCRIPTABLE
// ----------------------------------------------------------------------------- : Includes
#include <util/prec.hpp>
#include <util/reflect.hpp>
#include <util/defaultable.hpp>
DECLARE_INTRUSIVE_POINTER_TYPE(Script);
class Context;
// ----------------------------------------------------------------------------- : Store
/// Store a ScriptValue in a variable
void store(const ScriptValueP& val, String& var);
void store(const ScriptValueP& val, int& var);
void store(const ScriptValueP& val, double& var);
void store(const ScriptValueP& val, bool& var);
void store(const ScriptValueP& val, Defaultable<String>& var);
// ----------------------------------------------------------------------------- : OptionalScript
/// An optional script,
class OptionalScript {
public:
~OptionalScript();
/// Is the script set?
inline operator bool() { return !!script; }
/// Invoke the script, return the result, or script_nil if there is no script
ScriptValueP invoke(Context& ctx);
/// Invoke the script on a value
/** Assigns the result to value if it has changed.
* Returns true if the value has changed.
*/
template <typename T>
bool invokeOn(Context& ctx, T& value) {
if (script) {
T new_value;
store(new_value, script->invoke(ctx));
if (value != new_value) {
value = new_value;
return true;
}
}
return false;
}
private:
ScriptP script; ///< The script, may be null if there is no script
String unparsed; ///< Unparsed script, for writing back to a file
DECLARE_REFLECTION();
};
// ----------------------------------------------------------------------------- : Scriptable
/// A script that defines a calculation to find a value
/** NOTE: reading MUST happen inside a block */
template <typename T>
class Scriptable {
public:
Scriptable() : value() {}
Scriptable(const T& value) : value(value) {}
inline operator const T& () const { return value; }
inline bool isScripted() const { return script; }
// Updates the value by executing the script, returns true if the value has changed
inline bool update(Context& ctx) {
return script.invokeOn(ctx, value);
}
private:
OptionalScript script; ///< The optional script
T value; ///< The scripted value
DECLARE_REFLECTION();
};
// ----------------------------------------------------------------------------- : EOF
#endif
...@@ -56,6 +56,8 @@ ScriptValueP rangeIterator(int start, int end) { ...@@ -56,6 +56,8 @@ ScriptValueP rangeIterator(int start, int end) {
// ----------------------------------------------------------------------------- : Integers // ----------------------------------------------------------------------------- : Integers
#define USE_POOL_ALLOCATOR
// Integer values // Integer values
class ScriptInt : public ScriptValue { class ScriptInt : public ScriptValue {
public: public:
...@@ -66,23 +68,59 @@ class ScriptInt : public ScriptValue { ...@@ -66,23 +68,59 @@ class ScriptInt : public ScriptValue {
virtual operator double() const { return value; } virtual operator double() const { return value; }
virtual operator int() const { return value; } virtual operator int() const { return value; }
protected: protected:
#ifdef USE_POOL_ALLOCATOR
virtual void destroy() { virtual void destroy() {
boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::free(this); boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::free(this);
} }
#endif
private: private:
int value; int value;
}; };
#if defined(USE_POOL_ALLOCATOR) && !defined(USE_INTRUSIVE_PTR)
// deallocation function for pool allocated integers
void destroy_value(ScriptInt* v) {
boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::free(v);
}
#endif
ScriptValueP toScript(int v) { ScriptValueP toScript(int v) {
// return new_intrusive1<ScriptInt>(v); #ifdef USE_POOL_ALLOCATOR
return ScriptValueP( #ifdef USE_INTRUSIVE_PTR
new(boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::malloc()) return ScriptValueP(
ScriptInt(v)); new(boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::malloc())
ScriptInt(v));
#else
return ScriptValueP(
new(boost::singleton_pool<ScriptValue, sizeof(ScriptInt)>::malloc())
ScriptInt(v),
destroy_value); // deallocation function
#endif
#else
return new_intrusive1<ScriptInt>(v);
#endif
} }
// ----------------------------------------------------------------------------- : Booleans
// Boolean values
class ScriptBool : public ScriptValue {
public:
ScriptBool(bool v) : value(v) {}
virtual ScriptType type() const { return SCRIPT_INT; }
virtual String typeName() const { return _("boolean"); }
virtual operator String() const { return value ? _("true") : _("false"); }
virtual operator int() const { return value; }
private:
bool value;
};
// use integers to represent true/false // use integers to represent true/false
ScriptValueP script_true = toScript((int)true); /* NOTE: previous versions used ScriptInts as booleans, this gives problems
ScriptValueP script_false = toScript((int)false); * when we use a pool allocator for them, because the pool is destroyed before these globals.
*/
ScriptValueP script_true (new ScriptBool(true));
ScriptValueP script_false(new ScriptBool(false));
// ----------------------------------------------------------------------------- : Doubles // ----------------------------------------------------------------------------- : Doubles
......
...@@ -11,7 +11,6 @@ ...@@ -11,7 +11,6 @@
#include <util/prec.hpp> #include <util/prec.hpp>
#include <util/error.hpp> #include <util/error.hpp>
#include <boost/intrusive_ptr.hpp>
class Context; class Context;
class Dependency; class Dependency;
...@@ -46,7 +45,11 @@ enum ScriptType ...@@ -46,7 +45,11 @@ enum ScriptType
/// Actual values are derived types /// Actual values are derived types
class ScriptValue { class ScriptValue {
public: public:
inline ScriptValue() : refCount(0) {} inline ScriptValue()
#ifdef USE_INTRUSIVE_PTR
: refCount(0)
#endif
{}
virtual ~ScriptValue() {} virtual ~ScriptValue() {}
/// Information on the type of this value /// Information on the type of this value
...@@ -85,22 +88,26 @@ class ScriptValue { ...@@ -85,22 +88,26 @@ class ScriptValue {
virtual void destroy() { virtual void destroy() {
delete this; delete this;
} }
#ifdef USE_INTRUSIVE_PTR
private: private:
volatile LONG refCount; volatile LONG refCount;
friend void intrusive_ptr_add_ref(ScriptValue*); friend void intrusive_ptr_add_ref(ScriptValue*);
friend void intrusive_ptr_release(ScriptValue*); friend void intrusive_ptr_release(ScriptValue*);
#endif
}; };
inline void intrusive_ptr_add_ref(ScriptValue* p) { #ifdef USE_INTRUSIVE_PTR
//p->refCount += 1; inline void intrusive_ptr_add_ref(ScriptValue* p) {
InterlockedIncrement(&p->refCount); //p->refCount += 1;
} InterlockedIncrement(&p->refCount);
inline void intrusive_ptr_release(ScriptValue* p) { }
if (InterlockedDecrement(&p->refCount) == 0) { inline void intrusive_ptr_release(ScriptValue* p) {
//if (--p->refCount == 0) { if (InterlockedDecrement(&p->refCount) == 0) {
p->destroy(); //if (--p->refCount == 0) {
p->destroy();
}
} }
} #endif
extern ScriptValueP script_nil; ///< The preallocated nil value extern ScriptValueP script_nil; ///< The preallocated nil value
extern ScriptValueP script_true; ///< The preallocated true value extern ScriptValueP script_true; ///< The preallocated true value
......
...@@ -64,6 +64,14 @@ class ParseError : public Error { ...@@ -64,6 +64,14 @@ class ParseError : public Error {
inline ParseError(const String& str) : Error(str) {} inline ParseError(const String& str) : Error(str) {}
}; };
/// Parse error in a particular file
class FileParseError : public ParseError {
public:
inline FileParseError(const String& err, const String& file) :
ParseError(_("Error while parsing file '") + file + _("':\n") + err)
{}
};
/// Parse error in a script /// Parse error in a script
class ScriptParseError : public ParseError { class ScriptParseError : public ParseError {
public: public:
......
...@@ -24,9 +24,9 @@ template <> void GetMember::store(const unsigned int& v) { value = toScript((int ...@@ -24,9 +24,9 @@ template <> void GetMember::store(const unsigned int& v) { value = toScript((int
template <> void GetMember::store(const double& v) { value = toScript(v); } template <> void GetMember::store(const double& v) { value = toScript(v); }
template <> void GetMember::store(const bool& v) { value = toScript(v); } template <> void GetMember::store(const bool& v) { value = toScript(v); }
template <> void GetMember::store(const ScriptValueP& v) { value = v; }
template <> void GetMember::store(const ScriptP& v) { value = v; }
template <> void GetMember::store(const Vector2D& v) { template <> void GetMember::store(const Vector2D& v) {
value = toScript(String::Format(_("(%.10lf,%.10lf)"), v.x, v.y)); value = toScript(String::Format(_("(%.10lf,%.10lf)"), v.x, v.y));
} }
void GetMember::store(const ScriptValueP& v) { value = v; }
void GetMember::store(const ScriptP& v) { value = v; }
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include <util/prec.hpp> #include <util/prec.hpp>
DECLARE_INTRUSIVE_POINTER_TYPE(ScriptValue); DECLARE_INTRUSIVE_POINTER_TYPE(ScriptValue);
DECLARE_INTRUSIVE_POINTER_TYPE(Script);
inline void intrusive_ptr_add_ref(ScriptValue* p); inline void intrusive_ptr_add_ref(ScriptValue* p);
inline void intrusive_ptr_release(ScriptValue* p); inline void intrusive_ptr_release(ScriptValue* p);
...@@ -54,6 +55,8 @@ class GetMember { ...@@ -54,6 +55,8 @@ class GetMember {
template <typename T> void store(const shared_ptr<T>& pointer) { template <typename T> void store(const shared_ptr<T>& pointer) {
value = toScript(pointer); value = toScript(pointer);
} }
void store(const ScriptValueP&);
void store(const ScriptP&);
private: private:
const String& targetName; ///< The name we are looking for const String& targetName; ///< The name we are looking for
......
...@@ -379,11 +379,11 @@ Packaged::Packaged() { ...@@ -379,11 +379,11 @@ Packaged::Packaged() {
void Packaged::open(const String& package) { void Packaged::open(const String& package) {
Package::open(package); Package::open(package);
Reader reader(openIn(typeName()), absoluteFilename() + _("/") + typeName()); Reader reader(openIn(typeName()), absoluteFilename() + _("/") + typeName());
// try { try {
reader.handle(*this); reader.handle(*this);
// } catch (const ParseError& err) { } catch (const ParseError& err) {
// throw FileParseError(err.what(), filename+_("/")+typeName()); // more detailed message throw FileParseError(err.what(), absoluteFilename() + _("/") + typeName()); // more detailed message
// } }
} }
void Packaged::save() { void Packaged::save() {
//writeFile(thisT().fileName, thisT()); //writeFile(thisT().fileName, thisT());
......
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
// ----------------------------------------------------------------------------- : PackageManager // ----------------------------------------------------------------------------- : PackageManager
String programDir() { String program_dir() {
return _("."); //TODO return _("."); //TODO
} }
...@@ -20,18 +20,22 @@ PackageManager packages; ...@@ -20,18 +20,22 @@ PackageManager packages;
PackageManager::PackageManager() { PackageManager::PackageManager() {
// determine data directory // determine data directory
dataDirectory = programDir(); data_directory = program_dir();
// check if this is the actual data directory, especially during debugging, // check if this is the actual data directory, especially during debugging,
// the data may be higher up: // the data may be higher up:
// exe path = mse/build/debug/mse.exe // exe path = mse/build/debug/mse.exe
// data path = mse/data // data path = mse/data
while (!wxDirExists(dataDirectory + _("/data"))) { while (!wxDirExists(data_directory + _("/data"))) {
String d = dataDirectory; String d = data_directory;
dataDirectory = wxPathOnly(dataDirectory); data_directory = wxPathOnly(data_directory);
if (d == dataDirectory) { if (d == data_directory) {
// we are at the root -> 'data' not found anywhere in the path -> fatal error // we are at the root -> 'data' not found anywhere in the path -> fatal error
throw Error(_("The MSE data files can not be found, there should be a directory called 'data' with these files")); throw Error(_("The MSE data files can not be found, there should be a directory called 'data' with these files"));
} }
} }
dataDirectory += _("/data"); data_directory += _("/data");
} }
void PackageManager::destroy() {
loaded_packages.clear();
}
\ No newline at end of file
...@@ -28,11 +28,11 @@ class PackageManager { ...@@ -28,11 +28,11 @@ class PackageManager {
/// Open a package with the specified name (including extension) /// Open a package with the specified name (including extension)
template <typename T> template <typename T>
shared_ptr<T> open(const String& name) { shared_ptr<T> open(const String& name) {
wxFileName fn(dataDirectory + _("/") + name); wxFileName fn(data_directory + _("/") + name);
fn.Normalize(); fn.Normalize();
String filename = fn.GetFullPath(); String filename = fn.GetFullPath();
// Is this package already loaded? // Is this package already loaded?
PackagedP& p = loadedPackages[filename]; PackagedP& p = loaded_packages[filename];
shared_ptr<T> typedP = dynamic_pointer_cast<T>(p); shared_ptr<T> typedP = dynamic_pointer_cast<T>(p);
if (typedP) { if (typedP) {
return typedP; return typedP;
...@@ -48,9 +48,14 @@ class PackageManager { ...@@ -48,9 +48,14 @@ class PackageManager {
/// the type of package is determined by its extension! /// the type of package is determined by its extension!
PackagedP openAnyPackage(const String& filename); PackagedP openAnyPackage(const String& filename);
/// Empty the list of packages.
/** This function MUST be called before the program terminates, otherwise
* we could get into fights with pool allocators used by ScriptValues */
void destroy();
private: private:
map<String, PackagedP> loadedPackages; map<String, PackagedP> loaded_packages;
String dataDirectory; String data_directory;
}; };
/// The global PackageManager instance /// The global PackageManager instance
......
...@@ -92,11 +92,11 @@ inline shared_ptr<T> new_shared7(const A0& a0, const A1& a1, const A2& a2, const ...@@ -92,11 +92,11 @@ inline shared_ptr<T> new_shared7(const A0& a0, const A1& a1, const A2& a2, const
#else #else
#define DECLARE_INTRUSIVE_POINTER_TYPE DECLARE_POINTER_TYPE #define DECLARE_INTRUSIVE_POINTER_TYPE DECLARE_POINTER_TYPE
#define intrusive_ptr smart_ptr #define intrusive_ptr shared_ptr
#define new_intrusive new_smart #define new_intrusive new_shared
#define new_intrusive1 new_smart1 #define new_intrusive1 new_shared1
#define new_intrusive2 new_smart2 #define new_intrusive2 new_shared2
#define new_intrusive3 new_smart3 #define new_intrusive3 new_shared3
#endif #endif
// ----------------------------------------------------------------------------- : EOF // ----------------------------------------------------------------------------- : EOF
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment