mirror of
https://github.com/ksherlock/mpw.git
synced 2025-01-05 02:31:12 +00:00
more template code
This commit is contained in:
parent
7691267dea
commit
1cd59d36f3
@ -49,6 +49,7 @@
|
|||||||
|
|
||||||
#include "debugger.h"
|
#include "debugger.h"
|
||||||
#include "debugger_internal.h"
|
#include "debugger_internal.h"
|
||||||
|
#include "template.h"
|
||||||
|
|
||||||
#include <cpu/defs.h>
|
#include <cpu/defs.h>
|
||||||
#include <cpu/CpuModule.h>
|
#include <cpu/CpuModule.h>
|
||||||
@ -62,8 +63,6 @@
|
|||||||
#include <toolbox/loader.h>
|
#include <toolbox/loader.h>
|
||||||
#include <toolbox/mm.h>
|
#include <toolbox/mm.h>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
using namespace Debug::Internal;
|
using namespace Debug::Internal;
|
||||||
@ -84,6 +83,8 @@ namespace {
|
|||||||
ToolMap tbrkMap; // tool breaks.
|
ToolMap tbrkMap; // tool breaks.
|
||||||
|
|
||||||
|
|
||||||
|
std::unordered_map<std::string, Debug::Template> TemplateTable;
|
||||||
|
|
||||||
struct BackTraceInfo {
|
struct BackTraceInfo {
|
||||||
uint32_t a[8];
|
uint32_t a[8];
|
||||||
uint32_t d[8];
|
uint32_t d[8];
|
||||||
@ -1255,6 +1256,9 @@ void Shell()
|
|||||||
LoadTrapFile(MPW::RootDirPathForFile("Globals.text"), GlobalTable);
|
LoadTrapFile(MPW::RootDirPathForFile("Globals.text"), GlobalTable);
|
||||||
LoadTrapFile(MPW::RootDirPathForFile("Traps.text"), TrapTable);
|
LoadTrapFile(MPW::RootDirPathForFile("Traps.text"), TrapTable);
|
||||||
|
|
||||||
|
LoadTemplateFile(MPW::RootDirPathForFile("Templates.text"), TemplateTable);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// load the error code to error mnemonic
|
// load the error code to error mnemonic
|
||||||
ErrorTableInvert.reserve(ErrorTable.size());
|
ErrorTableInvert.reserve(ErrorTable.size());
|
||||||
|
101
bin/intern.cpp
Normal file
101
bin/intern.cpp
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
#include "intern.h"
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
std::unordered_multimap<unsigned, std::string *> InternTable;
|
||||||
|
|
||||||
|
std::string EmptyString;
|
||||||
|
|
||||||
|
unsigned int DJBHash(const char* begin, size_t length)
|
||||||
|
{
|
||||||
|
unsigned int hash = 5381;
|
||||||
|
|
||||||
|
for(size_t i = 0; i < length; ++i)
|
||||||
|
{
|
||||||
|
hash = ((hash << 5) + hash) + (begin[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace Intern {
|
||||||
|
|
||||||
|
const std::string *String(std::string &&str)
|
||||||
|
{
|
||||||
|
size_t size = str.size();
|
||||||
|
if (!size) return &EmptyString;
|
||||||
|
|
||||||
|
unsigned hash = DJBHash(str.data(), size);
|
||||||
|
|
||||||
|
auto range = InternTable.equal_range(hash);
|
||||||
|
auto iter = range.first;
|
||||||
|
auto endit = range.second;
|
||||||
|
|
||||||
|
for( ; iter != endit; ++iter)
|
||||||
|
{
|
||||||
|
// hash matches, make sure the string does.
|
||||||
|
const std::string *s = iter->second;
|
||||||
|
|
||||||
|
if (s->size() == size && std::memcmp(s->data(), str.data(), size) == 0)
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
// insert it. I suppose this could throw, in which case a string would leak.
|
||||||
|
std::string *s = new std::string(std::move(str));
|
||||||
|
InternTable.emplace(std::make_pair(hash, s));
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string *String(const char *begin, size_t size)
|
||||||
|
{
|
||||||
|
if (!size) return &EmptyString;
|
||||||
|
|
||||||
|
unsigned hash = DJBHash(begin, size);
|
||||||
|
|
||||||
|
auto range = InternTable.equal_range(hash);
|
||||||
|
auto iter = range.first;
|
||||||
|
auto endit = range.second;
|
||||||
|
|
||||||
|
for( ; iter != endit; ++iter)
|
||||||
|
{
|
||||||
|
// hash matches, make sure the string does.
|
||||||
|
const std::string *s = iter->second;
|
||||||
|
|
||||||
|
if (s->size() == size && std::memcmp(s->data(), begin, size) == 0)
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
// insert it. I suppose this could throw, in which case a string would leak.
|
||||||
|
std::string *s = new std::string(begin, size);
|
||||||
|
InternTable.emplace(std::make_pair(hash, s));
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string *String(const char *begin, const char *end)
|
||||||
|
{
|
||||||
|
return String(begin, end - begin);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const std::string *String(const char *cp)
|
||||||
|
{
|
||||||
|
if (!cp || !*cp) return &EmptyString;
|
||||||
|
|
||||||
|
return String(cp, strlen(cp));
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string *String(const std::string &s)
|
||||||
|
{
|
||||||
|
return String(s.data(), s.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
14
bin/intern.h
Normal file
14
bin/intern.h
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
#ifndef __intern_h__
|
||||||
|
#define __intern_h__
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace Intern {
|
||||||
|
const std::string *String(const std::string &s);
|
||||||
|
const std::string *String(std::string &&s);
|
||||||
|
const std::string *String(const char *);
|
||||||
|
const std::string *String(const char *, size_t size);
|
||||||
|
const std::string *String(const char *, const char *);
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
122
bin/template.h
122
bin/template.h
@ -6,40 +6,128 @@
|
|||||||
namespace Debug {
|
namespace Debug {
|
||||||
|
|
||||||
|
|
||||||
|
enum {
|
||||||
|
|
||||||
|
// 0x8000 = pointer
|
||||||
|
// 0x1000 = unsigned
|
||||||
|
// 0x0f00 = size
|
||||||
|
// 0-255 - type
|
||||||
|
|
||||||
|
kSInt64 = 0x0800,
|
||||||
|
kUInt64 = 0x1800,
|
||||||
|
|
||||||
|
kSInt32 = 0x0400,
|
||||||
|
kUInt32 = 0x1400,
|
||||||
|
|
||||||
|
kSInt16 = 0x0200,
|
||||||
|
kUInt16 = 0x1200,
|
||||||
|
|
||||||
|
kSInt8 = 0x0100,
|
||||||
|
kUInt8 = 0x1100,
|
||||||
|
|
||||||
|
kStruct = 0x0000,
|
||||||
|
kStructPtr = 0x8000,
|
||||||
|
|
||||||
|
// ptrs are a special case where size = 0
|
||||||
|
//kVoid = 0x0001,
|
||||||
|
kVoidPtr = 0x8001,
|
||||||
|
|
||||||
|
// these exist for display purposes.
|
||||||
|
kCStringPtr = 0x8002,
|
||||||
|
kPStringPtr = 0x8003,
|
||||||
|
|
||||||
|
kOSType = 0x00404,
|
||||||
|
kBoolean = 0x0105,
|
||||||
|
kOSErr = 0x0206,
|
||||||
|
kHandle = 0x04007,
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
inline unsigned MakePtr(unsigned type) { return type | 0x8000; }
|
||||||
|
inline unsigned MakeType(unsigned size, bool sign, bool ptr) {
|
||||||
|
unsigned rv = size << 8;
|
||||||
|
if (!sign) rv |= 0x0100;
|
||||||
|
if (ptr) rv |= 0x8000;
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
kUnknown = 0,
|
kDisplayNative = 0,
|
||||||
kStringPtr, // p-string
|
kDisplayStringPtr, // p-string
|
||||||
kCStringPtr, // c-string
|
kDisplayCStringPtr, // c-string
|
||||||
kPtr,
|
kDisplayOSType, // four-cc
|
||||||
kOSType, // four-cc
|
kDisplayBoolean, // unsigned char, display true/false
|
||||||
kBoolean, // unsigned char, display true/false
|
kDisplayOSErr,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
struct Template;
|
#if 0
|
||||||
|
|
||||||
|
struct Type {
|
||||||
|
enum {
|
||||||
|
kSimpleType,
|
||||||
|
kStructType,
|
||||||
|
kPointerType,
|
||||||
|
};
|
||||||
|
uint16_t tag;
|
||||||
|
uint16_t size;
|
||||||
|
};
|
||||||
|
|
||||||
|
// handles [signed/unsigned] type, type[], *type, and *type[]
|
||||||
|
struct SimpleType : public Type {
|
||||||
|
unsigned rank:16; // int == int[1]
|
||||||
|
unsigned display:14; //
|
||||||
|
|
||||||
|
unsigned sign:1;
|
||||||
|
unsigned pointer:1;
|
||||||
|
|
||||||
|
SimpleType() {
|
||||||
|
tag = kSimpleType;
|
||||||
|
size = 0;
|
||||||
|
rank = 0;
|
||||||
|
display = 0;
|
||||||
|
sign = 0;
|
||||||
|
pointer = 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct PointerType: public Type {
|
||||||
|
Type *type;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct StructType : public Type {
|
||||||
|
unsigned total_size;
|
||||||
|
FieldEntry *firstField;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
struct FieldEntry;
|
struct FieldEntry;
|
||||||
|
typedef FieldEntry *Template;
|
||||||
|
|
||||||
struct FieldEntry {
|
struct FieldEntry {
|
||||||
std::string *name;
|
std::string *name;
|
||||||
unsigned type;
|
unsigned type;
|
||||||
unsigned count;
|
unsigned count;
|
||||||
Template *tmpl;
|
Template tmpl;
|
||||||
FieldEntry *next;
|
FieldEntry *next;
|
||||||
|
unsigned struct_size; // only populated for head entry.
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Template {
|
struct TemplateParseInfo {
|
||||||
unsigned type; // 0 for structs, < 256 for types.
|
std::unordered_map<std::string, Template> *templates;
|
||||||
|
std::unordered_map<std::string, unsigned> *types;
|
||||||
FieldEntry *firstField;
|
int LineNumber;
|
||||||
unsigned size;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void CreateTypedef(const std::string *name, int type);
|
|
||||||
void CreateTemplate(const std::string *name, FieldEntry *firstField);
|
void CreateTypedef(const std::string *name, int type, TemplateParseInfo *);
|
||||||
|
void CreateTemplate(const std::string *name, FieldEntry *firstField, TemplateParseInfo *);
|
||||||
|
|
||||||
|
|
||||||
|
bool LoadTemplateFile(const std::string &filename, std::unordered_map<std::string, Template> &);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
extern std::unordered_map<std::string, Template *> Templates;
|
|
||||||
extern int TemplateLine;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
|
|
||||||
#include "template_parser.h"
|
#include "template_parser.h"
|
||||||
#include "template.h"
|
#include "template.h"
|
||||||
|
#include "intern.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -39,63 +40,26 @@ namespace {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
unsigned int DJBHash(const char* begin, const char *end)
|
|
||||||
{
|
|
||||||
unsigned int hash = 5381;
|
|
||||||
unsigned int i = 0;
|
|
||||||
|
|
||||||
for(const char *iter = begin; iter != end; ++iter)
|
|
||||||
{
|
|
||||||
hash = ((hash << 5) + hash) + (*iter);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return hash;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unordered_multimap<unsigned, std::string *> InternTable;
|
|
||||||
|
|
||||||
const std::string *InternString(const char *begin, const char *end)
|
|
||||||
{
|
|
||||||
bool found = false;
|
|
||||||
unsigned hash = DJBHash(begin, end);
|
|
||||||
size_t length = end - begin;
|
|
||||||
|
|
||||||
auto range = InternTable.equal_range(hash);
|
|
||||||
auto iter = range.first;
|
|
||||||
auto endit = range.second;
|
|
||||||
|
|
||||||
for( ; iter != endit; ++iter)
|
|
||||||
{
|
|
||||||
// hash matches, make sure the string does.
|
|
||||||
const std::string *s = iter->second;
|
|
||||||
|
|
||||||
if (s->length() == length && std::memcmp(s->data(), begin, length) == 0)
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
// insert it.
|
|
||||||
std::string *s = new std::string(begin, end);
|
|
||||||
InternTable.emplace(std::make_pair(hash, s));
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void TemplateParse(void *yyp, int yymajor, void *yyminor);
|
|
||||||
void *TemplateParseAlloc(void *(*mallocProc)(size_t));
|
void *TemplateParseAlloc(void *(*mallocProc)(size_t));
|
||||||
void TemplateParseFree(void *p, void (*freeProc)(void*));
|
void TemplateParseFree(void *p, void (*freeProc)(void*));
|
||||||
|
|
||||||
void TemplateParse(void *yyp, int yymajor, int yyminor)
|
void TemplateParse(void *yyp, int yymajor, void *yyminor, Debug::TemplateParseInfo *);
|
||||||
|
|
||||||
|
|
||||||
|
void TemplateParse(void *yyp, int yymajor, int yyminor, Debug::TemplateParseInfo *info)
|
||||||
{
|
{
|
||||||
TemplateParse(yyp, yymajor, &yyminor);
|
TemplateParse(yyp, yymajor, &yyminor, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TemplateParse(void *yyp, int yymajor, const std::string *yyminor)
|
void TemplateParse(void *yyp, int yymajor, const std::string *yyminor, Debug::TemplateParseInfo *info)
|
||||||
{
|
{
|
||||||
TemplateParse(yyp, yymajor, (void *)yyminor);
|
TemplateParse(yyp, yymajor, (void *)yyminor, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define TemplateParse(a,b,c) TemplateParse(a,b,c, &info)
|
||||||
|
|
||||||
%%{
|
%%{
|
||||||
machine lexer;
|
machine lexer;
|
||||||
|
|
||||||
@ -103,14 +67,14 @@ void TemplateParse(void *yyp, int yymajor, const std::string *yyminor)
|
|||||||
error := any* ${ fbreak; };
|
error := any* ${ fbreak; };
|
||||||
|
|
||||||
block_comment := |*
|
block_comment := |*
|
||||||
[\n\r] { TemplateLine++; };
|
[\n\r] { info.LineNumber++; };
|
||||||
'*/' { fgoto main; };
|
'*/' { fgoto main; };
|
||||||
any ;
|
any ;
|
||||||
*|;
|
*|;
|
||||||
|
|
||||||
main := |*
|
main := |*
|
||||||
|
|
||||||
[\n\r] { TemplateLine++; };
|
[\n\r] { info.LineNumber++; };
|
||||||
[ \t]+;
|
[ \t]+;
|
||||||
|
|
||||||
'//' [^\r\n]* ;
|
'//' [^\r\n]* ;
|
||||||
@ -139,24 +103,26 @@ void TemplateParse(void *yyp, int yymajor, const std::string *yyminor)
|
|||||||
'signed' { TemplateParse(parser, tkSIGNED, 0); };
|
'signed' { TemplateParse(parser, tkSIGNED, 0); };
|
||||||
'unsigned' { TemplateParse(parser, tkUNSIGNED, 0); };
|
'unsigned' { TemplateParse(parser, tkUNSIGNED, 0); };
|
||||||
|
|
||||||
'int64_t' { TemplateParse(parser, tkTYPECODE, 'q'); };
|
'int64_t' { TemplateParse(parser, tkTYPECODE, kSInt64); };
|
||||||
'uint64_t' { TemplateParse(parser, tkTYPECODE, 'Q'); };
|
'uint64_t' { TemplateParse(parser, tkTYPECODE, kUInt64); };
|
||||||
|
|
||||||
'int32_t' { TemplateParse(parser, tkTYPECODE, 'l'); };
|
'int32_t' { TemplateParse(parser, tkTYPECODE, kSInt32); };
|
||||||
'uint32_t' { TemplateParse(parser, tkTYPECODE, 'L'); };
|
'uint32_t' { TemplateParse(parser, tkTYPECODE, kUInt32); };
|
||||||
|
|
||||||
'int16_t' { TemplateParse(parser, tkTYPECODE, 's'); };
|
'int16_t' { TemplateParse(parser, tkTYPECODE, kSInt16); };
|
||||||
'uint16_t' { TemplateParse(parser, tkTYPECODE, 'S'); };
|
'uint16_t' { TemplateParse(parser, tkTYPECODE, kUInt16); };
|
||||||
|
|
||||||
'int8_t' { TemplateParse(parser, tkTYPECODE, 'c'); };
|
'int8_t' { TemplateParse(parser, tkTYPECODE, kSInt8); };
|
||||||
'uint8_t' { TemplateParse(parser, tkTYPECODE, 'C'); };
|
'uint8_t' { TemplateParse(parser, tkTYPECODE, kUInt8); };
|
||||||
|
|
||||||
|
|
||||||
'StringPtr' { TemplateParse(parser, tkTYPECODE, kStringPtr); };
|
'StringPtr' { TemplateParse(parser, tkTYPECODE, kPStringPtr); };
|
||||||
'CStringPtr' { TemplateParse(parser, tkTYPECODE, kCStringPtr); };
|
'CStringPtr' { TemplateParse(parser, tkTYPECODE, kCStringPtr); };
|
||||||
'Ptr' { TemplateParse(parser, tkTYPECODE, kPtr); };
|
'Ptr' { TemplateParse(parser, tkTYPECODE, kVoidPtr); };
|
||||||
'OSType' { TemplateParse(parser, tkTYPECODE, kOSType); };
|
'OSType' { TemplateParse(parser, tkTYPECODE, kOSType); };
|
||||||
|
'OSErr' { TemplateParse(parser, tkTYPECODE, kOSErr); };
|
||||||
'Boolean' { TemplateParse(parser, tkTYPECODE, kBoolean); };
|
'Boolean' { TemplateParse(parser, tkTYPECODE, kBoolean); };
|
||||||
|
'Handle' { TemplateParse(parser, tkTYPECODE, kHandle); };
|
||||||
|
|
||||||
|
|
||||||
# numbers. negative numbers are not allowed.
|
# numbers. negative numbers are not allowed.
|
||||||
@ -177,16 +143,26 @@ void TemplateParse(void *yyp, int yymajor, const std::string *yyminor)
|
|||||||
|
|
||||||
// intern the string.
|
// intern the string.
|
||||||
|
|
||||||
const std::string *name = InternString(ts, te);
|
const std::string *name = Intern::String(ts, te);
|
||||||
|
bool ok = false;
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
auto iter = Types.find(*name);
|
||||||
|
if (iter != Types.end())
|
||||||
|
{
|
||||||
|
TemplateParse(parser, tkTYPECODE, iter->second);
|
||||||
|
ok = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!ok) {
|
||||||
auto iter = Templates.find(*name);
|
auto iter = Templates.find(*name);
|
||||||
if (iter != Templates.end())
|
if (iter != Templates.end())
|
||||||
{
|
{
|
||||||
unsigned type = iter->second->type;
|
TemplateParse(parser, tkTEMPLATE, iter->second);
|
||||||
if (type) TemplateParse(parser, tkTYPECODE, type);
|
ok = true;
|
||||||
else TemplateParse(parser, tkTEMPLATE, iter->second);
|
|
||||||
}
|
}
|
||||||
else
|
}
|
||||||
|
if (!ok)
|
||||||
{
|
{
|
||||||
TemplateParse(parser, tkIDENTIFIER, name);
|
TemplateParse(parser, tkIDENTIFIER, name);
|
||||||
}
|
}
|
||||||
@ -198,18 +174,108 @@ void TemplateParse(void *yyp, int yymajor, const std::string *yyminor)
|
|||||||
|
|
||||||
namespace Debug {
|
namespace Debug {
|
||||||
|
|
||||||
std::unordered_map<std::string, Template *> Templates;
|
|
||||||
|
|
||||||
void CreateTypedef(const std::string *name, int type)
|
void CreateTypedef(const std::string *name, int type, TemplateParseInfo *info)
|
||||||
{
|
{
|
||||||
}
|
// check if it's an existing typedef...
|
||||||
void CreateTemplate(const std::string *name, FieldEntry *firstField)
|
|
||||||
|
auto &Templates = *info->templates;
|
||||||
|
auto &Types = *info->types;
|
||||||
|
|
||||||
|
|
||||||
|
auto iter = Types.find(*name);
|
||||||
|
if (iter != Types.end())
|
||||||
{
|
{
|
||||||
|
if (iter->second == type) return; // ok, just a duplicate.
|
||||||
|
fprintf(stderr, "Template Error: line %d - redefining %s\n",
|
||||||
|
info->LineNumber, name->c_str());
|
||||||
|
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int TemplateLine;
|
if (Templates.find(*name) != Templates.end())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Template Error: line %d - redefining %s\n",
|
||||||
|
info->LineNumber, name->c_str());
|
||||||
|
|
||||||
bool ParseTemplates(const std::string &filename)
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Types.emplace(std::make_pair(*name, type));
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned CalcSize(FieldEntry *e)
|
||||||
|
{
|
||||||
|
unsigned size = 0;
|
||||||
|
|
||||||
|
while (e)
|
||||||
|
{
|
||||||
|
unsigned s = (e->type & 0x0f00) >> 8;
|
||||||
|
|
||||||
|
if (!s) {
|
||||||
|
// struct or pointer...
|
||||||
|
if (e->type & 0x8000) s = 4;
|
||||||
|
else if (e->tmpl) s = e->tmpl->struct_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e->count != 0) s *= e->count;
|
||||||
|
|
||||||
|
size += s;
|
||||||
|
e = e->next;
|
||||||
|
}
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
FieldEntry *Reverse(FieldEntry *e)
|
||||||
|
{
|
||||||
|
if (!e) return e;
|
||||||
|
|
||||||
|
// reverse the order...
|
||||||
|
FieldEntry *prev;
|
||||||
|
FieldEntry *next;
|
||||||
|
|
||||||
|
prev = nullptr;
|
||||||
|
for(;;)
|
||||||
|
{
|
||||||
|
next = e->next;
|
||||||
|
e->next = prev;
|
||||||
|
|
||||||
|
prev = e;
|
||||||
|
e = next;
|
||||||
|
if (!e) return prev;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void CreateTemplate(const std::string *name, FieldEntry *firstField, TemplateParseInfo *info)
|
||||||
|
{
|
||||||
|
auto &Templates = *info->templates;
|
||||||
|
auto &Types = *info->types;
|
||||||
|
|
||||||
|
// check if it exists...
|
||||||
|
|
||||||
|
if (Templates.find(*name) != Templates.end())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Template Error: line %d - redefining %s\n",
|
||||||
|
info->LineNumber, name->c_str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Types.find(*name) != Types.end())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Template Error: line %d - redefining %s\n",
|
||||||
|
info->LineNumber, name->c_str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
firstField = Reverse(firstField);
|
||||||
|
firstField->struct_size = CalcSize(firstField);
|
||||||
|
|
||||||
|
Templates.emplace(std::make_pair(*name, firstField));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool LoadTemplateFile(const std::string &filename, std::unordered_map<std::string, Template> &Templates)
|
||||||
{
|
{
|
||||||
%% write data;
|
%% write data;
|
||||||
|
|
||||||
@ -219,6 +285,17 @@ bool ParseTemplates(const std::string &filename)
|
|||||||
struct stat st;
|
struct stat st;
|
||||||
char *buffer;
|
char *buffer;
|
||||||
|
|
||||||
|
std::unordered_map<std::string, unsigned> Types;
|
||||||
|
TemplateParseInfo info;
|
||||||
|
|
||||||
|
info.LineNumber = 1;
|
||||||
|
info.templates = &Templates;
|
||||||
|
info.types = &Types;
|
||||||
|
|
||||||
|
// simple types are handled via the lexer so no need to populate them here.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (stat(filename.c_str(), &st) < 0) return false;
|
if (stat(filename.c_str(), &st) < 0) return false;
|
||||||
if (st.st_size == 0) return false;
|
if (st.st_size == 0) return false;
|
||||||
|
|
||||||
@ -241,9 +318,6 @@ bool ParseTemplates(const std::string &filename)
|
|||||||
|
|
||||||
parser = TemplateParseAlloc(malloc);
|
parser = TemplateParseAlloc(malloc);
|
||||||
|
|
||||||
|
|
||||||
TemplateLine = 1;
|
|
||||||
|
|
||||||
const char *p = buffer;
|
const char *p = buffer;
|
||||||
const char *pe = buffer + st.st_size;
|
const char *pe = buffer + st.st_size;
|
||||||
const char *eof = pe;
|
const char *eof = pe;
|
||||||
@ -272,7 +346,8 @@ bool ParseTemplates(const std::string &filename)
|
|||||||
}
|
}
|
||||||
if (p == pe)
|
if (p == pe)
|
||||||
{
|
{
|
||||||
TemplateParse(parser, tkEOF, 0);
|
// ?
|
||||||
|
//TemplateParse(parser, tkEOF, 0);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
%token_prefix tk
|
%token_prefix tk
|
||||||
%name TemplateParse
|
%name TemplateParse
|
||||||
|
%extra_argument { Debug::TemplateParseInfo *info }
|
||||||
|
|
||||||
|
|
||||||
%include {
|
%include {
|
||||||
|
|
||||||
@ -13,24 +15,22 @@
|
|||||||
|
|
||||||
%type struct_fields { FieldEntry * }
|
%type struct_fields { FieldEntry * }
|
||||||
%type struct_field { FieldEntry * }
|
%type struct_field { FieldEntry * }
|
||||||
%type array_count { int }
|
|
||||||
%type opt_star { int }
|
|
||||||
%type typecode { int }
|
|
||||||
%type type { int }
|
|
||||||
|
|
||||||
start ::= templates EOF.
|
start ::= templates.
|
||||||
|
|
||||||
templates ::= templates struct.
|
templates ::= templates struct.
|
||||||
templates ::= templates typedef.
|
templates ::= templates typedef.
|
||||||
templates ::= .
|
templates ::= .
|
||||||
|
|
||||||
|
// typedeffing arrays or pointers is not allowed.
|
||||||
|
|
||||||
typedef ::= TYPEDEF type(a) IDENTIFIER(b). {
|
typedef ::= TYPEDEF type(a) IDENTIFIER(b). {
|
||||||
CreateTypedef((std::string *)b, a);
|
CreateTypedef((std::string *)b, a, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ::= STRUCT IDENTIFIER(a) LBRACE struct_fields(b) RBRACE SEMI.
|
struct ::= STRUCT IDENTIFIER(a) LBRACE struct_fields(b) RBRACE SEMI.
|
||||||
{
|
{
|
||||||
CreateTemplate((std::string *)a, b);
|
CreateTemplate((std::string *)a, b, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct_fields(rhs) ::= struct_fields(a) struct_field(b). {
|
struct_fields(rhs) ::= struct_fields(a) struct_field(b). {
|
||||||
@ -57,50 +57,63 @@ struct_field(rhs) ::= opt_volatile TEMPLATE(a) opt_star(star) IDENTIFIER(b) arra
|
|||||||
FieldEntry *e = (FieldEntry *)calloc(sizeof(FieldEntry), 1);
|
FieldEntry *e = (FieldEntry *)calloc(sizeof(FieldEntry), 1);
|
||||||
|
|
||||||
e->name = (std::string *)b;
|
e->name = (std::string *)b;
|
||||||
e->type = star ? kPtr : 0;
|
e->type = star ? kStruct : kStructPtr;
|
||||||
e->tmpl = (Template *)a;
|
e->tmpl = (Template)a;
|
||||||
e->count = c;
|
e->count = c;
|
||||||
|
|
||||||
rhs = e;
|
rhs = e;
|
||||||
}
|
}
|
||||||
|
|
||||||
array_count(rhs) ::= . { rhs = -1; }
|
%type array_count { int }
|
||||||
array_count(rhs) ::= LBRACKET INTEGER(a) RBRACKET. { rhs = *(int *)a; }
|
array_count(rhs) ::= . { rhs = 0; }
|
||||||
|
array_count(rhs) ::= LBRACKET INTEGER(a) RBRACKET. {
|
||||||
|
int i = *(int *)a;
|
||||||
|
if (i == 0) {
|
||||||
|
fprintf(stderr, "Template error: line %u: 0-sized arrays are not allowed.\n",
|
||||||
|
info->LineNumber);
|
||||||
|
i = 1;
|
||||||
|
}
|
||||||
|
rhs = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
%type type { int }
|
||||||
|
type(rhs) ::= opt_volatile typecode(a). { rhs = a; }
|
||||||
|
|
||||||
// this is an expected error...
|
// this is an expected error...
|
||||||
type(rhs) ::= opt_volatile IDENTIFIER(xxx). {
|
type(rhs) ::= opt_volatile IDENTIFIER(xxx). {
|
||||||
|
|
||||||
// ugh, Lemon will blindly replace text within a string.
|
// ugh, Lemon will blindly replace text within a string.
|
||||||
fprintf(stderr, "Template error: line %u: %s is not a known type.\n",
|
fprintf(stderr, "Template error: line %u: %s is not a known type.\n",
|
||||||
TemplateLine, ((std::string *)xxx)->c_str());
|
info->LineNumber, ((std::string *)xxx)->c_str());
|
||||||
|
|
||||||
rhs = 'i';
|
rhs = 'i';
|
||||||
}
|
}
|
||||||
|
|
||||||
type(rhs) ::= opt_volatile typecode(a). { rhs = a; }
|
|
||||||
|
|
||||||
opt_volatile ::= .
|
opt_volatile ::= .
|
||||||
opt_volatile ::= VOLATILE.
|
opt_volatile ::= VOLATILE.
|
||||||
|
|
||||||
typecode(rhs) ::= SIGNED. { rhs = 'i'; }
|
|
||||||
typecode(rhs) ::= UNSIGNED. {rhs = 'I'; }
|
|
||||||
|
|
||||||
typecode(rhs) ::= opt_signed CHAR. { rhs = 'c'; }
|
%type typecode { int }
|
||||||
typecode(rhs) ::= UNSIGNED CHAR. { rhs = 'C'; }
|
typecode(rhs) ::= SIGNED. { rhs = kSInt32; }
|
||||||
|
typecode(rhs) ::= UNSIGNED. {rhs = kUInt32; }
|
||||||
|
|
||||||
typecode(rhs) ::= opt_signed SHORT. { rhs = 's'; }
|
typecode(rhs) ::= opt_signed CHAR. { rhs = kSInt8; }
|
||||||
typecode(rhs) ::= UNSIGNED SHORT. { rhs = 'S'; }
|
typecode(rhs) ::= UNSIGNED CHAR. { rhs = kUInt8; }
|
||||||
|
|
||||||
typecode(rhs) ::= opt_signed LONG opt_int. { rhs = 'l'; }
|
typecode(rhs) ::= opt_signed SHORT. { rhs = kSInt16; }
|
||||||
typecode(rhs) ::= UNSIGNED LONG opt_int. { rhs = 'L'; }
|
typecode(rhs) ::= UNSIGNED SHORT. { rhs = kUInt16; }
|
||||||
|
|
||||||
typecode(rhs) ::= opt_signed LONG LONG. { rhs = 'q'; }
|
typecode(rhs) ::= opt_signed LONG opt_int. { rhs = kSInt32; }
|
||||||
typecode(rhs) ::= UNSIGNED LONG LONG. { rhs = 'Q'; }
|
typecode(rhs) ::= UNSIGNED LONG opt_int. { rhs = kUInt32; }
|
||||||
|
|
||||||
|
typecode(rhs) ::= opt_signed LONG LONG. { rhs = kSInt64; }
|
||||||
|
typecode(rhs) ::= UNSIGNED LONG LONG. { rhs = kUInt64; }
|
||||||
|
|
||||||
typecode(rhs) ::= TYPECODE(a). { rhs = *(int *)a; }
|
typecode(rhs) ::= TYPECODE(a). { rhs = *(int *)a; }
|
||||||
|
|
||||||
/* pointers are not fully supported yet */
|
/* pointers are not fully supported yet */
|
||||||
typecode(rhs) ::= VOID STAR. { rhs = kPtr; }
|
typecode(rhs) ::= VOID STAR. { rhs = kVoidPtr; }
|
||||||
|
|
||||||
opt_signed ::= .
|
opt_signed ::= .
|
||||||
opt_signed ::= SIGNED.
|
opt_signed ::= SIGNED.
|
||||||
@ -108,7 +121,7 @@ opt_signed ::= SIGNED.
|
|||||||
opt_int ::= .
|
opt_int ::= .
|
||||||
opt_int ::= INT.
|
opt_int ::= INT.
|
||||||
|
|
||||||
|
%type opt_star { int }
|
||||||
opt_star(rhs) ::= . { rhs = 0; }
|
opt_star(rhs) ::= . { rhs = 0; }
|
||||||
opt_star(rhs) ::= STAR. { rhs = 1; }
|
opt_star(rhs) ::= STAR. { rhs = 1; }
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user