ErrorReporting: set context for STG loading
Ensure the STG absolute path can be propagated to all files triggered by STG loading, including the delayed files and proxied files. This allows us to attribute errors to the correct scenery path.
This commit is contained in:
parent
f4dfa854ec
commit
802ce5ad23
@ -79,6 +79,11 @@ void setErrorContextCallback(ContextCallback cb)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ErrorReportContext::ErrorReportContext(const std::string& key, const std::string& value)
|
ErrorReportContext::ErrorReportContext(const std::string& key, const std::string& value)
|
||||||
|
{
|
||||||
|
add(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ErrorReportContext::add(const std::string& key, const std::string& value)
|
||||||
{
|
{
|
||||||
if (static_contextCallback) {
|
if (static_contextCallback) {
|
||||||
_keys.push_back(key);
|
_keys.push_back(key);
|
||||||
|
@ -49,7 +49,7 @@ enum class LoadFailure {
|
|||||||
translated error messages for the user.
|
translated error messages for the user.
|
||||||
*/
|
*/
|
||||||
enum class ErrorCode {
|
enum class ErrorCode {
|
||||||
MissingShader,
|
LoadEffectsShaders,
|
||||||
LoadingTexture,
|
LoadingTexture,
|
||||||
XMLModelLoad,
|
XMLModelLoad,
|
||||||
ThreeDModelLoad, // AC3D, OBJ, etc
|
ThreeDModelLoad, // AC3D, OBJ, etc
|
||||||
@ -60,8 +60,7 @@ enum class ErrorCode {
|
|||||||
XMLLoadCommand,
|
XMLLoadCommand,
|
||||||
AircraftSystems,
|
AircraftSystems,
|
||||||
InputDeviceConfig,
|
InputDeviceConfig,
|
||||||
AITrafficSchedule,
|
AITrafficSchedule
|
||||||
AirportDataLoad // ground-net, jetways, etc
|
|
||||||
};
|
};
|
||||||
/**
|
/**
|
||||||
@brief Define an error-reporting context value, for the duration of this
|
@brief Define an error-reporting context value, for the duration of this
|
||||||
@ -80,6 +79,8 @@ public:
|
|||||||
*/
|
*/
|
||||||
ErrorReportContext(const ContextMap& context = {});
|
ErrorReportContext(const ContextMap& context = {});
|
||||||
|
|
||||||
|
void add(const std::string& key, const std::string& value);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@brief allowed delayed add of values
|
@brief allowed delayed add of values
|
||||||
*/
|
*/
|
||||||
|
@ -846,7 +846,7 @@ void reload_shaders()
|
|||||||
} else {
|
} else {
|
||||||
SG_LOG(SG_INPUT, SG_ALERT, "Could not locate shader: " << fileName);
|
SG_LOG(SG_INPUT, SG_ALERT, "Could not locate shader: " << fileName);
|
||||||
simgear::reportFailure(simgear::LoadFailure::NotFound,
|
simgear::reportFailure(simgear::LoadFailure::NotFound,
|
||||||
simgear::ErrorCode::MissingShader,
|
simgear::ErrorCode::LoadEffectsShaders,
|
||||||
"Reload: couldn't find shader:" + sitr->first.first);
|
"Reload: couldn't find shader:" + sitr->first.first);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -940,7 +940,7 @@ void ShaderProgramBuilder::buildAttribute(Effect* effect, Pass* pass,
|
|||||||
if (fileName.empty())
|
if (fileName.empty())
|
||||||
{
|
{
|
||||||
SG_LOG(SG_INPUT, SG_ALERT, "Could not locate shader" << shaderName);
|
SG_LOG(SG_INPUT, SG_ALERT, "Could not locate shader" << shaderName);
|
||||||
simgear::reportFailure(simgear::LoadFailure::NotFound, simgear::ErrorCode::MissingShader,
|
simgear::reportFailure(simgear::LoadFailure::NotFound, simgear::ErrorCode::LoadEffectsShaders,
|
||||||
"Couldn't locate shader:" + shaderName, sg_location{shaderName});
|
"Couldn't locate shader:" + shaderName, sg_location{shaderName});
|
||||||
|
|
||||||
throw BuilderException(string("couldn't find shader ") +
|
throw BuilderException(string("couldn't find shader ") +
|
||||||
@ -968,7 +968,7 @@ void ShaderProgramBuilder::buildAttribute(Effect* effect, Pass* pass,
|
|||||||
if (loadShaderFromUTF8File(shader, fileName)) {
|
if (loadShaderFromUTF8File(shader, fileName)) {
|
||||||
if (!program->addShader(shader.get())) {
|
if (!program->addShader(shader.get())) {
|
||||||
simgear::reportFailure(simgear::LoadFailure::BadData,
|
simgear::reportFailure(simgear::LoadFailure::BadData,
|
||||||
simgear::ErrorCode::MissingShader,
|
simgear::ErrorCode::LoadEffectsShaders,
|
||||||
"Program::addShader failed",
|
"Program::addShader failed",
|
||||||
SGPath::fromUtf8(fileName));
|
SGPath::fromUtf8(fileName));
|
||||||
}
|
}
|
||||||
@ -976,7 +976,7 @@ void ShaderProgramBuilder::buildAttribute(Effect* effect, Pass* pass,
|
|||||||
shaderMap.insert(ShaderMap::value_type(skey, shader));
|
shaderMap.insert(ShaderMap::value_type(skey, shader));
|
||||||
} else {
|
} else {
|
||||||
simgear::reportFailure(simgear::LoadFailure::BadData,
|
simgear::reportFailure(simgear::LoadFailure::BadData,
|
||||||
simgear::ErrorCode::MissingShader,
|
simgear::ErrorCode::LoadEffectsShaders,
|
||||||
"Failed to read shader source code",
|
"Failed to read shader source code",
|
||||||
SGPath::fromUtf8(fileName));
|
SGPath::fromUtf8(fileName));
|
||||||
}
|
}
|
||||||
|
@ -60,12 +60,13 @@
|
|||||||
|
|
||||||
#include <simgear/scene/tgdb/VPBTechnique.hxx>
|
#include <simgear/scene/tgdb/VPBTechnique.hxx>
|
||||||
|
|
||||||
#include <simgear/structure/exception.hxx>
|
#include <simgear/debug/ErrorReportingCallback.hxx>
|
||||||
#include <simgear/props/props.hxx>
|
|
||||||
#include <simgear/props/props_io.hxx>
|
|
||||||
#include <simgear/props/condition.hxx>
|
|
||||||
#include <simgear/io/sg_file.hxx>
|
#include <simgear/io/sg_file.hxx>
|
||||||
#include <simgear/misc/lru_cache.hxx>
|
#include <simgear/misc/lru_cache.hxx>
|
||||||
|
#include <simgear/props/condition.hxx>
|
||||||
|
#include <simgear/props/props.hxx>
|
||||||
|
#include <simgear/props/props_io.hxx>
|
||||||
|
#include <simgear/structure/exception.hxx>
|
||||||
|
|
||||||
#include "BoundingVolumeBuildVisitor.hxx"
|
#include "BoundingVolumeBuildVisitor.hxx"
|
||||||
#include "model.hxx"
|
#include "model.hxx"
|
||||||
@ -293,6 +294,11 @@ ModelRegistry::readImage(const string& fileName,
|
|||||||
|
|
||||||
const SGReaderWriterOptions* sgoptC = dynamic_cast<const SGReaderWriterOptions*>(opt);
|
const SGReaderWriterOptions* sgoptC = dynamic_cast<const SGReaderWriterOptions*>(opt);
|
||||||
|
|
||||||
|
simgear::ErrorReportContext ec;
|
||||||
|
if (sgoptC && sgoptC->getModelData()) {
|
||||||
|
ec.addFromMap(sgoptC->getModelData()->getErrorContext());
|
||||||
|
}
|
||||||
|
|
||||||
if (cache_active && (!sgoptC || sgoptC->getLoadOriginHint() != SGReaderWriterOptions::LoadOriginHint::ORIGIN_SPLASH_SCREEN)) {
|
if (cache_active && (!sgoptC || sgoptC->getLoadOriginHint() != SGReaderWriterOptions::LoadOriginHint::ORIGIN_SPLASH_SCREEN)) {
|
||||||
if (fileExtension != "dds" && fileExtension != "gz") {
|
if (fileExtension != "dds" && fileExtension != "gz") {
|
||||||
|
|
||||||
@ -716,6 +722,17 @@ ReaderWriter::ReadResult
|
|||||||
ModelRegistry::readNode(const string& fileName,
|
ModelRegistry::readNode(const string& fileName,
|
||||||
const Options* opt)
|
const Options* opt)
|
||||||
{
|
{
|
||||||
|
// propogate error context from the caller
|
||||||
|
simgear::ErrorReportContext ec;
|
||||||
|
auto sgopt = dynamic_cast<const SGReaderWriterOptions*>(opt);
|
||||||
|
if (sgopt) {
|
||||||
|
if (sgopt->getModelData()) {
|
||||||
|
ec.addFromMap(sgopt->getModelData()->getErrorContext());
|
||||||
|
}
|
||||||
|
|
||||||
|
ec.addFromMap(sgopt->getErrorContext());
|
||||||
|
}
|
||||||
|
|
||||||
ReaderWriter::ReadResult res;
|
ReaderWriter::ReadResult res;
|
||||||
CallbackMap::iterator iter
|
CallbackMap::iterator iter
|
||||||
= nodeCallbackMap.find(getFileExtension(fileName));
|
= nodeCallbackMap.find(getFileExtension(fileName));
|
||||||
|
@ -24,11 +24,12 @@
|
|||||||
#include <simgear/scene/util/SplicingVisitor.hxx>
|
#include <simgear/scene/util/SplicingVisitor.hxx>
|
||||||
#include <simgear/scene/util/SGReaderWriterOptions.hxx>
|
#include <simgear/scene/util/SGReaderWriterOptions.hxx>
|
||||||
|
|
||||||
#include <simgear/structure/exception.hxx>
|
#include <simgear/debug/ErrorReportingCallback.hxx>
|
||||||
#include <simgear/structure/Singleton.hxx>
|
#include <simgear/props/condition.hxx>
|
||||||
#include <simgear/props/props.hxx>
|
#include <simgear/props/props.hxx>
|
||||||
#include <simgear/props/props_io.hxx>
|
#include <simgear/props/props_io.hxx>
|
||||||
#include <simgear/props/condition.hxx>
|
#include <simgear/structure/Singleton.hxx>
|
||||||
|
#include <simgear/structure/exception.hxx>
|
||||||
|
|
||||||
#include "model.hxx"
|
#include "model.hxx"
|
||||||
|
|
||||||
@ -364,6 +365,8 @@ ref_ptr<Node> instantiateMaterialEffects(osg::Node* modelGroup,
|
|||||||
} else {
|
} else {
|
||||||
effect = DefaultEffect::instance()->getEffect();
|
effect = DefaultEffect::instance()->getEffect();
|
||||||
SG_LOG( SG_TERRAIN, SG_ALERT, "Unable to get effect for " << options->getMaterialName());
|
SG_LOG( SG_TERRAIN, SG_ALERT, "Unable to get effect for " << options->getMaterialName());
|
||||||
|
simgear::reportFailure(simgear::LoadFailure::NotFound, simgear::ErrorCode::LoadEffectsShaders,
|
||||||
|
"Unable to get effect for material:" + options->getMaterialName());
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
effect = DefaultEffect::instance()->getEffect();
|
effect = DefaultEffect::instance()->getEffect();
|
||||||
|
@ -139,13 +139,6 @@ SGModelLib::loadModel(const string &path,
|
|||||||
opt->setPropertyNode(prop_root ? prop_root: static_propRoot.get());
|
opt->setPropertyNode(prop_root ? prop_root: static_propRoot.get());
|
||||||
opt->setModelData(data);
|
opt->setModelData(data);
|
||||||
|
|
||||||
// establish the error report context so we can attribute any load
|
|
||||||
// errors correctly
|
|
||||||
simgear::ErrorReportContext ec;
|
|
||||||
if (data) {
|
|
||||||
ec.addFromMap(data->getErrorContext());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (load2DPanels) {
|
if (load2DPanels) {
|
||||||
opt->setLoadPanel(static_panelFunc);
|
opt->setLoadPanel(static_panelFunc);
|
||||||
}
|
}
|
||||||
@ -168,11 +161,6 @@ SGModelLib::loadDeferredModel(const string &path, SGPropertyNode *prop_root,
|
|||||||
proxyNode->setLoadingExternalReferenceMode(osg::ProxyNode::DEFER_LOADING_TO_DATABASE_PAGER);
|
proxyNode->setLoadingExternalReferenceMode(osg::ProxyNode::DEFER_LOADING_TO_DATABASE_PAGER);
|
||||||
proxyNode->setFileName(0, path);
|
proxyNode->setFileName(0, path);
|
||||||
|
|
||||||
simgear::ErrorReportContext ec;
|
|
||||||
if (data) {
|
|
||||||
ec.addFromMap(data->getErrorContext());
|
|
||||||
}
|
|
||||||
|
|
||||||
osg::ref_ptr<SGReaderWriterOptions> opt;
|
osg::ref_ptr<SGReaderWriterOptions> opt;
|
||||||
opt = SGReaderWriterOptions::copyOrCreate(osgDB::Registry::instance()->getOptions());
|
opt = SGReaderWriterOptions::copyOrCreate(osgDB::Registry::instance()->getOptions());
|
||||||
opt->getDatabasePathList().push_front( osgDB::getFilePath(path) );
|
opt->getDatabasePathList().push_front( osgDB::getFilePath(path) );
|
||||||
@ -204,11 +192,6 @@ SGModelLib::loadPagedModel(SGPropertyNode *prop_root, SGModelData *data, SGModel
|
|||||||
unsigned int simple_models = 0;
|
unsigned int simple_models = 0;
|
||||||
osg::PagedLOD *plod = new osg::PagedLOD;
|
osg::PagedLOD *plod = new osg::PagedLOD;
|
||||||
|
|
||||||
simgear::ErrorReportContext ec;
|
|
||||||
if (data) {
|
|
||||||
ec.addFromMap(data->getErrorContext());
|
|
||||||
}
|
|
||||||
|
|
||||||
osg::ref_ptr<SGReaderWriterOptions> opt;
|
osg::ref_ptr<SGReaderWriterOptions> opt;
|
||||||
opt = SGReaderWriterOptions::copyOrCreate(osgDB::Registry::instance()->getOptions());
|
opt = SGReaderWriterOptions::copyOrCreate(osgDB::Registry::instance()->getOptions());
|
||||||
opt->setPropertyNode(prop_root ? prop_root: static_propRoot.get());
|
opt->setPropertyNode(prop_root ? prop_root: static_propRoot.get());
|
||||||
|
@ -177,7 +177,7 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
proxy->setName("proxyNode");
|
proxy->setName("proxyNode");
|
||||||
proxy->setLoadingExternalReferenceMode(osg::ProxyNode::DEFER_LOADING_TO_DATABASE_PAGER);
|
proxy->setLoadingExternalReferenceMode(osg::ProxyNode::DEFER_LOADING_TO_DATABASE_PAGER);
|
||||||
proxy->setFileName(0, o._name);
|
proxy->setFileName(0, o._name);
|
||||||
proxy->setDatabaseOptions(o._options.get());
|
proxy->setDatabaseOptions(o._options);
|
||||||
|
|
||||||
// Give the node some values so the Quadtree builder has
|
// Give the node some values so the Quadtree builder has
|
||||||
// a BoundingBox to work with prior to the model being loaded.
|
// a BoundingBox to work with prior to the model being loaded.
|
||||||
@ -186,6 +186,7 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
proxy->setCenterMode(osg::ProxyNode::UNION_OF_BOUNDING_SPHERE_AND_USER_DEFINED);
|
proxy->setCenterMode(osg::ProxyNode::UNION_OF_BOUNDING_SPHERE_AND_USER_DEFINED);
|
||||||
node = proxy;
|
node = proxy;
|
||||||
} else {
|
} else {
|
||||||
|
ErrorReportContext ec("terrain-stg", o._errorLocation.utf8Str());
|
||||||
node = osgDB::readRefNodeFile(o._name, o._options.get());
|
node = osgDB::readRefNodeFile(o._name, o._options.get());
|
||||||
if (!node.valid()) {
|
if (!node.valid()) {
|
||||||
SG_LOG(SG_TERRAIN, SG_ALERT, o._errorLocation << ": Failed to load "
|
SG_LOG(SG_TERRAIN, SG_ALERT, o._errorLocation << ": Failed to load "
|
||||||
@ -228,6 +229,8 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
virtual osgDB::ReaderWriter::ReadResult
|
virtual osgDB::ReaderWriter::ReadResult
|
||||||
readNode(const std::string&, const osgDB::Options*)
|
readNode(const std::string&, const osgDB::Options*)
|
||||||
{
|
{
|
||||||
|
ErrorReportContext ec("terrain-bucket", _bucket.gen_index_str());
|
||||||
|
|
||||||
STGObjectsQuadtree quadtree((GetModelLODCoord()), (AddModelLOD()));
|
STGObjectsQuadtree quadtree((GetModelLODCoord()), (AddModelLOD()));
|
||||||
quadtree.buildQuadTree(_objectStaticList.begin(), _objectStaticList.end());
|
quadtree.buildQuadTree(_objectStaticList.begin(), _objectStaticList.end());
|
||||||
osg::ref_ptr<osg::Group> group = quadtree.getRoot();
|
osg::ref_ptr<osg::Group> group = quadtree.getRoot();
|
||||||
@ -539,6 +542,7 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
opt->setInstantiateEffects(false);
|
opt->setInstantiateEffects(false);
|
||||||
_ObjectStatic obj;
|
_ObjectStatic obj;
|
||||||
|
|
||||||
|
opt->addErrorContext("terrain-stg", absoluteFileName.utf8Str());
|
||||||
obj._errorLocation = absoluteFileName;
|
obj._errorLocation = absoluteFileName;
|
||||||
obj._token = token;
|
obj._token = token;
|
||||||
obj._name = name;
|
obj._name = name;
|
||||||
@ -582,6 +586,8 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
_ObjectStatic obj;
|
_ObjectStatic obj;
|
||||||
|
|
||||||
opt->setInstantiateEffects(false);
|
opt->setInstantiateEffects(false);
|
||||||
|
opt->addErrorContext("terrain-stg", absoluteFileName.utf8Str());
|
||||||
|
|
||||||
if (SGPath(name).lower_extension() == "ac") {
|
if (SGPath(name).lower_extension() == "ac") {
|
||||||
// Generate material/Effects lookups for raw models, i.e.
|
// Generate material/Effects lookups for raw models, i.e.
|
||||||
// those not wrapped in an XML while will include Effects
|
// those not wrapped in an XML while will include Effects
|
||||||
@ -680,7 +686,7 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
std::string terrain_name = string("terrain ").append(bucket.gen_index_str());
|
std::string terrain_name = string("terrain ").append(bucket.gen_index_str());
|
||||||
terrainGroup->setName(terrain_name);
|
terrainGroup->setName(terrain_name);
|
||||||
|
|
||||||
simgear::ErrorReportContext ec{"bucket", bucket.gen_index_str()};
|
simgear::ErrorReportContext ec{"terrain-bucket", bucket.gen_index_str()};
|
||||||
|
|
||||||
bool vpb_active = SGSceneFeatures::instance()->getVPBActive();
|
bool vpb_active = SGSceneFeatures::instance()->getVPBActive();
|
||||||
if (vpb_active) {
|
if (vpb_active) {
|
||||||
@ -717,6 +723,7 @@ struct ReaderWriterSTG::_ModelBin {
|
|||||||
} else if (_foundBase) {
|
} else if (_foundBase) {
|
||||||
for (auto stgObject : _objectList) {
|
for (auto stgObject : _objectList) {
|
||||||
osg::ref_ptr<osg::Node> node;
|
osg::ref_ptr<osg::Node> node;
|
||||||
|
simgear::ErrorReportContext ec("terrain-stg", stgObject._errorLocation.utf8Str());
|
||||||
node = osgDB::readRefNodeFile(stgObject._name, stgObject._options.get());
|
node = osgDB::readRefNodeFile(stgObject._name, stgObject._options.get());
|
||||||
|
|
||||||
if (!node.valid()) {
|
if (!node.valid()) {
|
||||||
@ -838,11 +845,10 @@ ReaderWriterSTG::readNode(const std::string& fileName, const osgDB::Options* opt
|
|||||||
return ReadResult::FILE_NOT_FOUND;
|
return ReadResult::FILE_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
osg::ref_ptr<SGReaderWriterOptions> sgOpts(SGReaderWriterOptions::copyOrCreate(options));
|
const auto sgOpts = dynamic_cast<const SGReaderWriterOptions*>(options);
|
||||||
if (sgOpts->getSceneryPathSuffixes().empty()) {
|
if (!sgOpts || sgOpts->getSceneryPathSuffixes().empty()) {
|
||||||
SG_LOG(SG_TERRAIN, SG_ALERT, "Loading tile " << fileName << ", no scenery path suffixes were configured so giving up");
|
SG_LOG(SG_TERRAIN, SG_ALERT, "Loading tile " << fileName << ", no scenery path suffixes were configured so giving up");
|
||||||
return ReadResult::FILE_NOT_FOUND;
|
return ReadResult::FILE_NOT_FOUND;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
SG_LOG(SG_TERRAIN, SG_INFO, "Loading tile " << fileName);
|
SG_LOG(SG_TERRAIN, SG_INFO, "Loading tile " << fileName);
|
||||||
@ -869,7 +875,7 @@ ReaderWriterSTG::readNode(const std::string& fileName, const osgDB::Options* opt
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (auto suffix : sgOpts->getSceneryPathSuffixes()) {
|
for (auto suffix : sgOpts->getSceneryPathSuffixes()) {
|
||||||
SGPath p = base / suffix / basePath / fileName;
|
const auto p = base / suffix / basePath / fileName;
|
||||||
modelBin.read(p, options);
|
modelBin.read(p, options);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,9 +17,7 @@
|
|||||||
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
//
|
//
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
|
||||||
#include <simgear_config.h>
|
#include <simgear_config.h>
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <simgear/scene/util/OsgMath.hxx>
|
#include <simgear/scene/util/OsgMath.hxx>
|
||||||
|
|
||||||
@ -54,4 +52,10 @@ SGReaderWriterOptions::fromPath(const SGPath& path)
|
|||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SGReaderWriterOptions::addErrorContext(const std::string& key, const std::string& value)
|
||||||
|
{
|
||||||
|
_errorContext[key] = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace simgear
|
||||||
|
@ -82,8 +82,7 @@ public:
|
|||||||
_LoadOriginHint(ORIGIN_MODEL)
|
_LoadOriginHint(ORIGIN_MODEL)
|
||||||
{ }
|
{ }
|
||||||
SGReaderWriterOptions(const SGReaderWriterOptions& options,
|
SGReaderWriterOptions(const SGReaderWriterOptions& options,
|
||||||
const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY) :
|
const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY) : osgDB::Options(options, copyop),
|
||||||
osgDB::Options(options, copyop),
|
|
||||||
_propertyNode(options._propertyNode),
|
_propertyNode(options._propertyNode),
|
||||||
_materialLib(options._materialLib),
|
_materialLib(options._materialLib),
|
||||||
#ifdef ENABLE_GDAL
|
#ifdef ENABLE_GDAL
|
||||||
@ -97,7 +96,8 @@ public:
|
|||||||
_sceneryPathSuffixes(options._sceneryPathSuffixes),
|
_sceneryPathSuffixes(options._sceneryPathSuffixes),
|
||||||
_autoTooltipsMaster(options._autoTooltipsMaster),
|
_autoTooltipsMaster(options._autoTooltipsMaster),
|
||||||
_autoTooltipsMasterMax(options._autoTooltipsMasterMax),
|
_autoTooltipsMasterMax(options._autoTooltipsMasterMax),
|
||||||
_LoadOriginHint(ORIGIN_MODEL)
|
_LoadOriginHint(ORIGIN_MODEL),
|
||||||
|
_errorContext(options._errorContext)
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
META_Object(simgear, SGReaderWriterOptions);
|
META_Object(simgear, SGReaderWriterOptions);
|
||||||
@ -178,6 +178,15 @@ public:
|
|||||||
void setLoadOriginHint(LoadOriginHint _v) const { _LoadOriginHint = _v; }
|
void setLoadOriginHint(LoadOriginHint _v) const { _LoadOriginHint = _v; }
|
||||||
LoadOriginHint getLoadOriginHint() const { return _LoadOriginHint; }
|
LoadOriginHint getLoadOriginHint() const { return _LoadOriginHint; }
|
||||||
|
|
||||||
|
using ErrorContext = std::map<std::string, std::string>;
|
||||||
|
|
||||||
|
void addErrorContext(const std::string& key, const std::string& value);
|
||||||
|
|
||||||
|
ErrorContext getErrorContext() const
|
||||||
|
{
|
||||||
|
return _errorContext;
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual ~SGReaderWriterOptions();
|
virtual ~SGReaderWriterOptions();
|
||||||
|
|
||||||
@ -199,6 +208,7 @@ private:
|
|||||||
int _autoTooltipsMasterMax;
|
int _autoTooltipsMasterMax;
|
||||||
SGGeod _geod;
|
SGGeod _geod;
|
||||||
mutable LoadOriginHint _LoadOriginHint;
|
mutable LoadOriginHint _LoadOriginHint;
|
||||||
|
ErrorContext _errorContext;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user