metaforce/DataSpec/DNACommon/ANIM.cpp

496 lines
17 KiB
C++
Raw Normal View History

2016-03-04 15:04:53 -08:00
#include "zeus/Math.hpp"
2015-08-11 16:32:02 -07:00
#include "ANIM.hpp"
2016-02-13 01:02:47 -08:00
namespace DataSpec
2015-08-11 16:32:02 -07:00
{
namespace DNAANIM
{
size_t ComputeBitstreamSize(size_t keyFrameCount, const std::vector<Channel>& channels)
{
size_t bitsPerKeyFrame = 0;
for (const Channel& chan : channels)
{
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-13 14:29:07 -07:00
bitsPerKeyFrame += 1;
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
bitsPerKeyFrame += chan.q[0];
bitsPerKeyFrame += chan.q[1];
bitsPerKeyFrame += chan.q[2];
break;
2015-11-20 17:16:07 -08:00
case Channel::Type::KfHead:
2015-09-25 20:12:08 -07:00
bitsPerKeyFrame += 1;
break;
2015-11-20 17:16:07 -08:00
case Channel::Type::RotationMP3:
2015-09-25 20:12:08 -07:00
bitsPerKeyFrame += chan.q[0];
bitsPerKeyFrame += chan.q[1];
bitsPerKeyFrame += chan.q[2];
bitsPerKeyFrame += chan.q[3];
break;
2015-08-11 16:32:02 -07:00
default: break;
}
}
return (bitsPerKeyFrame * keyFrameCount + 31) / 32 * 4;
}
static inline QuantizedRot QuantizeRotation(const Value& quat, atUint32 div)
{
2016-03-04 15:04:53 -08:00
float q = M_PI / 2.0f / float(div);
2015-08-11 16:32:02 -07:00
return
{
{
2016-03-04 15:04:53 -08:00
atInt16(std::asin(quat.v4.vec[1]) / q),
atInt16(std::asin(quat.v4.vec[2]) / q),
atInt16(std::asin(quat.v4.vec[3]) / q),
2015-08-11 16:32:02 -07:00
},
2016-03-04 15:04:53 -08:00
(quat.v4.vec[0] < 0.f) ? true : false
2015-08-11 16:32:02 -07:00
};
}
2015-09-25 20:12:08 -07:00
2015-08-11 16:32:02 -07:00
static inline Value DequantizeRotation(const QuantizedRot& v, atUint32 div)
{
2016-03-04 15:04:53 -08:00
float q = M_PI / 2.0f / float(div);
2015-08-11 16:32:02 -07:00
Value retval =
{
2016-03-04 15:04:53 -08:00
0.0f,
std::sin(v.v[0] * q),
std::sin(v.v[1] * q),
std::sin(v.v[2] * q),
2015-08-11 16:32:02 -07:00
};
2016-03-04 15:04:53 -08:00
retval.v4.vec[0] = std::sqrt(std::max((1.0f -
2015-08-13 14:29:07 -07:00
(retval.v4.vec[1] * retval.v4.vec[1] +
retval.v4.vec[2] * retval.v4.vec[2] +
2016-03-04 15:04:53 -08:00
retval.v4.vec[3] * retval.v4.vec[3])), 0.0f));
2015-08-13 14:29:07 -07:00
retval.v4.vec[0] = v.w ? -retval.v4.vec[0] : retval.v4.vec[0];
2015-08-11 16:32:02 -07:00
return retval;
}
2015-09-27 14:49:18 -07:00
static inline Value DequantizeRotation_3(const QuantizedRot& v, atUint32 div)
2015-09-25 20:12:08 -07:00
{
2016-03-04 15:04:53 -08:00
float q = 1.0f / float(div);
2015-09-25 20:12:08 -07:00
Value retval =
{
2016-03-04 15:04:53 -08:00
0.0f,
2015-09-25 20:12:08 -07:00
v.v[0] * q,
v.v[1] * q,
v.v[2] * q,
};
2016-03-04 15:04:53 -08:00
retval.v4.vec[0] = std::sqrt(std::max((1.0f -
2015-09-27 14:49:18 -07:00
(retval.v4.vec[1] * retval.v4.vec[1] +
retval.v4.vec[2] * retval.v4.vec[2] +
2016-03-04 15:04:53 -08:00
retval.v4.vec[3] * retval.v4.vec[3])), 0.0f));
2015-09-27 14:49:18 -07:00
retval.v4.vec[0] = v.w ? -retval.v4.vec[0] : retval.v4.vec[0];
2015-09-25 20:12:08 -07:00
return retval;
}
2015-08-13 14:29:07 -07:00
bool BitstreamReader::dequantizeBit(const atUint8* data)
{
atUint32 byteCur = (m_bitCur / 32) * 4;
atUint32 bitRem = m_bitCur % 32;
/* Fill 32 bit buffer with region containing bits */
/* Make them least significant */
2016-03-04 15:04:53 -08:00
atUint32 tempBuf = hecl::SBig(*reinterpret_cast<const atUint32*>(data + byteCur)) >> bitRem;
2015-08-13 14:29:07 -07:00
/* That's it */
m_bitCur += 1;
return tempBuf & 0x1;
}
2015-08-11 16:32:02 -07:00
atInt16 BitstreamReader::dequantize(const atUint8* data, atUint8 q)
{
atUint32 byteCur = (m_bitCur / 32) * 4;
atUint32 bitRem = m_bitCur % 32;
/* Fill 32 bit buffer with region containing bits */
/* Make them least significant */
2016-03-04 15:04:53 -08:00
atUint32 tempBuf = hecl::SBig(*reinterpret_cast<const atUint32*>(data + byteCur)) >> bitRem;
2015-08-11 16:32:02 -07:00
/* If this shift underflows the value, buffer the next 32 bits */
/* And tack onto shifted buffer */
if ((bitRem + q) > 32)
{
2016-03-04 15:04:53 -08:00
atUint32 tempBuf2 = hecl::SBig(*reinterpret_cast<const atUint32*>(data + byteCur + 4));
2015-08-11 16:32:02 -07:00
tempBuf |= (tempBuf2 << (32 - bitRem));
}
2015-09-26 19:24:03 -07:00
/* Mask it */
atUint32 mask = (1 << q) - 1;
tempBuf &= mask;
/* Sign extend */
2015-08-11 16:32:02 -07:00
atUint32 sign = (tempBuf >> (q - 1)) & 0x1;
if (sign)
2015-09-26 19:24:03 -07:00
tempBuf |= ~0 << q;
2015-08-11 16:32:02 -07:00
/* Return delta value */
m_bitCur += q;
2015-09-26 19:24:03 -07:00
return atInt32(tempBuf);
2015-08-11 16:32:02 -07:00
}
std::vector<std::vector<Value>>
BitstreamReader::read(const atUint8* data,
size_t keyFrameCount,
const std::vector<Channel>& channels,
atUint32 rotDiv,
float transMult)
{
m_bitCur = 0;
std::vector<std::vector<Value>> chanKeys;
std::vector<QuantizedValue> chanAccum;
chanKeys.reserve(channels.size());
chanAccum.reserve(channels.size());
for (const Channel& chan : channels)
{
2015-09-26 19:24:03 -07:00
chanAccum.push_back(chan.i);
2015-08-11 16:32:02 -07:00
chanKeys.emplace_back();
std::vector<Value>& keys = chanKeys.back();
keys.reserve(keyFrameCount);
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-11 16:32:02 -07:00
{
QuantizedRot qr = {{chan.i[0], chan.i[1], chan.i[2]}, false};
keys.emplace_back(DequantizeRotation(qr, rotDiv));
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
keys.push_back({chan.i[0] * transMult, chan.i[1] * transMult, chan.i[2] * transMult});
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
{
2015-09-27 17:42:47 -07:00
keys.push_back({chan.i[0] / float(rotDiv), chan.i[1] / float(rotDiv), chan.i[2] / float(rotDiv)});
2015-08-11 16:32:02 -07:00
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::KfHead:
2015-09-25 20:12:08 -07:00
{
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::RotationMP3:
2015-09-25 20:12:08 -07:00
{
2015-09-26 19:24:03 -07:00
QuantizedRot qr = {{chan.i[1], chan.i[2], chan.i[3]}, bool(chan.i[0] & 0x1)};
2015-09-27 14:49:18 -07:00
keys.emplace_back(DequantizeRotation_3(qr, rotDiv));
2015-09-25 20:12:08 -07:00
break;
}
2015-08-11 16:32:02 -07:00
default: break;
}
}
for (size_t f=0 ; f<keyFrameCount ; ++f)
{
auto kit = chanKeys.begin();
auto ait = chanAccum.begin();
for (const Channel& chan : channels)
{
QuantizedValue& p = *ait;
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-11 16:32:02 -07:00
{
2015-08-13 14:29:07 -07:00
bool wBit = dequantizeBit(data);
2015-08-11 16:32:02 -07:00
p[0] += dequantize(data, chan.q[0]);
p[1] += dequantize(data, chan.q[1]);
p[2] += dequantize(data, chan.q[2]);
QuantizedRot qr = {{p[0], p[1], p[2]}, wBit};
kit->emplace_back(DequantizeRotation(qr, rotDiv));
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
2015-09-26 19:24:03 -07:00
atInt16 val1 = dequantize(data, chan.q[0]);
p[0] += val1;
atInt16 val2 = dequantize(data, chan.q[1]);
p[1] += val2;
atInt16 val3 = dequantize(data, chan.q[2]);
p[2] += val3;
2015-08-11 16:32:02 -07:00
kit->push_back({p[0] * transMult, p[1] * transMult, p[2] * transMult});
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
{
p[0] += dequantize(data, chan.q[0]);
2015-08-14 21:12:15 -07:00
p[1] += dequantize(data, chan.q[1]);
p[2] += dequantize(data, chan.q[2]);
2015-09-27 17:42:47 -07:00
kit->push_back({p[0] / float(rotDiv), p[1] / float(rotDiv), p[2] / float(rotDiv)});
2015-08-11 16:32:02 -07:00
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::KfHead:
2015-09-25 20:12:08 -07:00
{
bool aBit = dequantizeBit(data);
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::RotationMP3:
2015-09-25 20:12:08 -07:00
{
2015-09-26 19:24:03 -07:00
atInt16 val1 = dequantize(data, chan.q[0]);
p[0] += val1;
atInt16 val2 = dequantize(data, chan.q[1]);
p[1] += val2;
atInt16 val3 = dequantize(data, chan.q[2]);
p[2] += val3;
atInt16 val4 = dequantize(data, chan.q[3]);
p[3] += val4;
QuantizedRot qr = {{p[1], p[2], p[3]}, bool(p[0] & 0x1)};
2015-09-27 14:49:18 -07:00
kit->emplace_back(DequantizeRotation_3(qr, rotDiv));
2015-09-25 20:12:08 -07:00
break;
}
2015-08-11 16:32:02 -07:00
default: break;
}
++kit;
++ait;
}
}
return chanKeys;
}
2015-08-13 14:29:07 -07:00
void BitstreamWriter::quantizeBit(atUint8* data, bool val)
{
atUint32 byteCur = (m_bitCur / 32) * 4;
atUint32 bitRem = m_bitCur % 32;
/* Fill 32 bit buffer with region containing bits */
/* Make them least significant */
*(atUint32*)(data + byteCur) =
2016-03-04 15:04:53 -08:00
hecl::SBig(hecl::SBig(*(atUint32*)(data + byteCur)) | (val << bitRem));
2015-08-13 14:29:07 -07:00
m_bitCur += 1;
}
2015-08-11 16:32:02 -07:00
void BitstreamWriter::quantize(atUint8* data, atUint8 q, atInt16 val)
{
atUint32 byteCur = (m_bitCur / 32) * 4;
atUint32 bitRem = m_bitCur % 32;
atUint32 masked = val & ((1 << q) - 1);
/* Fill 32 bit buffer with region containing bits */
/* Make them least significant */
*(atUint32*)(data + byteCur) =
2016-03-04 15:04:53 -08:00
hecl::SBig(hecl::SBig(*(atUint32*)(data + byteCur)) | (masked << bitRem));
2015-08-11 16:32:02 -07:00
/* If this shift underflows the value, buffer the next 32 bits */
/* And tack onto shifted buffer */
if ((bitRem + q) > 32)
{
*(atUint32*)(data + byteCur + 4) =
2016-03-04 15:04:53 -08:00
hecl::SBig(hecl::SBig(*(atUint32*)(data + byteCur + 4)) | (masked >> (32 - bitRem)));
2015-08-11 16:32:02 -07:00
}
m_bitCur += q;
}
std::unique_ptr<atUint8[]>
BitstreamWriter::write(const std::vector<std::vector<Value>>& chanKeys,
size_t keyFrameCount, std::vector<Channel>& channels,
atUint32& rotDivOut,
float& transMultOut,
size_t& sizeOut)
{
m_bitCur = 0;
rotDivOut = 32767; /* Normalized range of values */
/* Pre-pass to calculate translation multiplier */
2016-03-04 15:04:53 -08:00
float maxTransDiff = 0.0f;
2015-08-11 16:32:02 -07:00
auto kit = chanKeys.begin();
for (Channel& chan : channels)
{
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
const Value* last = &(*kit)[0];
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
const Value* current = &*it;
2015-09-25 20:12:08 -07:00
maxTransDiff = std::max(maxTransDiff, current->v3.vec[0] - last->v3.vec[0]);
maxTransDiff = std::max(maxTransDiff, current->v3.vec[1] - last->v3.vec[1]);
maxTransDiff = std::max(maxTransDiff, current->v3.vec[2] - last->v3.vec[2]);
2015-08-11 16:32:02 -07:00
last = current;
}
break;
}
default: break;
}
++kit;
}
transMultOut = maxTransDiff / 32767;
/* Output channel inits */
kit = chanKeys.begin();
for (Channel& chan : channels)
{
chan.q[0] = 1;
chan.q[1] = 1;
chan.q[2] = 1;
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-11 16:32:02 -07:00
{
QuantizedRot qr = QuantizeRotation((*kit)[0], rotDivOut);
chan.i = qr.v;
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
chan.i = {atInt16((*kit)[0].v3.vec[0] / transMultOut),
atInt16((*kit)[0].v3.vec[1] / transMultOut),
atInt16((*kit)[0].v3.vec[2] / transMultOut)};
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
{
2015-08-14 21:12:15 -07:00
chan.i = {atInt16((*kit)[0].v3.vec[0] * rotDivOut),
atInt16((*kit)[0].v3.vec[1] * rotDivOut),
atInt16((*kit)[0].v3.vec[2] * rotDivOut)};
2015-08-11 16:32:02 -07:00
break;
}
default: break;
}
++kit;
}
/* Pre-pass to analyze quantization factors for channels */
kit = chanKeys.begin();
for (Channel& chan : channels)
{
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-11 16:32:02 -07:00
{
QuantizedRot qrLast = QuantizeRotation((*kit)[0], rotDivOut);
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
QuantizedRot qrCur = QuantizeRotation(*it, rotDivOut);
2015-09-25 20:12:08 -07:00
chan.q[0] = std::max(chan.q[0], atUint8(ceilf(log2f(qrCur.v[0] - qrLast.v[0]))));
chan.q[1] = std::max(chan.q[1], atUint8(ceilf(log2f(qrCur.v[1] - qrLast.v[1]))));
chan.q[2] = std::max(chan.q[2], atUint8(ceilf(log2f(qrCur.v[2] - qrLast.v[2]))));
2015-08-11 16:32:02 -07:00
qrLast = qrCur;
}
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
QuantizedValue last = {atInt16((*kit)[0].v3.vec[0] / transMultOut),
2015-08-14 21:12:15 -07:00
atInt16((*kit)[0].v3.vec[1] / transMultOut),
atInt16((*kit)[0].v3.vec[2] / transMultOut)};
2015-08-11 16:32:02 -07:00
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
QuantizedValue cur = {atInt16(it->v3.vec[0] / transMultOut),
2015-08-14 21:12:15 -07:00
atInt16(it->v3.vec[1] / transMultOut),
atInt16(it->v3.vec[2] / transMultOut)};
2015-09-25 20:12:08 -07:00
chan.q[0] = std::max(chan.q[0], atUint8(ceilf(log2f(cur[0] - last[0]))));
chan.q[1] = std::max(chan.q[1], atUint8(ceilf(log2f(cur[1] - last[1]))));
chan.q[2] = std::max(chan.q[2], atUint8(ceilf(log2f(cur[2] - last[2]))));
2015-08-11 16:32:02 -07:00
last = cur;
}
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
{
2015-08-14 21:12:15 -07:00
QuantizedValue last = {atInt16((*kit)[0].v3.vec[0] * rotDivOut),
atInt16((*kit)[0].v3.vec[1] * rotDivOut),
atInt16((*kit)[0].v3.vec[2] * rotDivOut)};
2015-08-11 16:32:02 -07:00
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
2015-08-14 21:12:15 -07:00
QuantizedValue cur = {atInt16(it->v3.vec[0] * rotDivOut),
atInt16(it->v3.vec[1] * rotDivOut),
atInt16(it->v3.vec[2] * rotDivOut)};
2015-09-25 20:12:08 -07:00
chan.q[0] = std::max(chan.q[0], atUint8(ceilf(log2f(cur[0] - last[0]))));
chan.q[1] = std::max(chan.q[1], atUint8(ceilf(log2f(cur[1] - last[1]))));
chan.q[2] = std::max(chan.q[2], atUint8(ceilf(log2f(cur[2] - last[2]))));
2015-08-11 16:32:02 -07:00
last = cur;
}
break;
}
default: break;
}
++kit;
}
/* Generate Bitstream */
sizeOut = ComputeBitstreamSize(keyFrameCount, channels);
atUint8* newData = new atUint8[sizeOut];
for (size_t f=0 ; f<keyFrameCount ; ++f)
{
kit = chanKeys.begin();
for (const Channel& chan : channels)
{
switch (chan.type)
{
2015-11-20 17:16:07 -08:00
case Channel::Type::Rotation:
2015-08-11 16:32:02 -07:00
{
QuantizedRot qrLast = QuantizeRotation((*kit)[0], rotDivOut);
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
QuantizedRot qrCur = QuantizeRotation(*it, rotDivOut);
2015-08-13 14:29:07 -07:00
quantizeBit(newData, qrCur.w);
2015-08-11 16:32:02 -07:00
quantize(newData, chan.q[0], qrCur.v[0] - qrLast.v[0]);
2015-10-03 22:08:56 -07:00
quantize(newData, chan.q[1], qrCur.v[1] - qrLast.v[1]);
quantize(newData, chan.q[2], qrCur.v[2] - qrLast.v[2]);
2015-08-11 16:32:02 -07:00
qrLast = qrCur;
}
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Translation:
2015-08-11 16:32:02 -07:00
{
QuantizedValue last = {atInt16((*kit)[0].v3.vec[0] / transMultOut),
2015-08-14 21:12:15 -07:00
atInt16((*kit)[0].v3.vec[1] / transMultOut),
atInt16((*kit)[0].v3.vec[2] / transMultOut)};
2015-08-11 16:32:02 -07:00
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
QuantizedValue cur = {atInt16(it->v3.vec[0] / transMultOut),
2015-08-14 21:12:15 -07:00
atInt16(it->v3.vec[1] / transMultOut),
atInt16(it->v3.vec[2] / transMultOut)};
2015-08-11 16:32:02 -07:00
quantize(newData, chan.q[0], cur[0] - last[0]);
2015-10-03 22:08:56 -07:00
quantize(newData, chan.q[1], cur[1] - last[1]);
quantize(newData, chan.q[2], cur[2] - last[2]);
2015-08-11 16:32:02 -07:00
last = cur;
}
break;
}
2015-11-20 17:16:07 -08:00
case Channel::Type::Scale:
2015-08-11 16:32:02 -07:00
{
2015-08-14 21:12:15 -07:00
QuantizedValue last = {atInt16((*kit)[0].v3.vec[0] * rotDivOut),
atInt16((*kit)[0].v3.vec[1] * rotDivOut),
atInt16((*kit)[0].v3.vec[2] * rotDivOut)};
2015-08-11 16:32:02 -07:00
for (auto it=kit->begin() + 1;
it != kit->end();
++it)
{
2015-08-14 21:12:15 -07:00
QuantizedValue cur = {atInt16(it->v3.vec[0] * rotDivOut),
atInt16(it->v3.vec[1] * rotDivOut),
atInt16(it->v3.vec[2] * rotDivOut)};
quantize(newData, chan.q[0], cur[0] - last[0]);
2015-10-03 22:08:56 -07:00
quantize(newData, chan.q[1], cur[1] - last[1]);
quantize(newData, chan.q[2], cur[2] - last[2]);
2015-08-11 16:32:02 -07:00
last = cur;
}
break;
}
default: break;
}
++kit;
}
}
return std::unique_ptr<atUint8[]>(newData);
}
}
}