2015-01-31 15:07:47 +01:00
|
|
|
/*
|
|
|
|
Copyright (c) 2015, Matthias Schiffer <mschiffer@universe-factory.net>
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer in the documentation
|
|
|
|
and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
|
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2015-02-02 17:56:43 +01:00
|
|
|
#include "Info.hpp"
|
2015-02-02 18:33:21 +01:00
|
|
|
#include "World/Level.hpp"
|
2015-02-01 00:21:17 +01:00
|
|
|
#include "World/Region.hpp"
|
|
|
|
|
2015-02-01 11:58:50 +01:00
|
|
|
#include <cerrno>
|
2015-02-02 02:25:56 +01:00
|
|
|
#include <climits>
|
2015-02-01 11:58:50 +01:00
|
|
|
#include <cstdio>
|
2015-02-02 02:25:56 +01:00
|
|
|
#include <cstring>
|
|
|
|
#include <cstdlib>
|
2015-02-01 00:21:17 +01:00
|
|
|
#include <iostream>
|
2015-02-02 04:47:40 +01:00
|
|
|
#include <set>
|
2015-02-01 11:58:50 +01:00
|
|
|
#include <system_error>
|
2015-02-01 00:21:17 +01:00
|
|
|
|
2015-02-02 02:25:56 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
|
2015-02-01 11:58:50 +01:00
|
|
|
#include <arpa/inet.h>
|
2015-02-02 02:25:56 +01:00
|
|
|
#include <dirent.h>
|
2015-02-02 03:06:22 +01:00
|
|
|
#include <fcntl.h>
|
2015-02-01 11:58:50 +01:00
|
|
|
#include <png.h>
|
2015-02-02 03:06:22 +01:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <unistd.h>
|
2015-02-01 00:21:17 +01:00
|
|
|
|
|
|
|
|
2015-02-01 11:58:50 +01:00
|
|
|
using namespace MinedMap;
|
|
|
|
|
|
|
|
|
2015-02-02 01:41:17 +01:00
|
|
|
static const size_t DIM = World::Region::SIZE*World::Chunk::SIZE;
|
|
|
|
|
|
|
|
|
|
|
|
static void addChunk(uint32_t image[DIM][DIM], size_t X, size_t Z, const World::Chunk *chunk) {
|
|
|
|
World::Chunk::Blocks layer = chunk->getTopLayer();
|
|
|
|
|
|
|
|
for (size_t x = 0; x < World::Chunk::SIZE; x++) {
|
|
|
|
for (size_t z = 0; z < World::Chunk::SIZE; z++)
|
|
|
|
image[Z*World::Chunk::SIZE+z][X*World::Chunk::SIZE+x] = htonl(layer.blocks[x][z].getColor());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void writePNG(const char *filename, const uint32_t data[DIM][DIM]) {
|
2015-02-01 11:58:50 +01:00
|
|
|
std::FILE *f = std::fopen(filename, "wb");
|
|
|
|
if (!f)
|
|
|
|
throw std::system_error(errno, std::generic_category(), "unable to open output file");
|
|
|
|
|
|
|
|
png_structp png_ptr = png_create_write_struct (PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
|
|
|
|
if (!png_ptr)
|
|
|
|
throw std::runtime_error("unable to create PNG write struct");
|
|
|
|
|
|
|
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
if (!info_ptr) {
|
|
|
|
png_destroy_write_struct(&png_ptr, nullptr);
|
|
|
|
throw std::runtime_error("unable to create PNG info struct");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr))) {
|
|
|
|
png_destroy_write_struct(&png_ptr, &info_ptr);
|
2015-02-02 18:52:33 +01:00
|
|
|
std::fclose(f);
|
2015-02-01 11:58:50 +01:00
|
|
|
throw std::runtime_error("unable to write PNG file");
|
|
|
|
}
|
|
|
|
|
|
|
|
png_init_io(png_ptr, f);
|
|
|
|
|
2015-02-02 01:47:16 +01:00
|
|
|
png_set_IHDR(png_ptr, info_ptr, DIM, DIM, 8, PNG_COLOR_TYPE_RGB_ALPHA,
|
|
|
|
PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT);
|
2015-02-01 11:58:50 +01:00
|
|
|
|
|
|
|
uint8_t *row_pointers[World::Region::SIZE*World::Chunk::SIZE];
|
|
|
|
for (size_t i = 0; i < World::Region::SIZE*World::Chunk::SIZE; i++)
|
|
|
|
row_pointers[i] = const_cast<uint8_t*>(reinterpret_cast<const uint8_t*>(data[i]));
|
|
|
|
|
|
|
|
png_set_rows(png_ptr, info_ptr, row_pointers);
|
|
|
|
png_write_png(png_ptr, info_ptr, PNG_TRANSFORM_IDENTITY, NULL);
|
|
|
|
|
2015-02-02 18:52:33 +01:00
|
|
|
png_destroy_write_struct(&png_ptr, &info_ptr);
|
2015-02-01 11:58:50 +01:00
|
|
|
std::fclose(f);
|
|
|
|
}
|
|
|
|
|
2015-02-02 02:25:56 +01:00
|
|
|
static void doRegion(const std::string &input, const std::string &output) {
|
2015-02-02 03:06:22 +01:00
|
|
|
struct stat instat, outstat;
|
|
|
|
|
|
|
|
if (stat(input.c_str(), &instat) < 0) {
|
|
|
|
std::fprintf(stderr, "Unable to stat %s: %s\n", input.c_str(), std::strerror(errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stat(output.c_str(), &outstat) == 0) {
|
|
|
|
if (instat.st_mtim.tv_sec < outstat.st_mtim.tv_sec ||
|
|
|
|
(instat.st_mtim.tv_sec == outstat.st_mtim.tv_sec && instat.st_mtim.tv_nsec <= outstat.st_mtim.tv_nsec)) {
|
2015-02-02 03:53:49 +01:00
|
|
|
std::printf("%s is up-to-date.\n", output.c_str());
|
2015-02-02 03:06:22 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-02 03:53:49 +01:00
|
|
|
std::printf("Generating %s from %s...\n", output.c_str(), input.c_str());
|
2015-02-02 02:25:56 +01:00
|
|
|
|
2015-02-02 03:06:22 +01:00
|
|
|
const std::string tmpfile = output + ".tmp";
|
|
|
|
|
2015-02-02 03:38:49 +01:00
|
|
|
try {
|
|
|
|
uint32_t image[DIM][DIM] = {};
|
|
|
|
World::Region::visitChunks(input.c_str(), [&image] (size_t X, size_t Z, const World::Chunk *chunk) { addChunk(image, X, Z, chunk); });
|
2015-02-02 03:06:22 +01:00
|
|
|
|
2015-02-02 03:38:49 +01:00
|
|
|
writePNG(tmpfile.c_str(), image);
|
2015-02-02 03:06:22 +01:00
|
|
|
|
2015-02-02 03:38:49 +01:00
|
|
|
struct timespec times[2] = {instat.st_mtim, instat.st_mtim};
|
|
|
|
if (utimensat(AT_FDCWD, tmpfile.c_str(), times, 0) < 0)
|
|
|
|
std::fprintf(stderr, "Warning: failed to set utime on %s: %s\n", tmpfile.c_str(), std::strerror(errno));
|
2015-02-02 03:06:22 +01:00
|
|
|
|
2015-02-02 03:38:49 +01:00
|
|
|
if (std::rename(tmpfile.c_str(), output.c_str()) < 0) {
|
|
|
|
std::fprintf(stderr, "Unable to save %s: %s\n", output.c_str(), std::strerror(errno));
|
|
|
|
unlink(tmpfile.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (const std::exception& ex) {
|
|
|
|
std::fprintf(stderr, "Failed to generate %s: %s\n", output.c_str(), ex.what());
|
2015-02-02 03:06:22 +01:00
|
|
|
unlink(tmpfile.c_str());
|
|
|
|
}
|
2015-02-02 02:25:56 +01:00
|
|
|
}
|
|
|
|
|
2015-02-02 04:47:40 +01:00
|
|
|
static bool checkFilename(const char *name, int *x, int *z) {
|
|
|
|
if (std::sscanf(name, "r.%i.%i.mca", x, z) != 2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
size_t l = strlen(name) + 1;
|
|
|
|
char buf[l];
|
|
|
|
std::snprintf(buf, l, "r.%i.%i.mca", *x, *z);
|
|
|
|
if (std::memcmp(name, buf, l))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-02-01 11:58:50 +01:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
if (argc < 3) {
|
2015-02-02 02:25:56 +01:00
|
|
|
std::fprintf(stderr, "Usage: %s <data directory> <output directory>\n", argv[0]);
|
2015-02-01 00:21:17 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-02-02 02:25:56 +01:00
|
|
|
std::string inputdir(argv[1]);
|
2015-02-02 18:33:21 +01:00
|
|
|
std::string regiondir = inputdir + "/region";
|
2015-02-02 02:25:56 +01:00
|
|
|
|
|
|
|
std::string outputdir(argv[2]);
|
2015-02-01 11:58:50 +01:00
|
|
|
|
2015-02-02 18:33:21 +01:00
|
|
|
DIR *dir = opendir(regiondir.c_str());
|
2015-02-02 02:25:56 +01:00
|
|
|
if (!dir) {
|
|
|
|
std::fprintf(stderr, "Unable to read input directory: %s\n", std::strerror(errno));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-02-02 17:56:43 +01:00
|
|
|
Info info;
|
2015-02-02 02:25:56 +01:00
|
|
|
|
|
|
|
struct dirent *entry;
|
|
|
|
while ((entry = readdir(dir)) != nullptr) {
|
2015-02-02 03:33:43 +01:00
|
|
|
int x, z;
|
2015-02-02 04:47:40 +01:00
|
|
|
if (!checkFilename(entry->d_name, &x, &z))
|
|
|
|
continue;
|
|
|
|
|
2015-02-02 17:56:43 +01:00
|
|
|
info.addRegion(x, z);
|
2015-02-02 04:47:40 +01:00
|
|
|
|
|
|
|
std::string name(entry->d_name);
|
2015-02-02 18:33:21 +01:00
|
|
|
doRegion(regiondir + "/" + name, outputdir + "/" + name.substr(0, name.length()-3) + "png");
|
2015-02-02 02:25:56 +01:00
|
|
|
}
|
2015-02-01 11:58:50 +01:00
|
|
|
|
2015-02-02 02:25:56 +01:00
|
|
|
closedir(dir);
|
2015-02-01 01:38:20 +01:00
|
|
|
|
2015-02-02 18:33:21 +01:00
|
|
|
World::Level level((inputdir + "/level.dat").c_str());
|
|
|
|
info.setSpawn(level.getSpawn());
|
|
|
|
|
2015-02-02 17:56:43 +01:00
|
|
|
info.writeJSON((outputdir + "/info.json").c_str());
|
2015-02-02 05:13:07 +01:00
|
|
|
|
2015-01-31 15:07:47 +01:00
|
|
|
return 0;
|
|
|
|
}
|