2016-06-25 17:58:52 +02:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cmath>
|
|
|
|
#include <cctype>
|
|
|
|
|
|
|
|
#include "rwbase.h"
|
|
|
|
#include "rwerror.h"
|
|
|
|
#include "rwplg.h"
|
2017-08-24 15:10:34 +02:00
|
|
|
#include "rwpipeline.h"
|
|
|
|
#include "rwobjects.h"
|
|
|
|
#include "rwengine.h"
|
2016-06-25 17:58:52 +02:00
|
|
|
|
|
|
|
namespace rw {
|
|
|
|
|
|
|
|
static void *defCtor(void *object, int32, int32) { return object; }
|
|
|
|
static void *defDtor(void *object, int32, int32) { return object; }
|
|
|
|
static void *defCopy(void *dst, void*, int32, int32) { return dst; }
|
|
|
|
|
|
|
|
void
|
|
|
|
PluginList::construct(void *object)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
p->constructor(object, p->offset, p->size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
PluginList::destruct(void *object)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
p->destructor(object, p->offset, p->size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
PluginList::copy(void *dst, void *src)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
p->copy(dst, src, p->offset, p->size);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
PluginList::streamRead(Stream *stream, void *object)
|
|
|
|
{
|
|
|
|
int32 length;
|
|
|
|
ChunkHeaderInfo header;
|
|
|
|
if(!findChunk(stream, ID_EXTENSION, (uint32*)&length, nil))
|
|
|
|
return false;
|
|
|
|
while(length > 0){
|
|
|
|
if(!readChunkHeaderInfo(stream, &header))
|
|
|
|
return false;
|
|
|
|
length -= 12;
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->id == header.type && p->read){
|
|
|
|
p->read(stream, header.length,
|
|
|
|
object, p->offset, p->size);
|
|
|
|
goto cont;
|
|
|
|
}
|
|
|
|
stream->seek(header.length);
|
|
|
|
cont:
|
|
|
|
length -= header.length;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
PluginList::streamWrite(Stream *stream, void *object)
|
|
|
|
{
|
|
|
|
int size = this->streamGetSize(object);
|
|
|
|
writeChunkHeader(stream, ID_EXTENSION, size);
|
|
|
|
for(Plugin *p = this->first; p; p = p->next){
|
|
|
|
if(p->getSize == nil ||
|
|
|
|
(size = p->getSize(object, p->offset, p->size)) <= 0)
|
|
|
|
continue;
|
|
|
|
writeChunkHeader(stream, p->id, size);
|
|
|
|
p->write(stream, size, object, p->offset, p->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
PluginList::streamGetSize(void *object)
|
|
|
|
{
|
|
|
|
int32 size = 0;
|
|
|
|
int32 plgsize;
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->getSize &&
|
|
|
|
(plgsize = p->getSize(object, p->offset, p->size)) > 0)
|
|
|
|
size += 12 + plgsize;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2017-09-16 23:19:54 +02:00
|
|
|
void
|
|
|
|
PluginList::streamSkip(Stream *stream)
|
|
|
|
{
|
|
|
|
int32 length;
|
|
|
|
ChunkHeaderInfo header;
|
|
|
|
if(!findChunk(stream, ID_EXTENSION, (uint32*)&length, nil))
|
|
|
|
return;
|
|
|
|
while(length > 0){
|
|
|
|
if(!readChunkHeaderInfo(stream, &header))
|
|
|
|
return;
|
|
|
|
stream->seek(header.length);
|
|
|
|
length -= 12 + header.length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-25 17:58:52 +02:00
|
|
|
void
|
|
|
|
PluginList::assertRights(void *object, uint32 pluginID, uint32 data)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->id == pluginID){
|
|
|
|
if(p->rightsCallback)
|
|
|
|
p->rightsCallback(object,
|
|
|
|
p->offset, p->size, data);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int32
|
|
|
|
PluginList::registerPlugin(int32 size, uint32 id,
|
|
|
|
Constructor ctor, Destructor dtor, CopyConstructor copy)
|
|
|
|
{
|
2017-08-24 15:10:34 +02:00
|
|
|
Plugin *p = (Plugin*)rwMalloc(sizeof(Plugin), MEMDUR_GLOBAL);
|
2016-06-25 17:58:52 +02:00
|
|
|
p->offset = this->size;
|
|
|
|
this->size += size;
|
2016-07-21 17:06:56 +02:00
|
|
|
int32 round = sizeof(void*)-1;
|
|
|
|
this->size = (this->size + round)&~round;
|
2016-06-25 17:58:52 +02:00
|
|
|
|
|
|
|
p->size = size;
|
|
|
|
p->id = id;
|
|
|
|
p->constructor = ctor ? ctor : defCtor;
|
|
|
|
p->destructor = dtor ? dtor : defDtor;
|
|
|
|
p->copy = copy ? copy : defCopy;
|
|
|
|
p->read = nil;
|
|
|
|
p->write = nil;
|
|
|
|
p->getSize = nil;
|
|
|
|
p->rightsCallback = nil;
|
|
|
|
p->next = nil;
|
|
|
|
p->prev = nil;
|
|
|
|
|
|
|
|
if(this->first == nil){
|
|
|
|
this->first = p;
|
|
|
|
this->last = p;
|
|
|
|
}else{
|
|
|
|
this->last->next = p;
|
|
|
|
p->prev = this->last;
|
|
|
|
this->last = p;
|
|
|
|
}
|
|
|
|
return p->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32
|
|
|
|
PluginList::registerStream(uint32 id,
|
|
|
|
StreamRead read, StreamWrite write, StreamGetSize getSize)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->id == id){
|
|
|
|
p->read = read;
|
|
|
|
p->write = write;
|
|
|
|
p->getSize = getSize;
|
|
|
|
return p->offset;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32
|
|
|
|
PluginList::setStreamRightsCallback(uint32 id, RightsCallback cb)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->id == id){
|
|
|
|
p->rightsCallback = cb;
|
|
|
|
return p->offset;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32
|
|
|
|
PluginList::getPluginOffset(uint32 id)
|
|
|
|
{
|
|
|
|
for(Plugin *p = this->first; p; p = p->next)
|
|
|
|
if(p->id == id)
|
|
|
|
return p->offset;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|