WorldGenerator.cpp 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. #include "WorldGenerator.h"
  2. #include <Betriebssystem.h>
  3. #include <functional>
  4. #include <Logging.h>
  5. #include "Dimension.h"
  6. #include "Game.h"
  7. #include "JsonUtils.h"
  8. #include "NoiseInterpolator.h"
  9. using namespace Framework;
  10. using namespace Framework::JSON;
  11. using namespace Framework::Validator;
  12. WorldGenerator::WorldGenerator(int seed)
  13. : Thread(),
  14. exit(0),
  15. seed(seed),
  16. chunksPerSecond(0.0)
  17. {
  18. setName("World Generator");
  19. Framework::Logging::info()
  20. << "loading world generator configs. Changes at the config files "
  21. "may lead to a sudden change in landscape.";
  22. DataValidator* configValidator
  23. = DataValidator::buildForArray()
  24. ->removeInvalidEntries()
  25. ->addAcceptedTypeInArray(Game::INSTANCE->zTypeRegistry()
  26. ->getValidator<DimensionGenerator>())
  27. ->finishArray();
  28. loadAllJsonsFromDirectory("data/generator",
  29. [this, configValidator, seed](
  30. Framework::JSON::JSONValue* zValue, Framework::Text path) {
  31. Framework::Logging::info()
  32. << "loading dimension configs from '" << path << "'";
  33. Framework::RCArray<ValidationResult> invalidParts;
  34. JSONValue* valid
  35. = configValidator->getValidParts(zValue, &invalidParts);
  36. for (ValidationResult* invalidPart : invalidParts)
  37. {
  38. Framework::Logging::error() << invalidPart->getInvalidInfo();
  39. }
  40. if (valid)
  41. {
  42. for (JSONValue* config : *valid->asArray())
  43. {
  44. DimensionGenerator* generator
  45. = Game::INSTANCE->zTypeRegistry()
  46. ->fromJson<DimensionGenerator>(config);
  47. generator->initialize(seed);
  48. dimensionGenerators.add(generator);
  49. }
  50. valid->release();
  51. }
  52. });
  53. Framework::JSON::JSONObject* schema = configValidator->getJsonSchema();
  54. Framework::Datei syntaxFile;
  55. syntaxFile.setDatei("data/syntax/schema/generator.json");
  56. syntaxFile.erstellen();
  57. syntaxFile.open(Framework::Datei::Style::schreiben);
  58. syntaxFile.schreibe(schema->toString(), schema->toString().getLength());
  59. syntaxFile.close();
  60. schema->release();
  61. configValidator->release();
  62. start();
  63. }
  64. WorldGenerator::~WorldGenerator() {}
  65. Dimension* WorldGenerator::createDimension(int dimensionId)
  66. {
  67. for (DimensionGenerator* generator : dimensionGenerators)
  68. {
  69. if (generator->getDimensionId() == dimensionId)
  70. return generator->createDimension();
  71. }
  72. Framework::Logging::error()
  73. << "no dimension generator found for dimension " << dimensionId;
  74. return 0;
  75. }
  76. DimensionGenerator* WorldGenerator::zGenerator(int dimensionId)
  77. {
  78. for (DimensionGenerator* generator : dimensionGenerators)
  79. {
  80. if (generator->getDimensionId() == dimensionId) return generator;
  81. }
  82. return 0;
  83. }
  84. void WorldGenerator::thread()
  85. {
  86. double completeTime = 0.0;
  87. int generatedChunks = 0;
  88. while (!exit)
  89. {
  90. cs.lock();
  91. Area next;
  92. bool hasNext = 0;
  93. if (requestQueue.getEintragAnzahl() > 0)
  94. {
  95. next = requestQueue.get(0);
  96. requestQueue.remove(0);
  97. hasNext = 1;
  98. }
  99. cs.unlock();
  100. if (!hasNext)
  101. {
  102. chunksPerSecond = 0.0;
  103. completeTime = 0;
  104. generatedChunks = 0;
  105. Sleep(1000);
  106. continue;
  107. }
  108. // TODO: prevent that world loader and world generator create the
  109. // dimension at the same time
  110. Dimension* dim = Game::INSTANCE->zDimension(next.dimensionId);
  111. if (!dim)
  112. {
  113. dim = new Dimension(next.dimensionId);
  114. Game::INSTANCE->addDimension(dim);
  115. }
  116. Punkt start = Game::INSTANCE->getChunkCenter(next.startX, next.startY);
  117. Punkt end = Game::INSTANCE->getChunkCenter(next.endX, next.endY);
  118. int xDir = start.x > end.x ? -1 : 1;
  119. int yDir = start.y > end.y ? -1 : 1;
  120. for (int x = start.x; xDir < 0 ? x >= end.x : x <= end.x;
  121. x += CHUNK_SIZE * xDir)
  122. {
  123. for (int y = start.y; yDir < 0 ? y >= end.y : y <= end.y;
  124. y += CHUNK_SIZE * yDir)
  125. {
  126. if (!Game::INSTANCE->doesChunkExist(x, y, next.dimensionId))
  127. {
  128. ZeitMesser zm;
  129. zm.messungStart();
  130. Chunk* generatedChunk
  131. = zGenerator(next.dimensionId)->generateChunk(x, y);
  132. #ifdef CHUNK_GENERATION_DEBUG_LOG
  133. zm.messungEnde();
  134. Framework::Logging::trace()
  135. << "block generation time: " << zm.getSekunden();
  136. completeTime += zm.getSekunden();
  137. zm.messungStart();
  138. #endif
  139. generatedChunk->initializeLightning();
  140. #ifdef CHUNK_GENERATION_DEBUG_LOG
  141. zm.messungEnde();
  142. Framework::Logging::trace()
  143. << "light calculation: " << zm.getSekunden();
  144. completeTime += zm.getSekunden();
  145. zm.messungStart();
  146. #endif
  147. generatedChunk->removeUnusedBlocks();
  148. #ifdef CHUNK_GENERATION_DEBUG_LOG
  149. zm.messungEnde();
  150. Framework::Logging::trace()
  151. << "unused block removal: " << zm.getSekunden();
  152. completeTime += zm.getSekunden();
  153. zm.messungStart();
  154. #endif
  155. generatedChunk->getThis();
  156. dim->setChunk(generatedChunk, Punkt(x, y));
  157. zGenerator(next.dimensionId)
  158. ->generateEntities(generatedChunk);
  159. generatedChunk->release();
  160. zm.messungEnde();
  161. #ifdef CHUNK_GENERATION_DEBUG_LOG
  162. Framework::Logging::trace()
  163. << "adding chunk to map: " << zm.getSekunden();
  164. #endif
  165. completeTime += zm.getSekunden();
  166. generatedChunks++;
  167. chunksPerSecond = generatedChunks / completeTime;
  168. }
  169. }
  170. }
  171. }
  172. Framework::Logging::info() << "World Generator thread exited";
  173. }
  174. void WorldGenerator::requestGeneration(Area request)
  175. {
  176. cs.lock();
  177. requestQueue.add(request);
  178. cs.unlock();
  179. }
  180. void WorldGenerator::exitAndWait()
  181. {
  182. exit = 1;
  183. warteAufThread(10000);
  184. ende();
  185. }
  186. Framework::Either<Block*, int> WorldGenerator::generateSingleBlock(
  187. Framework::Vec3<int> location, int dimensionId)
  188. {
  189. return zGenerator(dimensionId)->generateBlock(location);
  190. }
  191. bool WorldGenerator::spawnStructure(Framework::Vec3<int> location,
  192. int dimensionId,
  193. std::function<bool(GeneratorTemplate* tmpl)> filter)
  194. {
  195. return zGenerator(dimensionId)->spawnStructure(location, filter);
  196. }
  197. double WorldGenerator::getGeneratedChunksPerSecond()
  198. {
  199. return chunksPerSecond;
  200. }