2016-06-17 00:06:37 +02:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
|
|
#include "rwbase.h"
|
|
|
|
#include "rwerror.h"
|
|
|
|
#include "rwplg.h"
|
|
|
|
#include "rwpipeline.h"
|
|
|
|
#include "rwobjects.h"
|
|
|
|
|
|
|
|
#define PLUGIN_ID 0
|
|
|
|
|
|
|
|
namespace rw {
|
|
|
|
|
|
|
|
LinkList Frame::dirtyList;
|
|
|
|
|
|
|
|
Frame*
|
|
|
|
Frame::create(void)
|
|
|
|
{
|
2016-06-25 17:58:52 +02:00
|
|
|
Frame *f = (Frame*)malloc(s_plglist.size);
|
2016-06-21 23:16:09 +02:00
|
|
|
if(f == nil){
|
2016-06-25 17:58:52 +02:00
|
|
|
RWERROR((ERR_ALLOC, s_plglist.size));
|
2016-06-21 23:16:09 +02:00
|
|
|
return nil;
|
2016-06-17 00:06:37 +02:00
|
|
|
}
|
|
|
|
f->object.init(Frame::ID, 0);
|
|
|
|
f->objectList.init();
|
2016-06-21 23:16:09 +02:00
|
|
|
f->child = nil;
|
|
|
|
f->next = nil;
|
2016-06-17 00:06:37 +02:00
|
|
|
f->root = f;
|
|
|
|
f->matrix.setIdentity();
|
|
|
|
f->ltm.setIdentity();
|
2016-06-25 17:58:52 +02:00
|
|
|
s_plglist.construct(f);
|
2016-06-17 00:06:37 +02:00
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
Frame*
|
|
|
|
Frame::cloneHierarchy(void)
|
|
|
|
{
|
2016-06-21 23:16:09 +02:00
|
|
|
Frame *frame = this->cloneAndLink(nil);
|
2016-06-17 00:06:37 +02:00
|
|
|
frame->purgeClone();
|
|
|
|
return frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Frame::destroy(void)
|
|
|
|
{
|
2016-06-25 17:58:52 +02:00
|
|
|
s_plglist.destruct(this);
|
2016-08-03 20:32:43 +02:00
|
|
|
if(this->getParent())
|
|
|
|
this->removeChild();
|
|
|
|
if(this->object.privateFlags & Frame::HIERARCHYSYNC)
|
|
|
|
this->inDirtyList.remove();
|
2016-06-17 00:06:37 +02:00
|
|
|
for(Frame *f = this->child; f; f = f->next)
|
2016-06-21 23:16:09 +02:00
|
|
|
f->object.parent = nil;
|
2016-06-17 00:06:37 +02:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Frame::destroyHierarchy(void)
|
|
|
|
{
|
|
|
|
Frame *next;
|
|
|
|
for(Frame *child = this->child; child; child = next){
|
|
|
|
next = child->next;
|
|
|
|
child->destroyHierarchy();
|
|
|
|
}
|
2016-06-25 17:58:52 +02:00
|
|
|
s_plglist.destruct(this);
|
2016-08-03 20:32:43 +02:00
|
|
|
if(this->object.privateFlags & Frame::HIERARCHYSYNC)
|
|
|
|
this->inDirtyList.remove();
|
2016-06-17 00:06:37 +02:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
Frame*
|
|
|
|
Frame::addChild(Frame *child, bool32 append)
|
|
|
|
{
|
|
|
|
Frame *c;
|
|
|
|
if(child->getParent())
|
|
|
|
child->removeChild();
|
|
|
|
if(append){
|
2016-06-21 23:16:09 +02:00
|
|
|
if(this->child == nil)
|
2016-06-17 00:06:37 +02:00
|
|
|
this->child = child;
|
|
|
|
else{
|
|
|
|
for(c = this->child; c->next; c = c->next);
|
|
|
|
c->next = child;
|
|
|
|
}
|
2016-06-21 23:16:09 +02:00
|
|
|
child->next = nil;
|
2016-06-17 00:06:37 +02:00
|
|
|
}else{
|
|
|
|
child->next = this->child;
|
|
|
|
this->child = child;
|
|
|
|
}
|
|
|
|
child->object.parent = this;
|
|
|
|
child->root = this->root;
|
|
|
|
for(c = child->child; c; c = c->next)
|
|
|
|
c->setHierarchyRoot(this);
|
2016-06-21 23:16:09 +02:00
|
|
|
// If the child was a root, remove from dirty list
|
2016-06-17 00:06:37 +02:00
|
|
|
if(child->object.privateFlags & Frame::HIERARCHYSYNC){
|
|
|
|
child->inDirtyList.remove();
|
|
|
|
child->object.privateFlags &= ~Frame::HIERARCHYSYNC;
|
|
|
|
}
|
|
|
|
this->updateObjects();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Frame*
|
|
|
|
Frame::removeChild(void)
|
|
|
|
{
|
|
|
|
Frame *parent = this->getParent();
|
|
|
|
Frame *child = parent->child;
|
|
|
|
if(child == this)
|
|
|
|
parent->child = this->next;
|
|
|
|
else{
|
|
|
|
while(child->next != this)
|
|
|
|
child = child->next;
|
|
|
|
child->next = this->next;
|
|
|
|
}
|
2016-06-21 23:16:09 +02:00
|
|
|
this->object.parent = this->next = nil;
|
2016-08-03 20:32:43 +02:00
|
|
|
// give the hierarchy a new root
|
|
|
|
this->setHierarchyRoot(this);
|
2016-06-17 00:06:37 +02:00
|
|
|
this->updateObjects();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Frame*
|
|
|
|
Frame::forAllChildren(Callback cb, void *data)
|
|
|
|
{
|
|
|
|
for(Frame *f = this->child; f; f = f->next)
|
|
|
|
cb(f, data);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Frame*
|
|
|
|
countCB(Frame *f, void *count)
|
|
|
|
{
|
|
|
|
(*(int32*)count)++;
|
|
|
|
f->forAllChildren(countCB, count);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32
|
|
|
|
Frame::count(void)
|
|
|
|
{
|
|
|
|
int32 count = 1;
|
|
|
|
this->forAllChildren(countCB, (void*)&count);
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2016-06-21 23:16:09 +02:00
|
|
|
/*
|
|
|
|
* Synching is a bit complicated. If anything in the hierarchy is not synched,
|
|
|
|
* the root of the hierarchy is marked with the HIERARCHYSYNC flags.
|
|
|
|
* Every unsynched frame is marked with the SUBTREESYNC flags.
|
|
|
|
* If the LTM is not synched, the LTM flags are set.
|
|
|
|
* If attached objects need synching, the OBJ flags are set.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Synch just LTM matrices in a hierarchy */
|
2016-06-17 00:06:37 +02:00
|
|
|
static void
|
2016-06-21 23:16:09 +02:00
|
|
|
syncLTMRecurse(Frame *frame, uint8 hierarchyFlags)
|
2016-06-17 00:06:37 +02:00
|
|
|
{
|
|
|
|
for(; frame; frame = frame->next){
|
2016-06-21 23:16:09 +02:00
|
|
|
// If frame is dirty or any parent was dirty, update LTM
|
|
|
|
hierarchyFlags |= frame->object.privateFlags;
|
|
|
|
if(hierarchyFlags & Frame::SUBTREESYNCLTM){
|
2016-06-17 00:06:37 +02:00
|
|
|
Matrix::mult(&frame->ltm, &frame->getParent()->ltm,
|
|
|
|
&frame->matrix);
|
|
|
|
frame->object.privateFlags &= ~Frame::SUBTREESYNCLTM;
|
|
|
|
}
|
2016-06-21 23:16:09 +02:00
|
|
|
// And synch all children
|
|
|
|
syncLTMRecurse(frame->child, hierarchyFlags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Synch just objects in a hierarchy */
|
|
|
|
static void
|
|
|
|
syncObjRecurse(Frame *frame)
|
|
|
|
{
|
|
|
|
for(; frame; frame = frame->next){
|
|
|
|
// Synch attached objects
|
|
|
|
FORLIST(lnk, frame->objectList)
|
|
|
|
ObjectWithFrame::fromFrame(lnk)->sync();
|
|
|
|
frame->object.privateFlags &= ~Frame::SUBTREESYNCOBJ;
|
|
|
|
// And synch all children
|
|
|
|
syncObjRecurse(frame->child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Synch LTM and objects */
|
|
|
|
static void
|
|
|
|
syncRecurse(Frame *frame, uint8 hierarchyFlags)
|
|
|
|
{
|
|
|
|
for(; frame; frame = frame->next){
|
|
|
|
// If frame is dirty or any parent was dirty, update LTM
|
|
|
|
hierarchyFlags |= frame->object.privateFlags;
|
|
|
|
if(hierarchyFlags & Frame::SUBTREESYNCLTM)
|
|
|
|
Matrix::mult(&frame->ltm, &frame->getParent()->ltm,
|
|
|
|
&frame->matrix);
|
|
|
|
// Synch attached objects
|
|
|
|
FORLIST(lnk, frame->objectList)
|
|
|
|
ObjectWithFrame::fromFrame(lnk)->sync();
|
|
|
|
frame->object.privateFlags &= ~Frame::SUBTREESYNC;
|
|
|
|
// And synch all children
|
|
|
|
syncRecurse(frame->child, hierarchyFlags);
|
2016-06-17 00:06:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-21 23:16:09 +02:00
|
|
|
/* Sync the LTMs of the hierarchy of which 'this' is the root */
|
2016-06-17 00:06:37 +02:00
|
|
|
void
|
|
|
|
Frame::syncHierarchyLTM(void)
|
|
|
|
{
|
2016-06-21 23:16:09 +02:00
|
|
|
// Sync root's LTM
|
2016-06-17 00:06:37 +02:00
|
|
|
if(this->object.privateFlags & Frame::SUBTREESYNCLTM)
|
|
|
|
this->ltm = this->matrix;
|
2016-06-21 23:16:09 +02:00
|
|
|
// ...and children
|
|
|
|
syncLTMRecurse(this->child, this->object.privateFlags);
|
|
|
|
// all clean now
|
2016-06-17 00:06:37 +02:00
|
|
|
this->object.privateFlags &= ~Frame::SYNCLTM;
|
|
|
|
}
|
|
|
|
|
|
|
|
Matrix*
|
|
|
|
Frame::getLTM(void)
|
|
|
|
{
|
|
|
|
if(this->root->object.privateFlags & Frame::HIERARCHYSYNCLTM)
|
|
|
|
this->root->syncHierarchyLTM();
|
|
|
|
return &this->ltm;
|
|
|
|
}
|
|
|
|
|
2016-06-21 23:16:09 +02:00
|
|
|
/* Synch all dirty frames; LTMs and objects */
|
|
|
|
void
|
|
|
|
Frame::syncDirty(void)
|
|
|
|
{
|
|
|
|
Frame *frame;
|
|
|
|
FORLIST(lnk, Frame::dirtyList){
|
|
|
|
frame = LLLinkGetData(lnk, Frame, inDirtyList);
|
|
|
|
if(frame->object.privateFlags & Frame::HIERARCHYSYNCLTM){
|
|
|
|
// Sync root's LTM
|
|
|
|
if(frame->object.privateFlags & Frame::SUBTREESYNCLTM)
|
|
|
|
frame->ltm = frame->matrix;
|
|
|
|
// Synch attached objects
|
|
|
|
FORLIST(lnk, frame->objectList)
|
|
|
|
ObjectWithFrame::fromFrame(lnk)->sync();
|
|
|
|
// ...and children
|
|
|
|
syncRecurse(frame->child, frame->object.privateFlags);
|
|
|
|
}else{
|
|
|
|
// LTMs are clean, just synch objects
|
|
|
|
FORLIST(lnk, frame->objectList)
|
|
|
|
ObjectWithFrame::fromFrame(lnk)->sync();
|
|
|
|
syncObjRecurse(frame->child);
|
|
|
|
}
|
|
|
|
// all clean now
|
|
|
|
frame->object.privateFlags &= ~(Frame::SYNCLTM | Frame::SYNCOBJ);
|
|
|
|
}
|
|
|
|
Frame::dirtyList.init();
|
|
|
|
}
|
|
|
|
|
2016-06-17 00:06:37 +02:00
|
|
|
void
|
|
|
|
Frame::updateObjects(void)
|
|
|
|
{
|
2016-06-21 23:16:09 +02:00
|
|
|
// Mark root as dirty and insert into dirty list if necessary
|
2016-06-17 00:06:37 +02:00
|
|
|
if((this->root->object.privateFlags & HIERARCHYSYNC) == 0)
|
2016-06-21 23:16:09 +02:00
|
|
|
Frame::dirtyList.add(&this->root->inDirtyList);
|
2016-06-17 00:06:37 +02:00
|
|
|
this->root->object.privateFlags |= HIERARCHYSYNC;
|
2016-06-21 23:16:09 +02:00
|
|
|
// Mark subtree as dirty as well
|
2016-06-17 00:06:37 +02:00
|
|
|
this->object.privateFlags |= SUBTREESYNC;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Frame::setHierarchyRoot(Frame *root)
|
|
|
|
{
|
|
|
|
this->root = root;
|
|
|
|
for(Frame *child = this->child; child; child = child->next)
|
|
|
|
child->setHierarchyRoot(root);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clone a frame hierarchy. Link cloned frames into Frame::root of the originals.
|
|
|
|
Frame*
|
|
|
|
Frame::cloneAndLink(Frame *clonedroot)
|
|
|
|
{
|
|
|
|
Frame *frame = Frame::create();
|
2016-06-21 23:16:09 +02:00
|
|
|
if(clonedroot == nil)
|
2016-06-17 00:06:37 +02:00
|
|
|
clonedroot = frame;
|
|
|
|
frame->object.copy(&this->object);
|
|
|
|
frame->matrix = this->matrix;
|
|
|
|
frame->root = clonedroot;
|
|
|
|
this->root = frame; // Remember cloned frame
|
|
|
|
for(Frame *child = this->child; child; child = child->next){
|
|
|
|
Frame *clonedchild = child->cloneAndLink(clonedroot);
|
|
|
|
clonedchild->next = frame->child;
|
|
|
|
frame->child = clonedchild;
|
|
|
|
clonedchild->object.parent = frame;
|
|
|
|
}
|
2016-06-25 17:58:52 +02:00
|
|
|
s_plglist.copy(frame, this);
|
2016-06-17 00:06:37 +02:00
|
|
|
return frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove links to cloned frames from hierarchy.
|
|
|
|
void
|
|
|
|
Frame::purgeClone(void)
|
|
|
|
{
|
|
|
|
Frame *parent = this->getParent();
|
|
|
|
this->setHierarchyRoot(parent ? parent->root : this);
|
|
|
|
}
|
|
|
|
|
2016-06-17 13:29:49 +02:00
|
|
|
struct FrameStreamData
|
|
|
|
{
|
|
|
|
V3d right, up, at, pos;
|
|
|
|
int32 parent;
|
|
|
|
int32 matflag;
|
|
|
|
};
|
|
|
|
|
|
|
|
FrameList_*
|
|
|
|
FrameList_::streamRead(Stream *stream)
|
|
|
|
{
|
|
|
|
FrameStreamData buf;
|
|
|
|
this->numFrames = 0;
|
|
|
|
this->frames = nil;
|
|
|
|
if(!findChunk(stream, ID_STRUCT, nil, nil)){
|
|
|
|
RWERROR((ERR_CHUNK, "STRUCT"));
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
this->numFrames = stream->readI32();
|
2016-06-17 16:20:02 +02:00
|
|
|
this->frames = (Frame**)malloc(this->numFrames*sizeof(Frame*));
|
|
|
|
if(this->frames == nil){
|
2016-06-17 13:29:49 +02:00
|
|
|
RWERROR((ERR_ALLOC, this->numFrames*sizeof(Frame*)));
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
for(int32 i = 0; i < this->numFrames; i++){
|
|
|
|
Frame *f;
|
|
|
|
stream->read(&buf, sizeof(buf));
|
2016-06-17 16:20:02 +02:00
|
|
|
this->frames[i] = f = Frame::create();
|
2016-06-17 13:29:49 +02:00
|
|
|
if(f == nil){
|
|
|
|
// TODO: clean up frames?
|
|
|
|
free(this->frames);
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
f->matrix.right = buf.right;
|
|
|
|
f->matrix.rightw = 0.0f;
|
|
|
|
f->matrix.up = buf.up;
|
|
|
|
f->matrix.upw = 0.0f;
|
|
|
|
f->matrix.at = buf.at;
|
|
|
|
f->matrix.atw = 0.0f;
|
|
|
|
f->matrix.pos = buf.pos;
|
|
|
|
f->matrix.posw = 1.0f;
|
|
|
|
//f->matflag = buf.matflag;
|
|
|
|
if(buf.parent >= 0)
|
2016-07-19 18:40:10 +02:00
|
|
|
this->frames[buf.parent]->addChild(f, rw::streamAppendFrames);
|
2016-06-17 13:29:49 +02:00
|
|
|
}
|
|
|
|
for(int32 i = 0; i < this->numFrames; i++)
|
2016-06-25 17:58:52 +02:00
|
|
|
Frame::s_plglist.streamRead(stream, this->frames[i]);
|
2016-06-17 13:29:49 +02:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
FrameList_::streamWrite(Stream *stream)
|
|
|
|
{
|
|
|
|
FrameStreamData buf;
|
|
|
|
|
|
|
|
int size = 0, structsize = 0;
|
|
|
|
structsize = 4 + this->numFrames*sizeof(FrameStreamData);
|
|
|
|
size += 12 + structsize;
|
|
|
|
for(int32 i = 0; i < this->numFrames; i++)
|
2016-06-25 17:58:52 +02:00
|
|
|
size += 12 + Frame::s_plglist.streamGetSize(this->frames[i]);
|
2016-06-17 13:29:49 +02:00
|
|
|
|
|
|
|
writeChunkHeader(stream, ID_FRAMELIST, size);
|
|
|
|
writeChunkHeader(stream, ID_STRUCT, structsize);
|
|
|
|
stream->writeU32(this->numFrames);
|
|
|
|
for(int32 i = 0; i < this->numFrames; i++){
|
|
|
|
Frame *f = this->frames[i];
|
|
|
|
buf.right = f->matrix.right;
|
|
|
|
buf.up = f->matrix.up;
|
|
|
|
buf.at = f->matrix.at;
|
|
|
|
buf.pos = f->matrix.pos;
|
|
|
|
buf.parent = findPointer(f->getParent(), (void**)this->frames,
|
|
|
|
this->numFrames);
|
|
|
|
buf.matflag = 0; //f->matflag;
|
|
|
|
stream->write(&buf, sizeof(buf));
|
|
|
|
}
|
|
|
|
for(int32 i = 0; i < this->numFrames; i++)
|
2016-06-25 17:58:52 +02:00
|
|
|
Frame::s_plglist.streamWrite(stream, this->frames[i]);
|
2016-06-17 13:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static Frame*
|
|
|
|
sizeCB(Frame *f, void *size)
|
|
|
|
{
|
2016-06-25 17:58:52 +02:00
|
|
|
*(int32*)size += Frame::s_plglist.streamGetSize(f);
|
2016-06-17 13:29:49 +02:00
|
|
|
f->forAllChildren(sizeCB, size);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32
|
|
|
|
FrameList_::streamGetSize(Frame *f)
|
|
|
|
{
|
|
|
|
int32 numFrames = f->count();
|
|
|
|
uint32 size = 12 + 12 + 4 + numFrames*(sizeof(FrameStreamData)+12);
|
|
|
|
sizeCB(f, (void*)&size);
|
2016-06-17 16:20:02 +02:00
|
|
|
return size;
|
2016-06-17 13:29:49 +02:00
|
|
|
}
|
|
|
|
|
2016-06-17 00:06:37 +02:00
|
|
|
Frame**
|
|
|
|
makeFrameList(Frame *frame, Frame **flist)
|
|
|
|
{
|
|
|
|
*flist++ = frame;
|
|
|
|
if(frame->next)
|
|
|
|
flist = makeFrameList(frame->next, flist);
|
|
|
|
if(frame->child)
|
|
|
|
flist = makeFrameList(frame->child, flist);
|
|
|
|
return flist;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|