123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- import decodeGoogleEarthEnterpriseData from '../Core/decodeGoogleEarthEnterpriseData.js';
- import GoogleEarthEnterpriseTileInformation from '../Core/GoogleEarthEnterpriseTileInformation.js';
- import RuntimeError from '../Core/RuntimeError.js';
- import pako from '../ThirdParty/pako_inflate.js';
- import createTaskProcessorWorker from './createTaskProcessorWorker.js';
- // Datatype sizes
- var sizeOfUint16 = Uint16Array.BYTES_PER_ELEMENT;
- var sizeOfInt32 = Int32Array.BYTES_PER_ELEMENT;
- var sizeOfUint32 = Uint32Array.BYTES_PER_ELEMENT;
- var Types = {
- METADATA : 0,
- TERRAIN : 1,
- DBROOT : 2
- };
- Types.fromString = function(s) {
- if (s === 'Metadata') {
- return Types.METADATA;
- } else if (s === 'Terrain') {
- return Types.TERRAIN;
- } else if (s === 'DbRoot') {
- return Types.DBROOT;
- }
- };
- function decodeGoogleEarthEnterprisePacket(parameters, transferableObjects) {
- var type = Types.fromString(parameters.type);
- var buffer = parameters.buffer;
- decodeGoogleEarthEnterpriseData(parameters.key, buffer);
- var uncompressedTerrain = uncompressPacket(buffer);
- buffer = uncompressedTerrain.buffer;
- var length = uncompressedTerrain.length;
- switch (type) {
- case Types.METADATA:
- return processMetadata(buffer, length, parameters.quadKey);
- case Types.TERRAIN:
- return processTerrain(buffer, length, transferableObjects);
- case Types.DBROOT:
- transferableObjects.push(buffer);
- return {
- buffer : buffer
- };
- }
- }
- var qtMagic = 32301;
- function processMetadata(buffer, totalSize, quadKey) {
- var dv = new DataView(buffer);
- var offset = 0;
- var magic = dv.getUint32(offset, true);
- offset += sizeOfUint32;
- if (magic !== qtMagic) {
- throw new RuntimeError('Invalid magic');
- }
- var dataTypeId = dv.getUint32(offset, true);
- offset += sizeOfUint32;
- if (dataTypeId !== 1) {
- throw new RuntimeError('Invalid data type. Must be 1 for QuadTreePacket');
- }
- // Tile format version
- var quadVersion = dv.getUint32(offset, true);
- offset += sizeOfUint32;
- if (quadVersion !== 2) {
- throw new RuntimeError('Invalid QuadTreePacket version. Only version 2 is supported.');
- }
- var numInstances = dv.getInt32(offset, true);
- offset += sizeOfInt32;
- var dataInstanceSize = dv.getInt32(offset, true);
- offset += sizeOfInt32;
- if (dataInstanceSize !== 32) {
- throw new RuntimeError('Invalid instance size.');
- }
- var dataBufferOffset = dv.getInt32(offset, true);
- offset += sizeOfInt32;
- var dataBufferSize = dv.getInt32(offset, true);
- offset += sizeOfInt32;
- var metaBufferSize = dv.getInt32(offset, true);
- offset += sizeOfInt32;
- // Offset from beginning of packet (instances + current offset)
- if (dataBufferOffset !== (numInstances * dataInstanceSize + offset)) {
- throw new RuntimeError('Invalid dataBufferOffset');
- }
- // Verify the packets is all there header + instances + dataBuffer + metaBuffer
- if (dataBufferOffset + dataBufferSize + metaBufferSize !== totalSize) {
- throw new RuntimeError('Invalid packet offsets');
- }
- // Read all the instances
- var instances = [];
- for (var i = 0; i < numInstances; ++i) {
- var bitfield = dv.getUint8(offset);
- ++offset;
- ++offset; // 2 byte align
- var cnodeVersion = dv.getUint16(offset, true);
- offset += sizeOfUint16;
- var imageVersion = dv.getUint16(offset, true);
- offset += sizeOfUint16;
- var terrainVersion = dv.getUint16(offset, true);
- offset += sizeOfUint16;
- // Number of channels stored in the dataBuffer
- offset += sizeOfUint16;
- offset += sizeOfUint16; // 4 byte align
- // Channel type offset into dataBuffer
- offset += sizeOfInt32;
- // Channel version offset into dataBuffer
- offset += sizeOfInt32;
- offset += 8; // Ignore image neighbors for now
- // Data providers
- var imageProvider = dv.getUint8(offset++);
- var terrainProvider = dv.getUint8(offset++);
- offset += sizeOfUint16; // 4 byte align
- instances.push(new GoogleEarthEnterpriseTileInformation(bitfield, cnodeVersion,
- imageVersion, terrainVersion, imageProvider, terrainProvider));
- }
- var tileInfo = [];
- var index = 0;
- function populateTiles(parentKey, parent, level) {
- var isLeaf = false;
- if (level === 4) {
- if (parent.hasSubtree()) {
- return; // We have a subtree, so just return
- }
- isLeaf = true; // No subtree, so set all children to null
- }
- for (var i = 0; i < 4; ++i) {
- var childKey = parentKey + i.toString();
- if (isLeaf) {
- // No subtree so set all children to null
- tileInfo[childKey] = null;
- } else if (level < 4) {
- // We are still in the middle of the subtree, so add child
- // only if their bits are set, otherwise set child to null.
- if (!parent.hasChild(i)) {
- tileInfo[childKey] = null;
- } else {
- if (index === numInstances) {
- console.log('Incorrect number of instances');
- return;
- }
- var instance = instances[index++];
- tileInfo[childKey] = instance;
- populateTiles(childKey, instance, level + 1);
- }
- }
- }
- }
- var level = 0;
- var root = instances[index++];
- if (quadKey === '') {
- // Root tile has data at its root and one less level
- ++level;
- } else {
- tileInfo[quadKey] = root; // This will only contain the child bitmask
- }
- populateTiles(quadKey, root, level);
- return tileInfo;
- }
- function processTerrain(buffer, totalSize, transferableObjects) {
- var dv = new DataView(buffer);
- var offset = 0;
- var terrainTiles = [];
- while (offset < totalSize) {
- // Each tile is split into 4 parts
- var tileStart = offset;
- for (var quad = 0; quad < 4; ++quad) {
- var size = dv.getUint32(offset, true);
- offset += sizeOfUint32;
- offset += size;
- }
- var tile = buffer.slice(tileStart, offset);
- transferableObjects.push(tile);
- terrainTiles.push(tile);
- }
- return terrainTiles;
- }
- var compressedMagic = 0x7468dead;
- var compressedMagicSwap = 0xadde6874;
- function uncompressPacket(data) {
- // The layout of this decoded data is
- // Magic Uint32
- // Size Uint32
- // [GZipped chunk of Size bytes]
- // Pullout magic and verify we have the correct data
- var dv = new DataView(data);
- var offset = 0;
- var magic = dv.getUint32(offset, true);
- offset += sizeOfUint32;
- if (magic !== compressedMagic && magic !== compressedMagicSwap) {
- throw new RuntimeError('Invalid magic');
- }
- // Get the size of the compressed buffer - the endianness depends on which magic was used
- var size = dv.getUint32(offset, (magic === compressedMagic));
- offset += sizeOfUint32;
- var compressedPacket = new Uint8Array(data, offset);
- var uncompressedPacket = pako.inflate(compressedPacket);
- if (uncompressedPacket.length !== size) {
- throw new RuntimeError('Size of packet doesn\'t match header');
- }
- return uncompressedPacket;
- }
- export default createTaskProcessorWorker(decodeGoogleEarthEnterprisePacket);
|