2011-12-26 18:23:05 -05:00
2012-01-29 14:28:19 -05:00
# include "Globals.h" // NOTE: MSVC stupidness requires this to be the same across all modules
2011-10-03 14:41:19 -04:00
# ifndef _WIN32
2012-02-01 07:46:44 -05:00
# include <cstdlib>
2011-10-03 14:41:19 -04:00
# endif
2011-12-22 16:36:24 -05:00
2011-10-03 14:41:19 -04:00
# include "cChunk.h"
# include "cWorld.h"
2011-12-22 16:36:24 -05:00
# include "cWaterSimulator.h"
# include "cLavaSimulator.h"
2011-10-03 14:41:19 -04:00
# include "cClientHandle.h"
# include "cServer.h"
# include "zlib.h"
# include "Defines.h"
# include "cChestEntity.h"
# include "cFurnaceEntity.h"
# include "cSignEntity.h"
# include "cTorch.h"
# include "cLadder.h"
# include "cPickup.h"
2011-11-07 17:59:29 -05:00
# include "cRedstone.h"
2011-10-03 14:41:19 -04:00
# include "cItem.h"
# include "cNoise.h"
# include "cRoot.h"
2011-12-25 21:39:43 -05:00
# include "cWorldGenerator.h"
2011-11-07 17:59:29 -05:00
# include "cBlockToPickup.h"
2011-12-26 04:09:47 -05:00
# include "MersenneTwister.h"
2011-10-03 14:41:19 -04:00
# include "packets/cPacket_DestroyEntity.h"
# include "packets/cPacket_PreChunk.h"
# include "packets/cPacket_BlockChange.h"
# include "packets/cPacket_MapChunk.h"
# include "packets/cPacket_MultiBlock.h"
# include <json/json.h>
2012-01-29 14:28:19 -05:00
2011-10-21 17:25:29 -04:00
2011-10-03 14:41:19 -04:00
extern bool g_bWaterPhysics ;
2012-01-01 11:20:52 -05:00
typedef std : : map < int , std : : string > ReferenceMap ;
2011-10-03 14:41:19 -04:00
typedef std : : list < cFurnaceEntity * > FurnaceEntityList ;
typedef std : : list < cClientHandle * > ClientHandleList ;
typedef std : : list < cBlockEntity * > BlockEntityList ;
typedef std : : list < cEntity * > EntityList ;
2012-01-29 14:28:19 -05:00
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
struct cChunk : : sChunkState
{
2012-01-01 11:20:52 -05:00
sChunkState ( )
: TotalReferencesEver ( 0 )
, MinusReferences ( 0 )
2012-01-19 13:12:39 -05:00
, NumRefs ( 0 )
2012-01-01 11:20:52 -05:00
{ }
2011-12-26 16:54:08 -05:00
FurnaceEntityList TickBlockEntities ;
2011-12-28 09:52:31 -05:00
std : : map < unsigned int , int > ToTickBlocks ; // Protected by BlockListCriticalSection
std : : vector < unsigned int > PendingSendBlocks ; // Protected by BlockListCriticalSection
2011-12-26 16:54:08 -05:00
ClientHandleList LoadedByClient ;
ClientHandleList UnloadQuery ;
2011-12-28 09:52:31 -05:00
BlockEntityList BlockEntities ; // Protected by BlockListCriticalSection
2011-12-26 16:54:08 -05:00
EntityList Entities ;
2011-12-28 09:52:31 -05:00
cCriticalSection BlockListCriticalSection ;
2012-01-01 11:20:52 -05:00
// Reference counting
cCriticalSection ReferenceCriticalSection ;
ReferenceMap References ;
int MinusReferences ; // References.size() - MinusReferences = Actual amount of references. This is due to removal of reference without an ID (don't know which to remove, so remove none)
int TotalReferencesEver ; // For creating a unique reference ID
2012-01-19 13:12:39 -05:00
int NumRefs ;
2011-10-03 14:41:19 -04:00
} ;
2012-01-30 11:01:45 -05:00
cChunk : : cChunk ( int a_X , int a_Y , int a_Z , cWorld * a_World )
: m_pState ( new sChunkState )
, m_bCalculateLighting ( false )
, m_bCalculateHeightmap ( false )
, m_PosX ( a_X )
, m_PosY ( a_Y )
, m_PosZ ( a_Z )
, m_BlockType ( m_BlockData ) // Offset the pointers
, m_BlockMeta ( m_BlockType + c_NumBlocks )
, m_BlockLight ( m_BlockMeta + c_NumBlocks / 2 )
, m_BlockSkyLight ( m_BlockLight + c_NumBlocks / 2 )
, m_BlockTickNum ( 0 )
, m_BlockTickX ( 0 )
, m_BlockTickY ( 0 )
, m_BlockTickZ ( 0 )
, m_EntitiesCriticalSection ( 0 )
, m_World ( a_World )
{
//LOG("cChunk::cChunk(%i, %i, %i)", a_X, a_Y, a_Z);
m_EntitiesCriticalSection = new cCriticalSection ( ) ;
}
2011-10-03 14:41:19 -04:00
cChunk : : ~ cChunk ( )
{
//LOG("~cChunk() %i %i %i", m_PosX, m_PosY, m_PosZ );
2011-12-26 16:54:08 -05:00
if ( ! m_pState - > LoadedByClient . empty ( ) )
2011-12-24 18:34:30 -05:00
{
2011-12-26 16:54:08 -05:00
LOGWARN ( " WARNING: Deleting cChunk while it contains %i clients! " , m_pState - > LoadedByClient . size ( ) ) ;
2011-12-24 18:34:30 -05:00
}
2012-01-01 11:20:52 -05:00
m_pState - > ReferenceCriticalSection . Lock ( ) ;
if ( GetReferenceCount ( ) > 0 )
{
LOGWARN ( " WARNING: Deleting cChunk while it still has %i references! " , GetReferenceCount ( ) ) ;
}
m_pState - > ReferenceCriticalSection . Unlock ( ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
for ( std : : list < cBlockEntity * > : : iterator itr = m_pState - > BlockEntities . begin ( ) ; itr ! = m_pState - > BlockEntities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
delete * itr ;
}
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . clear ( ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
LockEntities ( ) ;
2011-12-26 16:54:08 -05:00
if ( m_pState - > Entities . size ( ) > 0 )
2011-10-03 14:41:19 -04:00
{
2011-12-26 16:54:08 -05:00
EntityList Entities = m_pState - > Entities ; // Copy list to a temporary list
2011-11-02 16:19:57 -04:00
for ( EntityList : : iterator itr = Entities . begin ( ) ; itr ! = Entities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
2011-11-02 16:19:57 -04:00
if ( ( * itr ) - > GetEntityType ( ) ! = cEntity : : E_PLAYER )
{
( * itr ) - > RemoveFromChunk ( this ) ;
( * itr ) - > Destroy ( ) ;
}
2011-10-03 14:41:19 -04:00
}
2011-12-26 16:54:08 -05:00
m_pState - > Entities . clear ( ) ;
2011-10-03 14:41:19 -04:00
}
UnlockEntities ( ) ;
if ( m_EntitiesCriticalSection )
{
2011-10-21 17:25:29 -04:00
delete m_EntitiesCriticalSection ;
2011-10-03 14:41:19 -04:00
m_EntitiesCriticalSection = 0 ;
}
delete m_pState ;
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
void cChunk : : Initialize ( )
{
2012-01-30 11:01:45 -05:00
if ( ! LoadFromDisk ( ) )
2011-10-03 14:41:19 -04:00
{
// Clear memory
memset ( m_BlockData , 0x00 , c_BlockDataSize ) ;
2011-12-26 18:23:05 -05:00
m_World - > GetWorldGenerator ( ) - > GenerateChunk ( this ) ;
2011-10-03 14:41:19 -04:00
CalculateHeightmap ( ) ;
CalculateLighting ( ) ;
CreateBlockEntities ( ) ;
2011-12-24 18:34:30 -05:00
// During generation, some blocks might have been set by using (Fast)SetBlock() causing this list to fill.
// This chunk has not been sent to anybody yet, so there is no need for separately sending block changes when you can send an entire chunk
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > BlockListCriticalSection ) ;
2011-12-26 16:54:08 -05:00
m_pState - > PendingSendBlocks . clear ( ) ;
2011-10-03 14:41:19 -04:00
}
else
{
//LOGINFO("Successfully loaded from disk");
CalculateHeightmap ( ) ;
}
}
2012-01-30 11:01:45 -05:00
2012-02-08 07:36:54 -05:00
void cChunk : : Tick ( float a_Dt , MTRand & a_TickRandom )
2011-10-03 14:41:19 -04:00
{
2012-01-30 11:01:45 -05:00
if ( m_bCalculateLighting )
{
2011-10-03 14:41:19 -04:00
CalculateLighting ( ) ;
2012-01-30 11:01:45 -05:00
}
if ( m_bCalculateHeightmap )
{
2011-10-03 14:41:19 -04:00
CalculateHeightmap ( ) ;
2012-01-30 11:01:45 -05:00
}
2011-10-03 14:41:19 -04:00
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > BlockListCriticalSection ) ;
2011-12-26 16:54:08 -05:00
unsigned int PendingSendBlocks = m_pState - > PendingSendBlocks . size ( ) ;
2012-01-26 15:39:46 -05:00
if ( PendingSendBlocks > 1 )
{
cPacket_MultiBlock MultiBlock ;
MultiBlock . m_ChunkX = m_PosX ;
MultiBlock . m_ChunkZ = m_PosZ ;
MultiBlock . m_NumBlocks = ( short ) PendingSendBlocks ;
MultiBlock . m_BlockCoordinates = new unsigned short [ PendingSendBlocks ] ;
MultiBlock . m_BlockTypes = new char [ PendingSendBlocks ] ;
MultiBlock . m_BlockMetas = new char [ PendingSendBlocks ] ;
//LOG("Sending multiblock packet for %i blocks", PendingSendBlocks );
for ( unsigned int i = 0 ; i < PendingSendBlocks ; i + + )
{
unsigned int index = m_pState - > PendingSendBlocks [ i ] ;
unsigned int Y = index % 128 ;
unsigned int Z = ( index / 128 ) % 16 ;
unsigned int X = ( index / ( 128 * 16 ) ) ;
MultiBlock . m_BlockCoordinates [ i ] = ( Z & 0xf ) | ( X & 0xf ) < < 4 | ( Y & 0xff ) < < 8 ;
//LOG("X: %i Y: %i Z: %i Combo: 0x%04x", X, Y, Z, MultiBlock.m_BlockCoordinates[i] );
MultiBlock . m_BlockTypes [ i ] = m_BlockType [ index ] ;
MultiBlock . m_BlockMetas [ i ] = GetLight ( m_BlockMeta , index ) ;
}
m_pState - > PendingSendBlocks . clear ( ) ;
PendingSendBlocks = m_pState - > PendingSendBlocks . size ( ) ;
Broadcast ( MultiBlock ) ;
}
2011-10-03 14:41:19 -04:00
if ( PendingSendBlocks > 0 )
{
for ( unsigned int i = 0 ; i < PendingSendBlocks ; i + + )
{
2011-12-26 16:54:08 -05:00
unsigned int index = m_pState - > PendingSendBlocks [ i ] ;
2011-10-03 14:41:19 -04:00
int Y = index % 128 ;
int Z = ( index / 128 ) % 16 ;
int X = ( index / ( 128 * 16 ) ) ;
cPacket_BlockChange BlockChange ;
BlockChange . m_PosX = X + m_PosX * 16 ;
BlockChange . m_PosY = ( char ) ( Y + m_PosY * 128 ) ;
BlockChange . m_PosZ = Z + m_PosZ * 16 ;
BlockChange . m_BlockType = m_BlockType [ index ] ;
BlockChange . m_BlockMeta = GetLight ( m_BlockMeta , index ) ;
Broadcast ( BlockChange ) ;
}
2011-12-26 16:54:08 -05:00
m_pState - > PendingSendBlocks . clear ( ) ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 16:54:40 -05:00
Lock . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
2011-12-26 16:54:08 -05:00
while ( ! m_pState - > UnloadQuery . empty ( ) )
2011-10-03 14:41:19 -04:00
{
cPacket_PreChunk UnloadPacket ;
UnloadPacket . m_PosX = GetPosX ( ) ;
UnloadPacket . m_PosZ = GetPosZ ( ) ;
UnloadPacket . m_bLoad = false ; // Unload
2011-12-26 16:54:08 -05:00
( * m_pState - > UnloadQuery . begin ( ) ) - > Send ( UnloadPacket ) ;
m_pState - > UnloadQuery . remove ( * m_pState - > UnloadQuery . begin ( ) ) ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 16:54:40 -05:00
cCSLock Lock2 ( m_pState - > BlockListCriticalSection ) ;
2011-12-26 16:54:08 -05:00
std : : map < unsigned int , int > ToTickBlocks = m_pState - > ToTickBlocks ;
m_pState - > ToTickBlocks . clear ( ) ;
2012-01-30 16:54:40 -05:00
Lock2 . Unlock ( ) ;
2011-12-28 09:52:31 -05:00
2011-11-07 17:59:29 -05:00
bool isRedstone = false ;
2011-10-03 14:41:19 -04:00
for ( std : : map < unsigned int , int > : : iterator itr = ToTickBlocks . begin ( ) ; itr ! = ToTickBlocks . end ( ) ; + + itr )
{
if ( ( * itr ) . second < 0 ) continue ;
unsigned int index = ( * itr ) . first ;
int Y = index % 128 ;
int Z = ( index / 128 ) % 16 ;
int X = ( index / ( 128 * 16 ) ) ;
char BlockID = GetBlock ( index ) ;
switch ( BlockID )
{
2011-11-07 17:59:29 -05:00
case E_BLOCK_REDSTONE_REPEATER_OFF :
case E_BLOCK_REDSTONE_REPEATER_ON :
case E_BLOCK_REDSTONE_WIRE :
{
isRedstone = true ;
}
2012-01-23 10:11:45 -05:00
case E_BLOCK_CACTUS :
2011-10-03 14:41:19 -04:00
case E_BLOCK_REEDS :
case E_BLOCK_WOODEN_PRESSURE_PLATE :
case E_BLOCK_STONE_PRESSURE_PLATE :
case E_BLOCK_MINECART_TRACKS :
case E_BLOCK_SIGN_POST :
case E_BLOCK_CROPS :
case E_BLOCK_SAPLING :
case E_BLOCK_YELLOW_FLOWER :
case E_BLOCK_RED_ROSE :
case E_BLOCK_RED_MUSHROOM :
2011-11-07 17:59:29 -05:00
case E_BLOCK_BROWN_MUSHROOM : // Stuff that drops when block below is destroyed
2011-10-03 14:41:19 -04:00
{
if ( GetBlock ( X , Y - 1 , Z ) = = E_BLOCK_AIR )
{
2011-12-22 16:36:24 -05:00
SetBlock ( X , Y , Z , E_BLOCK_AIR , 0 ) ;
int wX , wY , wZ ;
PositionToWorldPosition ( X , Y , Z , wX , wY , wZ ) ;
2011-12-26 15:57:12 -05:00
m_World - > GetSimulatorManager ( ) - > WakeUp ( wX , wY , wZ ) ;
2011-11-07 17:59:29 -05:00
if ( isRedstone ) {
cRedstone Redstone ( m_World ) ;
2011-11-07 18:09:03 -05:00
Redstone . ChangeRedstone ( ( X + m_PosX * 16 ) , ( Y + m_PosY * 16 ) , ( Z + m_PosZ * 16 ) , false ) ;
2011-11-07 17:59:29 -05:00
}
cPickup * Pickup = new cPickup ( ( X + m_PosX * 16 ) * 32 + 16 , ( Y + m_PosY * 128 ) * 32 + 16 , ( Z + m_PosZ * 16 ) * 32 + 16 , cItem ( cBlockToPickup : : ToPickup ( ( ENUM_ITEM_ID ) BlockID , E_ITEM_EMPTY ) , 1 ) ) ;
2011-10-30 20:52:20 -04:00
Pickup - > Initialize ( m_World ) ;
2011-10-03 14:41:19 -04:00
}
}
break ;
case E_BLOCK_REDSTONE_TORCH_OFF :
case E_BLOCK_REDSTONE_TORCH_ON :
2011-11-07 17:59:29 -05:00
isRedstone = true ;
2011-10-03 14:41:19 -04:00
case E_BLOCK_TORCH :
{
char Dir = cTorch : : MetaDataToDirection ( GetLight ( m_BlockMeta , X , Y , Z ) ) ;
LOG ( " MetaData: %i " , Dir ) ;
int XX = X + m_PosX * 16 ;
char YY = ( char ) Y ;
int ZZ = Z + m_PosZ * 16 ;
AddDirection ( XX , YY , ZZ , Dir , true ) ;
2011-10-30 20:52:20 -04:00
if ( m_World - > GetBlock ( XX , YY , ZZ ) = = E_BLOCK_AIR )
2011-10-03 14:41:19 -04:00
{
SetBlock ( X , Y , Z , 0 , 0 ) ;
2011-11-07 17:59:29 -05:00
if ( isRedstone ) {
cRedstone Redstone ( m_World ) ;
2011-11-07 18:09:03 -05:00
Redstone . ChangeRedstone ( ( X + m_PosX * 16 ) , ( Y + m_PosY * 16 ) , ( Z + m_PosZ * 16 ) , false ) ;
2011-11-07 17:59:29 -05:00
}
cPickup * Pickup = new cPickup ( ( X + m_PosX * 16 ) * 32 + 16 , ( Y + m_PosY * 128 ) * 32 + 16 , ( Z + m_PosZ * 16 ) * 32 + 16 , cItem ( cBlockToPickup : : ToPickup ( ( ENUM_ITEM_ID ) BlockID , E_ITEM_EMPTY ) , 1 ) ) ;
2011-10-30 20:52:20 -04:00
Pickup - > Initialize ( m_World ) ;
2011-10-03 14:41:19 -04:00
}
}
break ;
case E_BLOCK_LADDER :
{
char Dir = cLadder : : MetaDataToDirection ( GetLight ( m_BlockMeta , X , Y , Z ) ) ;
int XX = X + m_PosX * 16 ;
char YY = ( char ) Y ;
int ZZ = Z + m_PosZ * 16 ;
AddDirection ( XX , YY , ZZ , Dir , true ) ;
2011-10-30 20:52:20 -04:00
if ( m_World - > GetBlock ( XX , YY , ZZ ) = = E_BLOCK_AIR )
2011-10-03 14:41:19 -04:00
{
2011-12-22 16:36:24 -05:00
SetBlock ( X , Y , Z , E_BLOCK_AIR , 0 ) ;
2011-10-03 14:41:19 -04:00
cPickup * Pickup = new cPickup ( ( X + m_PosX * 16 ) * 32 + 16 , ( Y + m_PosY * 128 ) * 32 + 16 , ( Z + m_PosZ * 16 ) * 32 + 16 , cItem ( ( ENUM_ITEM_ID ) BlockID , 1 ) ) ;
2011-10-30 20:52:20 -04:00
Pickup - > Initialize ( m_World ) ;
2011-10-03 14:41:19 -04:00
}
}
break ;
default :
break ;
} ;
}
2011-12-28 09:52:31 -05:00
2011-10-03 14:41:19 -04:00
// Tick dem blocks
2012-02-08 07:36:54 -05:00
int RandomX = a_TickRandom . randInt ( ) ;
int RandomY = a_TickRandom . randInt ( ) ;
int RandomZ = a_TickRandom . randInt ( ) ;
2011-10-03 14:41:19 -04:00
for ( int i = 0 ; i < 50 ; i + + )
{
m_BlockTickX = ( m_BlockTickX + RandomX ) % 16 ;
m_BlockTickY = ( m_BlockTickY + RandomY ) % 128 ;
m_BlockTickZ = ( m_BlockTickZ + RandomZ ) % 16 ;
//LOG("%03i %03i %03i", m_BlockTickX, m_BlockTickY, m_BlockTickZ);
if ( m_BlockTickY > m_HeightMap [ m_BlockTickX + m_BlockTickZ * 16 ] ) continue ; // It's all air up here
//m_BlockTickNum = (m_BlockTickNum + 1 ) % c_NumBlocks;
unsigned int Index = MakeIndex ( m_BlockTickX , m_BlockTickY , m_BlockTickZ ) ;
char ID = m_BlockType [ Index ] ;
switch ( ID )
{
case E_BLOCK_DIRT :
{
char AboveBlock = GetBlock ( Index + 1 ) ;
2011-11-05 15:31:25 -04:00
if ( ( AboveBlock = = 0 ) & & GetLight ( m_BlockSkyLight , Index ) > 0xf / 2 ) // Half lit //changed to not allow grass if any one hit object is on top
2011-10-03 14:41:19 -04:00
{
FastSetBlock ( m_BlockTickX , m_BlockTickY , m_BlockTickZ , E_BLOCK_GRASS , GetLight ( m_BlockMeta , Index ) ) ;
}
2011-11-05 15:31:25 -04:00
if ( ( g_BlockOneHitDig [ AboveBlock ] ) & & GetLight ( m_BlockSkyLight , Index + 1 ) > 0xf / 2 ) // Half lit //ch$
2011-11-08 20:31:19 -05:00
{
FastSetBlock ( m_BlockTickX , m_BlockTickY , m_BlockTickZ , E_BLOCK_GRASS , GetLight ( m_BlockMeta , Index ) ) ;
}
2011-11-05 15:31:25 -04:00
2011-10-03 14:41:19 -04:00
}
break ;
case E_BLOCK_GRASS :
{
char AboveBlock = GetBlock ( Index + 1 ) ;
2011-11-06 04:23:20 -05:00
if ( ! ( ( AboveBlock = = 0 ) | | ( g_BlockOneHitDig [ AboveBlock ] ) | | ( g_BlockTransparent [ AboveBlock ] ) ) ) //changed to not allow grass if any one hit object is on top
2011-10-03 14:41:19 -04:00
{
FastSetBlock ( m_BlockTickX , m_BlockTickY , m_BlockTickZ , E_BLOCK_DIRT , GetLight ( m_BlockMeta , Index ) ) ;
}
}
break ;
2011-11-05 15:31:25 -04:00
case E_BLOCK_SAPLING : //todo: check meta of sapling. change m_World->GrowTree to look change trunk and leaves based on meta of sapling
2011-10-03 14:41:19 -04:00
{
FastSetBlock ( m_BlockTickX , m_BlockTickY , m_BlockTickZ , E_BLOCK_AIR , GetLight ( m_BlockMeta , Index ) ) ;
2011-10-30 20:52:20 -04:00
m_World - > GrowTree ( m_BlockTickX + m_PosX * 16 , m_BlockTickY , m_BlockTickZ + m_PosZ * 16 ) ;
2011-10-03 14:41:19 -04:00
}
2011-11-09 22:35:46 -05:00
break ;
case E_BLOCK_LEAVES : //todo, http://www.minecraftwiki.net/wiki/Data_values#Leaves
{
}
break ;
2011-10-03 14:41:19 -04:00
default :
break ;
}
}
// Tick block entities (furnace)
2011-12-26 16:54:08 -05:00
std : : list < cFurnaceEntity * > TickBlockEntites = m_pState - > TickBlockEntities ; // Dangerous stuff, better make a copy.
2011-10-03 14:41:19 -04:00
for ( std : : list < cFurnaceEntity * > : : iterator itr = TickBlockEntites . begin ( ) ; itr ! = TickBlockEntites . end ( ) ; + + itr )
{
if ( ! ( * itr ) - > Tick ( a_Dt ) ) // Remove from list
{
2011-12-26 16:54:08 -05:00
m_pState - > TickBlockEntities . remove ( * itr ) ;
2011-10-03 14:41:19 -04:00
}
}
}
2012-02-08 07:36:54 -05:00
2011-10-03 14:41:19 -04:00
char cChunk : : GetHeight ( int a_X , int a_Z )
{
if ( a_X > = 0 & & a_X < 16 & & a_Z > = 0 & & a_Z < 16 )
return m_HeightMap [ a_X + a_Z * 16 ] ;
return 0 ;
}
void cChunk : : CreateBlockEntities ( )
{
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-10-03 14:41:19 -04:00
for ( int x = 0 ; x < 16 ; x + + )
{
for ( int z = 0 ; z < 16 ; z + + )
{
for ( int y = 0 ; y < 128 ; y + + )
{
ENUM_BLOCK_ID BlockType = ( ENUM_BLOCK_ID ) m_BlockData [ MakeIndex ( x , y , z ) ] ;
switch ( BlockType )
{
case E_BLOCK_CHEST :
{
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . push_back ( new cChestEntity ( x + m_PosX * 16 , y + m_PosY * 128 , z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
}
break ;
case E_BLOCK_FURNACE :
{
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . push_back ( new cFurnaceEntity ( x + m_PosX * 16 , y + m_PosY * 128 , z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
}
break ;
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
{
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . push_back ( new cSignEntity ( BlockType , x + m_PosX * 16 , y + m_PosY * 128 , z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
}
break ;
default :
{
}
break ;
}
}
}
}
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : CalculateHeightmap ( )
{
m_bCalculateHeightmap = false ;
for ( int x = 0 ; x < 16 ; x + + )
{
for ( int z = 0 ; z < 16 ; z + + )
{
for ( int y = 127 ; y > - 1 ; y - - )
{
int index = MakeIndex ( x , y , z ) ;
if ( m_BlockData [ index ] ! = E_BLOCK_AIR )
{
m_HeightMap [ x + z * 16 ] = ( char ) y ;
break ;
}
}
}
}
}
void cChunk : : CalculateLighting ( )
{
// Calculate sunlight
memset ( m_BlockSkyLight , 0xff , c_NumBlocks / 2 ) ; // Set all to fully lit, so everything above HeightMap is lit
for ( int x = 0 ; x < 16 ; x + + )
{
for ( int z = 0 ; z < 16 ; z + + )
{
char sunlight = 0xf ;
for ( int y = m_HeightMap [ x + z * 16 ] ; y > - 1 ; y - - )
{
int index = y + ( z * 128 ) + ( x * 128 * 16 ) ;
if ( g_BlockTransparent [ ( int ) m_BlockData [ index ] ] = = false )
{
sunlight = 0x0 ;
}
SetLight ( m_BlockSkyLight , x , y , z , sunlight ) ;
}
}
}
// Calculate blocklights
for ( int x = 0 ; x < 16 ; x + + )
{
for ( int z = 0 ; z < 16 ; z + + )
{
int MaxHeight = m_HeightMap [ x + z * 16 ] ;
for ( int y = 0 ; y < MaxHeight ; y + + )
{
char BlockID = GetBlock ( x , y , z ) ;
SetLight ( m_BlockLight , x , y , z , g_BlockLightValue [ ( int ) BlockID ] ) ;
}
}
}
SpreadLight ( m_BlockSkyLight ) ;
SpreadLight ( m_BlockLight ) ;
// Stop it from calculating again :P
m_bCalculateLighting = false ;
}
void cChunk : : SpreadLight ( char * a_LightBuffer )
{
// Spread the sunlight
for ( int x = 0 ; x < 16 ; x + + ) for ( int z = 0 ; z < 16 ; z + + ) for ( int y = 0 ; y < 128 ; y + + )
{
int index = y + ( z * 128 ) + ( x * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
SpreadLightOfBlock ( a_LightBuffer , x , y , z , g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ;
}
}
for ( int x = 15 ; x > - 1 ; x - - ) for ( int z = 15 ; z > - 1 ; z - - ) for ( int y = 127 ; y > - 1 ; y - - )
{
int index = y + ( z * 128 ) + ( x * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
SpreadLightOfBlock ( a_LightBuffer , x , y , z , g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ;
}
}
bool bCalcLeft , bCalcRight , bCalcFront , bCalcBack ;
bCalcLeft = bCalcRight = bCalcFront = bCalcBack = false ;
// Spread to neighbour chunks X-axis
2012-01-19 13:12:39 -05:00
ptr_cChunk LeftChunk = m_World - > GetChunkUnreliable ( m_PosX - 1 , m_PosY , m_PosZ ) ;
ptr_cChunk RightChunk = m_World - > GetChunkUnreliable ( m_PosX + 1 , m_PosY , m_PosZ ) ;
2011-10-03 14:41:19 -04:00
char * LeftSky = 0 , * RightSky = 0 ;
if ( LeftChunk ) LeftSky = ( a_LightBuffer = = m_BlockSkyLight ) ? LeftChunk - > pGetSkyLight ( ) : LeftChunk - > pGetLight ( ) ;
if ( RightChunk ) RightSky = ( a_LightBuffer = = m_BlockSkyLight ) ? RightChunk - > pGetSkyLight ( ) : RightChunk - > pGetLight ( ) ;
for ( int z = 0 ; z < 16 ; z + + ) for ( int y = 0 ; y < 128 ; y + + )
{
if ( LeftChunk )
{
int index = y + ( z * 128 ) + ( 0 * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
char CurrentLight = GetLight ( a_LightBuffer , 0 , y , z ) ;
char LeftLight = GetLight ( LeftSky , 15 , y , z ) ;
if ( LeftLight < CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] )
{
SetLight ( LeftSky , 15 , y , z , MAX ( 0 , CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ) ;
bCalcLeft = true ;
}
}
}
if ( RightChunk )
{
int index = y + ( z * 128 ) + ( 15 * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
char CurrentLight = GetLight ( a_LightBuffer , 15 , y , z ) ;
char RightLight = GetLight ( RightSky , 0 , y , z ) ;
if ( RightLight < CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] )
{
SetLight ( RightSky , 0 , y , z , MAX ( 0 , CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ) ;
bCalcRight = true ;
}
}
}
}
// Spread to neighbour chunks Z-axis
2012-01-19 13:12:39 -05:00
ptr_cChunk FrontChunk = m_World - > GetChunkUnreliable ( m_PosX , m_PosY , m_PosZ - 1 ) ;
ptr_cChunk BackChunk = m_World - > GetChunkUnreliable ( m_PosX , m_PosY , m_PosZ + 1 ) ;
2011-10-03 14:41:19 -04:00
char * FrontSky = 0 , * BackSky = 0 ;
if ( FrontChunk ) FrontSky = ( a_LightBuffer = = m_BlockSkyLight ) ? FrontChunk - > pGetSkyLight ( ) : FrontChunk - > pGetLight ( ) ;
if ( BackChunk ) BackSky = ( a_LightBuffer = = m_BlockSkyLight ) ? BackChunk - > pGetSkyLight ( ) : BackChunk - > pGetLight ( ) ;
for ( int x = 0 ; x < 16 ; x + + ) for ( int y = 0 ; y < 128 ; y + + )
{
if ( FrontChunk )
{
int index = y + ( 0 * 128 ) + ( x * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
char CurrentLight = GetLight ( a_LightBuffer , x , y , 0 ) ;
char FrontLight = GetLight ( FrontSky , x , y , 15 ) ;
if ( FrontLight < CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] )
{
SetLight ( FrontSky , x , y , 15 , MAX ( 0 , CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ) ;
bCalcFront = true ;
}
}
}
if ( BackChunk )
{
int index = y + ( 15 * 128 ) + ( x * 128 * 16 ) ;
if ( g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] > 0 )
{
char CurrentLight = GetLight ( a_LightBuffer , x , y , 15 ) ;
char BackLight = GetLight ( BackSky , x , y , 0 ) ;
if ( BackLight < CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] )
{
SetLight ( BackSky , x , y , 0 , MAX ( 0 , CurrentLight - g_BlockSpreadLightFalloff [ m_BlockData [ index ] ] ) ) ;
bCalcBack = true ;
}
}
}
}
2011-10-30 20:52:20 -04:00
if ( bCalcLeft ) m_World - > ReSpreadLighting ( LeftChunk ) ;
if ( bCalcRight ) m_World - > ReSpreadLighting ( RightChunk ) ;
if ( bCalcFront ) m_World - > ReSpreadLighting ( FrontChunk ) ;
if ( bCalcBack ) m_World - > ReSpreadLighting ( BackChunk ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : AsyncUnload ( cClientHandle * a_Client )
{
2011-12-26 16:54:08 -05:00
m_pState - > UnloadQuery . remove ( a_Client ) ; // Make sure this client is only in the list once
m_pState - > UnloadQuery . push_back ( a_Client ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : Send ( cClientHandle * a_Client )
{
cPacket_PreChunk PreChunk ;
PreChunk . m_PosX = m_PosX ;
PreChunk . m_PosZ = m_PosZ ;
PreChunk . m_bLoad = true ;
a_Client - > Send ( PreChunk ) ;
a_Client - > Send ( cPacket_MapChunk ( this ) ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
for ( BlockEntityList : : iterator itr = m_pState - > BlockEntities . begin ( ) ; itr ! = m_pState - > BlockEntities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
( * itr ) - > SendTo ( a_Client ) ;
}
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : SetBlock ( int a_X , int a_Y , int a_Z , char a_BlockType , char a_BlockMeta )
{
if ( a_X < 0 | | a_X > = 16 | | a_Y < 0 | | a_Y > = 128 | | a_Z < 0 | | a_Z > = 16 )
{
//printf(">>>>>>>>>>>>>>>> CLIPPED SETBLOCK %i %i %i\n", a_X, a_Y, a_Z );
return ; // Clip
}
int index = a_Y + ( a_Z * 128 ) + ( a_X * 128 * 16 ) ;
char OldBlockMeta = GetLight ( m_BlockMeta , index ) ;
char OldBlockType = m_BlockType [ index ] ;
m_BlockType [ index ] = a_BlockType ;
SetLight ( m_BlockMeta , index , a_BlockMeta ) ;
if ( OldBlockType ! = a_BlockType | | OldBlockMeta ! = a_BlockMeta )
{
//LOG("Old: %i %i New: %i %i", OldBlockType, OldBlockMeta, a_BlockType, a_BlockMeta );
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > PendingSendBlocks . push_back ( index ) ;
2011-10-03 14:41:19 -04:00
2011-12-26 16:54:08 -05:00
m_pState - > ToTickBlocks [ MakeIndex ( a_X , a_Y , a_Z ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X + 1 , a_Y , a_Z ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X - 1 , a_Y , a_Z ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X , a_Y + 1 , a_Z ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X , a_Y - 1 , a_Z ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X , a_Y , a_Z + 1 ) ] + + ;
m_pState - > ToTickBlocks [ MakeIndex ( a_X , a_Y , a_Z - 1 ) ] + + ;
2011-10-03 14:41:19 -04:00
cBlockEntity * BlockEntity = GetBlockEntity ( a_X + m_PosX * 16 , a_Y + m_PosY * 128 , a_Z + m_PosZ * 16 ) ;
if ( BlockEntity )
{
BlockEntity - > Destroy ( ) ;
RemoveBlockEntity ( BlockEntity ) ;
delete BlockEntity ;
}
switch ( a_BlockType )
{
case E_BLOCK_CHEST :
2011-10-30 20:52:20 -04:00
AddBlockEntity ( new cChestEntity ( a_X + m_PosX * 16 , a_Y + m_PosY * 128 , a_Z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
break ;
case E_BLOCK_FURNACE :
2011-10-30 20:52:20 -04:00
AddBlockEntity ( new cFurnaceEntity ( a_X + m_PosX * 16 , a_Y + m_PosY * 128 , a_Z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
break ;
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
2011-10-30 20:52:20 -04:00
AddBlockEntity ( new cSignEntity ( ( ENUM_BLOCK_ID ) a_BlockType , a_X + m_PosX * 16 , a_Y + m_PosY * 128 , a_Z + m_PosZ * 16 , this ) ) ;
2011-10-03 14:41:19 -04:00
break ;
default :
break ;
} ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
2011-12-31 16:08:23 -05:00
//RecalculateHeightmap();
//RecalculateLighting();
2011-10-03 14:41:19 -04:00
}
void cChunk : : FastSetBlock ( int a_X , int a_Y , int a_Z , char a_BlockType , char a_BlockMeta )
{
if ( a_X < 0 | | a_X > = 16 | | a_Y < 0 | | a_Y > = 128 | | a_Z < 0 | | a_Z > = 16 )
{
//printf(">>>>>>>>>>>>>>>> CLIPPED SETBLOCK %i %i %i\n", a_X, a_Y, a_Z );
return ; // Clip
}
const int index = a_Y + ( a_Z * 128 ) + ( a_X * 128 * 16 ) ;
const char OldBlock = m_BlockType [ index ] ;
m_BlockType [ index ] = a_BlockType ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > PendingSendBlocks . push_back ( index ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
SetLight ( m_BlockMeta , index , a_BlockMeta ) ;
2011-12-24 18:34:30 -05:00
// ONLY recalculate lighting if it's necessary!
2011-10-03 14:41:19 -04:00
if ( g_BlockLightValue [ OldBlock ] ! = g_BlockLightValue [ a_BlockType ]
| | g_BlockSpreadLightFalloff [ OldBlock ] ! = g_BlockSpreadLightFalloff [ a_BlockType ]
| | g_BlockTransparent [ OldBlock ] ! = g_BlockTransparent [ a_BlockType ] )
{
RecalculateLighting ( ) ;
}
// Recalculate next tick
RecalculateHeightmap ( ) ;
}
void cChunk : : SendBlockTo ( int a_X , int a_Y , int a_Z , cClientHandle * a_Client )
{
if ( a_Client = = 0 )
{
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > PendingSendBlocks . push_back ( MakeIndex ( a_X , a_Y , a_Z ) ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
return ;
}
2011-12-26 16:54:08 -05:00
for ( std : : list < cClientHandle * > : : iterator itr = m_pState - > LoadedByClient . begin ( ) ; itr ! = m_pState - > LoadedByClient . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
if ( * itr = = a_Client )
{
unsigned int index = MakeIndex ( a_X , a_Y , a_Z ) ;
cPacket_BlockChange BlockChange ;
BlockChange . m_PosX = a_X + m_PosX * 16 ;
BlockChange . m_PosY = ( char ) ( a_Y + m_PosY * 128 ) ;
BlockChange . m_PosZ = a_Z + m_PosZ * 16 ;
BlockChange . m_BlockType = m_BlockType [ index ] ;
BlockChange . m_BlockMeta = GetLight ( m_BlockMeta , index ) ;
a_Client - > Send ( BlockChange ) ;
break ;
}
}
}
void cChunk : : AddBlockEntity ( cBlockEntity * a_BlockEntity )
{
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . push_back ( a_BlockEntity ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : RemoveBlockEntity ( cBlockEntity * a_BlockEntity )
{
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > BlockEntities . remove ( a_BlockEntity ) ;
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : AddClient ( cClientHandle * a_Client )
{
2011-12-26 16:54:08 -05:00
m_pState - > LoadedByClient . remove ( a_Client ) ;
m_pState - > LoadedByClient . push_back ( a_Client ) ;
2011-10-03 14:41:19 -04:00
LockEntities ( ) ;
2011-12-26 16:54:08 -05:00
for ( EntityList : : iterator itr = m_pState - > Entities . begin ( ) ; itr ! = m_pState - > Entities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
2012-02-01 17:38:03 -05:00
LOG ( " %i %i %i Spawning on %s " , m_PosX , m_PosY , m_PosZ , a_Client - > GetUsername ( ) . c_str ( ) ) ;
2011-10-03 14:41:19 -04:00
( * itr ) - > SpawnOn ( a_Client ) ;
}
UnlockEntities ( ) ;
}
void cChunk : : RemoveClient ( cClientHandle * a_Client )
{
2011-12-26 16:54:08 -05:00
m_pState - > LoadedByClient . remove ( a_Client ) ;
2011-10-03 14:41:19 -04:00
2011-11-02 16:19:57 -04:00
if ( ! a_Client - > IsDestroyed ( ) )
2011-10-03 14:41:19 -04:00
{
2011-11-02 16:19:57 -04:00
LockEntities ( ) ;
2011-12-26 16:54:08 -05:00
for ( EntityList : : iterator itr = m_pState - > Entities . begin ( ) ; itr ! = m_pState - > Entities . end ( ) ; + + itr )
2011-11-02 16:19:57 -04:00
{
2012-02-01 17:38:03 -05:00
LOG ( " %i %i %i Destroying on %s " , m_PosX , m_PosY , m_PosZ , a_Client - > GetUsername ( ) . c_str ( ) ) ;
2011-11-02 16:19:57 -04:00
cPacket_DestroyEntity DestroyEntity ( * itr ) ;
a_Client - > Send ( DestroyEntity ) ;
}
UnlockEntities ( ) ;
2011-10-03 14:41:19 -04:00
}
}
void cChunk : : AddEntity ( cEntity & a_Entity )
{
LockEntities ( ) ;
2011-12-26 16:54:08 -05:00
m_pState - > Entities . push_back ( & a_Entity ) ;
2011-10-03 14:41:19 -04:00
UnlockEntities ( ) ;
}
bool cChunk : : RemoveEntity ( cEntity & a_Entity , cChunk * a_CalledFrom /* = 0 */ )
{
LockEntities ( ) ;
2011-12-26 16:54:08 -05:00
unsigned int SizeBefore = m_pState - > Entities . size ( ) ;
m_pState - > Entities . remove ( & a_Entity ) ;
if ( SizeBefore = = m_pState - > Entities . size ( ) )
2011-10-03 14:41:19 -04:00
{
LOG ( " WARNING: Entity was not in chunk %i %i %i " , m_PosX , m_PosY , m_PosZ ) ;
if ( ! a_CalledFrom )
{
UnlockEntities ( ) ;
2011-10-30 20:52:20 -04:00
return m_World - > RemoveEntityFromChunk ( a_Entity , this ) ;
2011-10-03 14:41:19 -04:00
}
UnlockEntities ( ) ;
return false ;
}
UnlockEntities ( ) ;
return true ;
}
void cChunk : : LockEntities ( )
{
2011-10-21 17:25:29 -04:00
m_EntitiesCriticalSection - > Lock ( ) ;
2011-10-03 14:41:19 -04:00
}
void cChunk : : UnlockEntities ( )
{
2011-10-21 17:25:29 -04:00
m_EntitiesCriticalSection - > Unlock ( ) ;
2011-10-03 14:41:19 -04:00
}
char cChunk : : GetBlock ( int a_X , int a_Y , int a_Z )
{
if ( a_X < 0 | | a_X > = 16 | | a_Y < 0 | | a_Y > = 128 | | a_Z < 0 | | a_Z > = 16 ) return 0 ; // Clip
int index = a_Y + ( a_Z * 128 ) + ( a_X * 128 * 16 ) ;
return m_BlockType [ index ] ;
}
char cChunk : : GetBlock ( int a_BlockIdx )
{
if ( a_BlockIdx < 0 | | a_BlockIdx > = c_NumBlocks ) return 0 ;
return m_BlockType [ a_BlockIdx ] ;
}
cBlockEntity * cChunk : : GetBlockEntity ( int a_X , int a_Y , int a_Z )
{
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Lock ( ) ;
2011-12-26 16:54:08 -05:00
for ( std : : list < cBlockEntity * > : : iterator itr = m_pState - > BlockEntities . begin ( ) ; itr ! = m_pState - > BlockEntities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
if ( ( * itr ) - > GetPosX ( ) = = a_X & &
( * itr ) - > GetPosY ( ) = = a_Y & &
( * itr ) - > GetPosZ ( ) = = a_Z )
{
2011-12-28 09:52:31 -05:00
cBlockEntity * BlockEnt = * itr ;
m_pState - > BlockListCriticalSection . Unlock ( ) ;
return BlockEnt ;
2011-10-03 14:41:19 -04:00
}
}
2011-12-28 09:52:31 -05:00
m_pState - > BlockListCriticalSection . Unlock ( ) ;
2011-10-03 14:41:19 -04:00
return 0 ;
}
2012-01-30 11:01:45 -05:00
/// Loads the chunk from the old-format disk file, erases the file afterwards. Returns true if successful
2011-10-03 14:41:19 -04:00
bool cChunk : : LoadFromDisk ( )
{
2012-02-01 08:43:47 -05:00
AString SourceFile ;
Printf ( SourceFile , " world/X%i_Y%i_Z%i.bin " , m_PosX , m_PosY , m_PosZ ) ;
2012-01-30 11:01:45 -05:00
cFile f ;
if ( ! f . Open ( SourceFile , cFile : : fmRead ) )
2011-10-03 14:41:19 -04:00
{
2012-01-30 11:01:45 -05:00
return false ;
}
2011-10-03 14:41:19 -04:00
2012-01-30 11:01:45 -05:00
if ( f . Read ( m_BlockData , sizeof ( m_BlockData ) ) ! = sizeof ( m_BlockData ) )
{
2012-02-01 08:50:09 -05:00
LOGERROR ( " ERROR READING FROM FILE %s " , SourceFile . c_str ( ) ) ;
2012-01-30 11:01:45 -05:00
return false ;
}
// Now load Block Entities
cCSLock Lock ( m_pState - > BlockListCriticalSection ) ;
ENUM_BLOCK_ID BlockType ;
while ( f . Read ( & BlockType , sizeof ( ENUM_BLOCK_ID ) ) = = sizeof ( ENUM_BLOCK_ID ) )
{
switch ( BlockType )
2011-10-03 14:41:19 -04:00
{
case E_BLOCK_CHEST :
2012-01-30 11:01:45 -05:00
{
cChestEntity * ChestEntity = new cChestEntity ( 0 , 0 , 0 , this ) ;
if ( ! ChestEntity - > LoadFromFile ( f ) )
2011-10-03 14:41:19 -04:00
{
2012-02-01 08:50:09 -05:00
LOGERROR ( " ERROR READING CHEST FROM FILE %s " , SourceFile . c_str ( ) ) ;
2012-01-30 11:01:45 -05:00
delete ChestEntity ;
return false ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
m_pState - > BlockEntities . push_back ( ChestEntity ) ;
2011-10-03 14:41:19 -04:00
break ;
2012-01-30 11:01:45 -05:00
}
2011-10-03 14:41:19 -04:00
case E_BLOCK_FURNACE :
2012-01-30 11:01:45 -05:00
{
cFurnaceEntity * FurnaceEntity = new cFurnaceEntity ( 0 , 0 , 0 , this ) ;
if ( ! FurnaceEntity - > LoadFromFile ( f ) )
2011-10-03 14:41:19 -04:00
{
2012-02-01 08:50:09 -05:00
LOGERROR ( " ERROR READING FURNACE FROM FILE %s " , SourceFile . c_str ( ) ) ;
2012-01-30 11:01:45 -05:00
delete FurnaceEntity ;
return false ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
m_pState - > BlockEntities . push_back ( FurnaceEntity ) ;
m_pState - > TickBlockEntities . push_back ( FurnaceEntity ) ; // They need tickin'
2011-10-03 14:41:19 -04:00
break ;
2012-01-30 11:01:45 -05:00
}
2011-10-03 14:41:19 -04:00
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
2012-01-30 11:01:45 -05:00
{
cSignEntity * SignEntity = new cSignEntity ( BlockType , 0 , 0 , 0 , this ) ;
if ( ! SignEntity - > LoadFromFile ( f ) )
2011-10-03 14:41:19 -04:00
{
2012-02-01 08:50:09 -05:00
LOGERROR ( " ERROR READING SIGN FROM FILE %s " , SourceFile . c_str ( ) ) ;
2012-01-30 11:01:45 -05:00
delete SignEntity ;
return false ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
m_pState - > BlockEntities . push_back ( SignEntity ) ;
2011-10-03 14:41:19 -04:00
break ;
2012-01-30 11:01:45 -05:00
}
2011-10-03 14:41:19 -04:00
default :
2012-01-30 11:01:45 -05:00
{
assert ( ! " Unhandled block entity in file " ) ;
2011-10-03 14:41:19 -04:00
break ;
}
}
2012-01-30 11:01:45 -05:00
}
Lock . Unlock ( ) ;
f . Close ( ) ;
2011-10-03 14:41:19 -04:00
2012-01-30 11:01:45 -05:00
// Delete old format file
2012-02-01 08:43:47 -05:00
if ( std : : remove ( SourceFile . c_str ( ) ) ! = 0 )
2012-01-30 11:01:45 -05:00
{
2012-02-01 08:50:09 -05:00
LOGERROR ( " Could not delete file %s " , SourceFile . c_str ( ) ) ;
2011-10-03 14:41:19 -04:00
}
else
{
2012-02-01 08:50:09 -05:00
LOGINFO ( " Successfully deleted old format file \" %s \" " , SourceFile . c_str ( ) ) ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
return true ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
bool cChunk : : SaveToDisk ( )
{
2012-01-30 11:01:45 -05:00
assert ( ! " Old save format not supported anymore " ) ; // Remove the call to this function
2012-02-01 08:50:09 -05:00
return false ; //no more saving old format!
2012-01-30 11:01:45 -05:00
}
2011-10-03 14:41:19 -04:00
void cChunk : : Broadcast ( const cPacket & a_Packet , cClientHandle * a_Exclude /* = 0 */ ) const
{
2011-12-26 16:54:08 -05:00
for ( std : : list < cClientHandle * > : : const_iterator itr = m_pState - > LoadedByClient . begin ( ) ; itr ! = m_pState - > LoadedByClient . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
2012-01-30 11:01:45 -05:00
if ( * itr = = a_Exclude )
{
continue ;
}
2011-10-03 14:41:19 -04:00
( * itr ) - > Send ( a_Packet ) ;
2012-01-30 11:01:45 -05:00
} // for itr - LoadedByClient[]
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
void cChunk : : LoadFromJson ( const Json : : Value & a_Value )
{
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > BlockListCriticalSection ) ;
2011-10-03 14:41:19 -04:00
// Load chests
Json : : Value AllChests = a_Value . get ( " Chests " , Json : : nullValue ) ;
2012-01-30 11:01:45 -05:00
if ( ! AllChests . empty ( ) )
2011-10-03 14:41:19 -04:00
{
for ( Json : : Value : : iterator itr = AllChests . begin ( ) ; itr ! = AllChests . end ( ) ; + + itr )
{
Json : : Value & Chest = * itr ;
2011-10-30 20:52:20 -04:00
cChestEntity * ChestEntity = new cChestEntity ( 0 , 0 , 0 , this ) ;
2011-10-03 14:41:19 -04:00
if ( ! ChestEntity - > LoadFromJson ( Chest ) )
{
LOGERROR ( " ERROR READING CHEST FROM JSON! " ) ;
delete ChestEntity ;
}
2011-12-26 16:54:08 -05:00
else m_pState - > BlockEntities . push_back ( ChestEntity ) ;
2011-10-03 14:41:19 -04:00
}
}
// Load furnaces
Json : : Value AllFurnaces = a_Value . get ( " Furnaces " , Json : : nullValue ) ;
if ( ! AllFurnaces . empty ( ) )
{
for ( Json : : Value : : iterator itr = AllFurnaces . begin ( ) ; itr ! = AllFurnaces . end ( ) ; + + itr )
{
Json : : Value & Furnace = * itr ;
2011-10-30 20:52:20 -04:00
cFurnaceEntity * FurnaceEntity = new cFurnaceEntity ( 0 , 0 , 0 , this ) ;
2011-10-03 14:41:19 -04:00
if ( ! FurnaceEntity - > LoadFromJson ( Furnace ) )
{
LOGERROR ( " ERROR READING FURNACE FROM JSON! " ) ;
delete FurnaceEntity ;
}
2011-12-26 16:54:08 -05:00
else m_pState - > BlockEntities . push_back ( FurnaceEntity ) ;
2011-10-03 14:41:19 -04:00
}
}
// Load signs
Json : : Value AllSigns = a_Value . get ( " Signs " , Json : : nullValue ) ;
if ( ! AllSigns . empty ( ) )
{
for ( Json : : Value : : iterator itr = AllSigns . begin ( ) ; itr ! = AllSigns . end ( ) ; + + itr )
{
Json : : Value & Sign = * itr ;
2011-10-30 20:52:20 -04:00
cSignEntity * SignEntity = new cSignEntity ( E_BLOCK_SIGN_POST , 0 , 0 , 0 , this ) ;
2011-10-03 14:41:19 -04:00
if ( ! SignEntity - > LoadFromJson ( Sign ) )
{
LOGERROR ( " ERROR READING SIGN FROM JSON! " ) ;
delete SignEntity ;
}
2011-12-26 16:54:08 -05:00
else m_pState - > BlockEntities . push_back ( SignEntity ) ;
2011-10-03 14:41:19 -04:00
}
}
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
void cChunk : : SaveToJson ( Json : : Value & a_Value )
{
Json : : Value AllChests ;
Json : : Value AllFurnaces ;
Json : : Value AllSigns ;
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > BlockListCriticalSection ) ;
for ( std : : list < cBlockEntity * > : : iterator itr = m_pState - > BlockEntities . begin ( ) ; itr ! = m_pState - > BlockEntities . end ( ) ; + + itr )
2011-10-03 14:41:19 -04:00
{
cBlockEntity * BlockEntity = * itr ;
switch ( BlockEntity - > GetBlockType ( ) )
{
2012-01-30 11:01:45 -05:00
case E_BLOCK_CHEST :
2011-10-03 14:41:19 -04:00
{
cChestEntity * ChestEntity = reinterpret_cast < cChestEntity * > ( BlockEntity ) ;
Json : : Value NewChest ;
ChestEntity - > SaveToJson ( NewChest ) ;
AllChests . append ( NewChest ) ;
2012-01-30 11:01:45 -05:00
break ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
case E_BLOCK_FURNACE :
2011-10-03 14:41:19 -04:00
{
cFurnaceEntity * FurnaceEntity = reinterpret_cast < cFurnaceEntity * > ( BlockEntity ) ;
Json : : Value NewFurnace ;
FurnaceEntity - > SaveToJson ( NewFurnace ) ;
AllFurnaces . append ( NewFurnace ) ;
2012-01-30 11:01:45 -05:00
break ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
2011-10-03 14:41:19 -04:00
{
cSignEntity * SignEntity = reinterpret_cast < cSignEntity * > ( BlockEntity ) ;
Json : : Value NewSign ;
SignEntity - > SaveToJson ( NewSign ) ;
AllSigns . append ( NewSign ) ;
2012-01-30 11:01:45 -05:00
break ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
default :
{
assert ( ! " Unhandled blocktype in BlockEntities list while saving to JSON " ) ;
break ;
}
} // switch (BlockEntity->GetBlockType())
} // for itr - BlockEntities[]
2011-10-03 14:41:19 -04:00
if ( ! AllChests . empty ( ) )
a_Value [ " Chests " ] = AllChests ;
if ( ! AllFurnaces . empty ( ) )
a_Value [ " Furnaces " ] = AllFurnaces ;
if ( ! AllSigns . empty ( ) )
a_Value [ " Signs " ] = AllSigns ;
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
EntityList & cChunk : : GetEntities ( )
{
2011-12-26 16:54:08 -05:00
return m_pState - > Entities ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
const ClientHandleList & cChunk : : GetClients ( )
{
2011-12-26 16:54:08 -05:00
return m_pState - > LoadedByClient ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
void cChunk : : AddTickBlockEntity ( cFurnaceEntity * a_Entity )
{
2011-12-26 16:54:08 -05:00
m_pState - > TickBlockEntities . remove ( a_Entity ) ;
m_pState - > TickBlockEntities . push_back ( a_Entity ) ;
2011-10-03 14:41:19 -04:00
}
2012-01-30 11:01:45 -05:00
2011-10-03 14:41:19 -04:00
void cChunk : : RemoveTickBlockEntity ( cFurnaceEntity * a_Entity )
{
2011-12-26 16:54:08 -05:00
m_pState - > TickBlockEntities . remove ( a_Entity ) ;
2011-11-01 18:27:09 -04:00
}
2011-12-22 16:36:24 -05:00
2012-01-30 11:01:45 -05:00
2011-12-22 16:36:24 -05:00
void cChunk : : PositionToWorldPosition ( int a_ChunkX , int a_ChunkY , int a_ChunkZ , int & a_X , int & a_Y , int & a_Z )
{
a_Y = a_ChunkY ;
a_X = m_PosX * 16 + a_ChunkX ;
a_Z = m_PosZ * 16 + a_ChunkZ ;
2011-12-31 16:08:23 -05:00
}
2012-01-30 11:01:45 -05:00
2012-01-19 13:12:39 -05:00
void cChunk : : AddReference ( )
2012-01-01 11:20:52 -05:00
{
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > ReferenceCriticalSection ) ;
2012-01-19 13:12:39 -05:00
m_pState - > NumRefs + + ;
2012-01-01 11:20:52 -05:00
}
2012-01-30 11:01:45 -05:00
2012-01-19 13:12:39 -05:00
void cChunk : : RemoveReference ( )
2012-01-01 11:20:52 -05:00
{
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > ReferenceCriticalSection ) ;
2012-01-19 13:12:39 -05:00
m_pState - > NumRefs - - ;
2012-01-30 11:01:45 -05:00
assert ( m_pState - > NumRefs > = 0 ) ;
if ( m_pState - > NumRefs < 0 )
2012-01-01 11:20:52 -05:00
{
2012-01-19 13:12:39 -05:00
LOGWARN ( " WARNING: cChunk: Tried to remove reference, but the chunk is not referenced! " ) ;
2012-01-01 11:20:52 -05:00
}
}
2012-01-30 11:01:45 -05:00
2012-01-01 11:20:52 -05:00
int cChunk : : GetReferenceCount ( )
{
2012-01-30 11:01:45 -05:00
cCSLock Lock ( m_pState - > ReferenceCriticalSection ) ;
2012-01-26 08:13:44 -05:00
int Refs = m_pState - > NumRefs ;
2012-01-01 11:20:52 -05:00
return Refs ;
}
2011-12-31 16:08:23 -05:00
2012-01-30 11:01:45 -05:00
2011-12-31 16:08:23 -05:00
# if !C_CHUNK_USE_INLINE
# include "cChunk.inc"
2012-01-30 11:01:45 -05:00
# endif