Rez: support resource attributes & names

This commit is contained in:
Wolfgang Thaller 2015-07-20 01:51:33 +02:00
parent d26d88f5ee
commit 334f06ed83
4 changed files with 52 additions and 9 deletions

View File

@ -27,7 +27,9 @@ void Resources::writeFork(std::ostream& out) const
{ {
const Resource& r = rr.second; const Resource& r = rr.second;
const std::string& data = r.getData(); const std::string& data = r.getData();
resourceInfos[ r.getType() ][ r.getID() ] = out.tellp() - datastart; unsigned offset = out.tellp() - datastart;
offset = (r.getAttr() << 24) | (offset & 0xFFFFFF);
resourceInfos[ r.getType() ][ r.getID() ] = (int)offset;
longword(out, data.size()); longword(out, data.size());
out << data; out << data;
} }

View File

@ -22,6 +22,7 @@ public:
inline int getID() const { return id; } inline int getID() const { return id; }
inline ResRef getTypeAndID() const { return ResRef(type, id); } inline ResRef getTypeAndID() const { return ResRef(type, id); }
std::string getName() const { return name; } std::string getName() const { return name; }
int getAttr() const { return attr; }
}; };
class Fork class Fork

View File

@ -10,7 +10,7 @@ class ResSpec : public ResRef
std::string name_; std::string name_;
public: public:
ResSpec() {} ResSpec() : attr_(0) {}
ResSpec(ResType type, int id, int attr = 0, std::string name = "") ResSpec(ResType type, int id, int attr = 0, std::string name = "")
: ResRef(type, id), attr_(attr), name_(name) : ResRef(type, id), attr_(attr), name_(name)
{} {}

View File

@ -125,6 +125,7 @@
#include "RezLexer.h" #include "RezLexer.h"
#include "RezWorld.h" #include "RezWorld.h"
#include "ResourceCompiler.h" #include "ResourceCompiler.h"
#include "Diagnostic.h"
static yy::RezParser::symbol_type yylex(RezLexer& lexer, RezWorld&) static yy::RezParser::symbol_type yylex(RezLexer& lexer, RezWorld&)
{ {
@ -133,7 +134,7 @@
void yy::RezParser::error(const location_type& loc, std::string const& err) void yy::RezParser::error(const location_type& loc, std::string const& err)
{ {
std::cerr << loc << ": " << err << std::endl; world.problem(Diagnostic(Diagnostic::error, err, loc));
} }
static std::string fromHex(std::string hex) static std::string fromHex(std::string hex)
@ -431,17 +432,56 @@ resource : "resource" res_spec "{" resource_body "}"
; ;
%type <ResSpec> res_spec; %type <ResSpec> res_spec;
%type <ResSpec> resource_attributes;
res_spec : res_type "(" expression resource_attributes ")" res_spec : res_type "(" expression resource_attributes ")"
{ $$ = $resource_attributes( ResSpec($res_type, $expression->evaluateInt(nullptr)) ); } {
$$ = $resource_attributes;
%type <std::function<ResSpec(ResSpec)>> resource_attributes ; $$.type() = $res_type;
resource_attributes : %empty { $$ = [](ResSpec s){ return s; }; } $$.id() = $expression->evaluateInt(nullptr);
| resource_attributes "," IDENTIFIER { $$ = $1; } }
| resource_attributes "," string_expression { $$ = $1; }
; ;
resource_attributes : %empty { $$ = ResSpec(); }
| resource_attributes "," IDENTIFIER
{
$$ = $1;
if($IDENTIFIER == "changed")
$$.attr() |= 2;
else if($IDENTIFIER == "preload")
$$.attr() |= 4;
else if($IDENTIFIER == "protected")
$$.attr() |= 8;
else if($IDENTIFIER == "locked")
$$.attr() |= 16;
else if($IDENTIFIER == "purgeable")
$$.attr() |= 32;
else if($IDENTIFIER == "sysheap")
$$.attr() |= 64;
else if($IDENTIFIER == "unchanged")
$$.attr() &= ~2;
else if($IDENTIFIER == "nonpreload")
$$.attr() &= ~4;
else if($IDENTIFIER == "unprotected")
$$.attr() &= ~8;
else if($IDENTIFIER == "unlocked")
$$.attr() &= ~16;
else if($IDENTIFIER == "nonpurgeable")
$$.attr() &= ~32;
else if($IDENTIFIER == "appheap")
$$.attr() &= ~64;
else
world.problem(Diagnostic(Diagnostic::error, "illegal attribute " + $IDENTIFIER, @1));
}
| resource_attributes "," string_expression
{
$$ = $1;
$$.name() = $3->evaluateString(nullptr);
}
;
%type <CompoundExprPtr> resource_body resource_body1; %type <CompoundExprPtr> resource_body resource_body1;
resource_body : %empty { $$ = std::make_shared<CompoundExpr>(yy::location()); } resource_body : %empty { $$ = std::make_shared<CompoundExpr>(yy::location()); }
| resource_body1 { $$ = $1; $$->location = @1; } | resource_body1 { $$ = $1; $$->location = @1; }