mirror of
https://github.com/CPunch/gopenfusion.git
synced 2024-11-21 23:10:06 +00:00
started chunking
This commit is contained in:
parent
f0b9bc6ed6
commit
f6ab7a9b5d
@ -3,43 +3,27 @@ package entity
|
|||||||
import (
|
import (
|
||||||
"log"
|
"log"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/CPunch/gopenfusion/config"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ChunkPosition struct {
|
|
||||||
X int
|
|
||||||
Y int
|
|
||||||
}
|
|
||||||
|
|
||||||
func makeChunkPosition(x, y int) ChunkPosition {
|
|
||||||
return ChunkPosition{
|
|
||||||
X: x / (config.VIEW_DISTANCE / 3),
|
|
||||||
Y: y / (config.VIEW_DISTANCE / 3),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type Chunk struct {
|
type Chunk struct {
|
||||||
Position ChunkPosition
|
Position ChunkPosition
|
||||||
entities map[Entity]struct{}
|
Entities map[Entity]struct{}
|
||||||
lock sync.Mutex
|
lock sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewChunk(position ChunkPosition) *Chunk {
|
func NewChunk(position ChunkPosition) *Chunk {
|
||||||
return &Chunk{
|
return &Chunk{
|
||||||
Position: position,
|
Position: position,
|
||||||
entities: make(map[Entity]struct{}),
|
Entities: make(map[Entity]struct{}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) AddEntity(entity Entity) {
|
func (c *Chunk) AddEntity(entity Entity) {
|
||||||
entity.SetChunk(c)
|
c.Entities[entity] = struct{}{}
|
||||||
c.entities[entity] = struct{}{}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) RemoveEntity(entity Entity) {
|
func (c *Chunk) RemoveEntity(entity Entity) {
|
||||||
entity.SetChunk(nil)
|
delete(c.Entities, entity)
|
||||||
delete(c.entities, entity)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// send packet to all peers in this chunk and kill each peer if error
|
// send packet to all peers in this chunk and kill each peer if error
|
||||||
@ -47,7 +31,7 @@ func (c *Chunk) SendPacket(typeID uint32, pkt ...interface{}) {
|
|||||||
c.lock.Lock()
|
c.lock.Lock()
|
||||||
defer c.lock.Unlock()
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
for entity := range c.entities {
|
for entity := range c.Entities {
|
||||||
if entity.GetKind() != ENTITY_KIND_PLAYER {
|
if entity.GetKind() != ENTITY_KIND_PLAYER {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -59,6 +43,7 @@ func (c *Chunk) SendPacket(typeID uint32, pkt ...interface{}) {
|
|||||||
peer := plr.Peer
|
peer := plr.Peer
|
||||||
|
|
||||||
if err := peer.Send(typeID, pkt...); err != nil {
|
if err := peer.Send(typeID, pkt...); err != nil {
|
||||||
|
log.Printf("Error sending packet to peer %p: %v", peer, err)
|
||||||
peer.Kill()
|
peer.Kill()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -77,3 +62,20 @@ func (c *Chunk) GetAdjacentPositions() []ChunkPosition {
|
|||||||
{c.Position.X + 1, c.Position.Y + 1},
|
{c.Position.X + 1, c.Position.Y + 1},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://stackoverflow.com/a/45428032 lol
|
||||||
|
func ChunkSliceDifference(a, b []*Chunk) []*Chunk {
|
||||||
|
m := make(map[*Chunk]struct{})
|
||||||
|
for _, item := range b {
|
||||||
|
m[item] = struct{}{}
|
||||||
|
}
|
||||||
|
|
||||||
|
var diff []*Chunk
|
||||||
|
for _, item := range a {
|
||||||
|
if _, ok := m[item]; !ok {
|
||||||
|
diff = append(diff, item)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return diff
|
||||||
|
}
|
||||||
|
15
core/entity/chunkposition.go
Normal file
15
core/entity/chunkposition.go
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
package entity
|
||||||
|
|
||||||
|
import "github.com/CPunch/gopenfusion/config"
|
||||||
|
|
||||||
|
type ChunkPosition struct {
|
||||||
|
X int
|
||||||
|
Y int
|
||||||
|
}
|
||||||
|
|
||||||
|
func MakeChunkPosition(x, y int) ChunkPosition {
|
||||||
|
return ChunkPosition{
|
||||||
|
X: x / (config.VIEW_DISTANCE / 3),
|
||||||
|
Y: y / (config.VIEW_DISTANCE / 3),
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,7 @@
|
|||||||
package entity
|
package entity
|
||||||
|
|
||||||
|
import "github.com/CPunch/gopenfusion/core/protocol"
|
||||||
|
|
||||||
type EntityKind int
|
type EntityKind int
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -10,11 +12,14 @@ const (
|
|||||||
type Entity interface {
|
type Entity interface {
|
||||||
GetKind() EntityKind
|
GetKind() EntityKind
|
||||||
|
|
||||||
GetChunk() *Chunk
|
GetChunk() ChunkPosition
|
||||||
GetPosition() (x int, y int, z int)
|
GetPosition() (x int, y int, z int)
|
||||||
GetAngle() int
|
GetAngle() int
|
||||||
|
|
||||||
SetChunk(chunk *Chunk)
|
SetChunk(chunk ChunkPosition)
|
||||||
SetPosition(x, y, z int)
|
SetPosition(x, y, z int)
|
||||||
SetAngle(angle int)
|
SetAngle(angle int)
|
||||||
|
|
||||||
|
DisappearFromViewOf(peer *protocol.CNPeer)
|
||||||
|
EnterIntoViewOf(peer *protocol.CNPeer)
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@ import (
|
|||||||
|
|
||||||
type Player struct {
|
type Player struct {
|
||||||
Peer *protocol.CNPeer
|
Peer *protocol.CNPeer
|
||||||
CurrentChunk *Chunk
|
Chunk ChunkPosition
|
||||||
PlayerID int
|
PlayerID int
|
||||||
AccountID int
|
AccountID int
|
||||||
AccountLevel int
|
AccountLevel int
|
||||||
@ -35,6 +35,7 @@ type Player struct {
|
|||||||
ActiveNanoSlotNum int
|
ActiveNanoSlotNum int
|
||||||
Fatigue int
|
Fatigue int
|
||||||
CurrentMissionID int
|
CurrentMissionID int
|
||||||
|
IPCState int8
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== Entity interface ====================
|
// ==================== Entity interface ====================
|
||||||
@ -43,8 +44,8 @@ func (plr *Player) GetKind() EntityKind {
|
|||||||
return ENTITY_KIND_PLAYER
|
return ENTITY_KIND_PLAYER
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) GetChunk() *Chunk {
|
func (plr *Player) GetChunk() ChunkPosition {
|
||||||
return plr.CurrentChunk
|
return plr.Chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) GetPosition() (x int, y int, z int) {
|
func (plr *Player) GetPosition() (x int, y int, z int) {
|
||||||
@ -55,6 +56,10 @@ func (plr *Player) GetAngle() int {
|
|||||||
return plr.Angle
|
return plr.Angle
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (plr *Player) SetChunk(chunk ChunkPosition) {
|
||||||
|
plr.Chunk = chunk
|
||||||
|
}
|
||||||
|
|
||||||
func (plr *Player) SetPosition(x, y, z int) {
|
func (plr *Player) SetPosition(x, y, z int) {
|
||||||
plr.X = x
|
plr.X = x
|
||||||
plr.Y = y
|
plr.Y = y
|
||||||
@ -65,8 +70,16 @@ func (plr *Player) SetAngle(angle int) {
|
|||||||
plr.Angle = angle
|
plr.Angle = angle
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) SetChunk(chunk *Chunk) {
|
func (plr *Player) DisappearFromViewOf(peer *protocol.CNPeer) {
|
||||||
plr.CurrentChunk = chunk
|
peer.Send(protocol.P_FE2CL_PC_EXIT, protocol.SP_FE2CL_PC_EXIT{
|
||||||
|
IID: int32(plr.PlayerID),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plr *Player) EnterIntoViewOf(peer *protocol.CNPeer) {
|
||||||
|
peer.Send(protocol.P_FE2CL_PC_NEW, protocol.SP_FE2CL_PC_NEW{
|
||||||
|
PCAppearanceData: plr.GetAppearanceData(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) ToPCLoadData2CL() protocol.SPCLoadData2CL {
|
func (plr *Player) ToPCLoadData2CL() protocol.SPCLoadData2CL {
|
||||||
@ -96,3 +109,19 @@ func (plr *Player) ToPCLoadData2CL() protocol.SPCLoadData2CL {
|
|||||||
IFatigue: 50,
|
IFatigue: 50,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (plr *Player) GetAppearanceData() protocol.SPCAppearanceData {
|
||||||
|
return protocol.SPCAppearanceData{
|
||||||
|
IID: int32(plr.PlayerID),
|
||||||
|
IHP: int32(plr.HP),
|
||||||
|
ILv: int16(plr.Level),
|
||||||
|
IX: int32(plr.X),
|
||||||
|
IY: int32(plr.Y),
|
||||||
|
IZ: int32(plr.Z),
|
||||||
|
IAngle: int32(plr.Angle),
|
||||||
|
PCStyle: plr.PCStyle,
|
||||||
|
IPCState: plr.IPCState,
|
||||||
|
ItemEquip: plr.Equip,
|
||||||
|
Nano: protocol.SNano{}, //plr.Nanos[plr.ActiveNanoSlotNum],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
package shard
|
package shard
|
||||||
|
|
||||||
import "github.com/CPunch/gopenfusion/core/entity"
|
import (
|
||||||
|
"github.com/CPunch/gopenfusion/core/entity"
|
||||||
|
)
|
||||||
|
|
||||||
func (server *ShardServer) getChunk(pos entity.ChunkPosition) *entity.Chunk {
|
func (server *ShardServer) getChunk(pos entity.ChunkPosition) *entity.Chunk {
|
||||||
chunk, ok := server.chunks[pos]
|
chunk, ok := server.chunks[pos]
|
||||||
@ -12,12 +14,67 @@ func (server *ShardServer) getChunk(pos entity.ChunkPosition) *entity.Chunk {
|
|||||||
return chunk
|
return chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) getViewableChunks(plr *entity.Player) []*entity.Chunk {
|
func (server *ShardServer) getViewableChunks(pos entity.ChunkPosition) []*entity.Chunk {
|
||||||
chunks := make([]*entity.Chunk, 0, 9)
|
chunks := make([]*entity.Chunk, 0, 9)
|
||||||
|
for _, pos := range server.getChunk(pos).GetAdjacentPositions() {
|
||||||
for _, pos := range plr.GetChunk().GetAdjacentPositions() {
|
|
||||||
chunks = append(chunks, server.getChunk(pos))
|
chunks = append(chunks, server.getChunk(pos))
|
||||||
}
|
}
|
||||||
|
|
||||||
return chunks
|
return chunks
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (server *ShardServer) sendPacketToViewableChunks(plr *entity.Player, typeID uint32, pkt ...interface{}) error {
|
||||||
|
for _, chunk := range server.getViewableChunks(plr.Chunk) {
|
||||||
|
chunk.SendPacket(typeID, pkt...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (server *ShardServer) removeEntityFromChunks(chunks []*entity.Chunk, this entity.Entity) {
|
||||||
|
for _, chunk := range chunks {
|
||||||
|
for e, _ := range chunk.Entities {
|
||||||
|
if e.GetKind() == entity.ENTITY_KIND_PLAYER {
|
||||||
|
otherPlr := e.(*entity.Player)
|
||||||
|
this.DisappearFromViewOf(otherPlr.Peer)
|
||||||
|
}
|
||||||
|
|
||||||
|
if this.GetKind() == entity.ENTITY_KIND_PLAYER {
|
||||||
|
thisPlr := this.(*entity.Player)
|
||||||
|
e.DisappearFromViewOf(thisPlr.Peer)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (server *ShardServer) addEntityToChunks(chunks []*entity.Chunk, this entity.Entity) {
|
||||||
|
for _, chunk := range chunks {
|
||||||
|
for e, _ := range chunk.Entities {
|
||||||
|
if e.GetKind() == entity.ENTITY_KIND_PLAYER {
|
||||||
|
otherPlr := e.(*entity.Player)
|
||||||
|
this.EnterIntoViewOf(otherPlr.Peer)
|
||||||
|
}
|
||||||
|
|
||||||
|
if this.GetKind() == entity.ENTITY_KIND_PLAYER {
|
||||||
|
thisPlr := this.(*entity.Player)
|
||||||
|
e.EnterIntoViewOf(thisPlr.Peer)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (server *ShardServer) updateEntityChunk(e entity.Entity, from entity.ChunkPosition, to entity.ChunkPosition) {
|
||||||
|
oldViewables := server.getViewableChunks(from)
|
||||||
|
newViewables := server.getViewableChunks(to)
|
||||||
|
|
||||||
|
// compute differences
|
||||||
|
toExit := entity.ChunkSliceDifference(oldViewables, newViewables)
|
||||||
|
toEnter := entity.ChunkSliceDifference(newViewables, oldViewables)
|
||||||
|
|
||||||
|
// update chunks
|
||||||
|
server.removeEntityFromChunks(toExit, e)
|
||||||
|
server.addEntityToChunks(toEnter, e)
|
||||||
|
server.getChunk(from).RemoveEntity(e)
|
||||||
|
server.getChunk(to).AddEntity(e)
|
||||||
|
e.SetChunk(to)
|
||||||
|
}
|
||||||
|
@ -22,9 +22,10 @@ func (server *ShardServer) attachPlayer(peer *protocol.CNPeer, meta redis.LoginM
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
plr.Peer = peer
|
||||||
|
|
||||||
server.setPlayer(peer, plr)
|
server.setPlayer(peer, plr)
|
||||||
return plr, err
|
return plr, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) RequestEnter(peer *protocol.CNPeer, pkt protocol.Packet) error {
|
func (server *ShardServer) RequestEnter(peer *protocol.CNPeer, pkt protocol.Packet) error {
|
||||||
@ -57,6 +58,8 @@ func (server *ShardServer) RequestEnter(peer *protocol.CNPeer, pkt protocol.Pack
|
|||||||
peer.SetActiveKey(protocol.USE_FE)
|
peer.SetActiveKey(protocol.USE_FE)
|
||||||
|
|
||||||
log.Printf("Player %d (AccountID %d) entered\n", resp.IID, loginData.AccountID)
|
log.Printf("Player %d (AccountID %d) entered\n", resp.IID, loginData.AccountID)
|
||||||
|
|
||||||
|
server.updatePlayerPosition(peer, int(plr.X), int(plr.Y), int(plr.Z), int(plr.Angle))
|
||||||
return peer.Send(protocol.P_FE2CL_REP_PC_ENTER_SUCC, resp)
|
return peer.Send(protocol.P_FE2CL_REP_PC_ENTER_SUCC, resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package shard
|
package shard
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/CPunch/gopenfusion/core/entity"
|
||||||
"github.com/CPunch/gopenfusion/core/protocol"
|
"github.com/CPunch/gopenfusion/core/protocol"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -9,12 +10,16 @@ func (server *ShardServer) updatePlayerPosition(peer *protocol.CNPeer, X, Y, Z,
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
newPos := entity.MakeChunkPosition(X, Y)
|
||||||
|
oldPos := plr.Chunk
|
||||||
|
|
||||||
plr.X = X
|
plr.X = X
|
||||||
plr.Y = Y
|
plr.Y = Y
|
||||||
plr.Z = Z
|
plr.Z = Z
|
||||||
plr.Angle = Angle
|
plr.Angle = Angle
|
||||||
|
if newPos != oldPos {
|
||||||
|
server.updateEntityChunk(plr, oldPos, newPos)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -115,6 +115,14 @@ func (server *ShardServer) disconnect(peer *protocol.CNPeer) {
|
|||||||
server.peerLock.Lock()
|
server.peerLock.Lock()
|
||||||
defer server.peerLock.Unlock()
|
defer server.peerLock.Unlock()
|
||||||
|
|
||||||
|
// remove from chunk(s)
|
||||||
|
plr, ok := server.peers[peer]
|
||||||
|
if ok {
|
||||||
|
log.Printf("Player %d (AccountID %d) disconnected\n", plr.PlayerID, plr.AccountID)
|
||||||
|
server.removeEntityFromChunks(server.getViewableChunks(plr.Chunk), plr)
|
||||||
|
server.getChunk(plr.Chunk).RemoveEntity(plr)
|
||||||
|
}
|
||||||
|
|
||||||
log.Printf("Peer %p disconnected from SHARD\n", peer)
|
log.Printf("Peer %p disconnected from SHARD\n", peer)
|
||||||
delete(server.peers, peer)
|
delete(server.peers, peer)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user