struct column and chunk, and optimization
This commit is contained in:
1
.idea/dictionaries/cjd00.xml
generated
1
.idea/dictionaries/cjd00.xml
generated
@ -7,6 +7,7 @@
|
|||||||
<w>minecraft</w>
|
<w>minecraft</w>
|
||||||
<w>mojang</w>
|
<w>mojang</w>
|
||||||
<w>rcon</w>
|
<w>rcon</w>
|
||||||
|
<w>teleport</w>
|
||||||
</words>
|
</words>
|
||||||
</dictionary>
|
</dictionary>
|
||||||
</component>
|
</component>
|
7
.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
7
.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<component name="InspectionProjectProfileManager">
|
||||||
|
<profile version="1.0">
|
||||||
|
<option name="myName" value="Project Default" />
|
||||||
|
<inspection_tool class="GoCommentStart" enabled="true" level="WEAK WARNING" enabled_by_default="true" />
|
||||||
|
<inspection_tool class="GoExportedElementShouldHaveComment" enabled="true" level="WEAK WARNING" enabled_by_default="true" />
|
||||||
|
</profile>
|
||||||
|
</component>
|
@ -1,6 +1,6 @@
|
|||||||
# Go-MC
|
# Go-MC
|
||||||

|

|
||||||

|

|
||||||
[](https://godoc.org/github.com/Tnze/go-mc)
|
[](https://godoc.org/github.com/Tnze/go-mc)
|
||||||
[](https://goreportcard.com/report/github.com/Tnze/go-mc)
|
[](https://goreportcard.com/report/github.com/Tnze/go-mc)
|
||||||
[](https://travis-ci.org/Tnze/go-mc)
|
[](https://travis-ci.org/Tnze/go-mc)
|
||||||
|
@ -465,6 +465,7 @@ type blockEntities []blockEntitie
|
|||||||
type blockEntitie struct {
|
type blockEntitie struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Decode implement net.packet.FieldDecoder
|
||||||
func (c *chunkData) Decode(r pk.DecodeReader) error {
|
func (c *chunkData) Decode(r pk.DecodeReader) error {
|
||||||
var Size pk.VarInt
|
var Size pk.VarInt
|
||||||
if err := Size.Decode(r); err != nil {
|
if err := Size.Decode(r); err != nil {
|
||||||
@ -477,14 +478,15 @@ func (c *chunkData) Decode(r pk.DecodeReader) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Decode implement net.packet.FieldDecoder
|
||||||
func (b *blockEntities) Decode(r pk.DecodeReader) error {
|
func (b *blockEntities) Decode(r pk.DecodeReader) error {
|
||||||
var NumberofBlockEntities pk.VarInt
|
var nobe pk.VarInt // Number of BlockEntities
|
||||||
if err := NumberofBlockEntities.Decode(r); err != nil {
|
if err := nobe.Decode(r); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
*b = make(blockEntities, NumberofBlockEntities)
|
*b = make(blockEntities, nobe)
|
||||||
decoder := nbt.NewDecoder(r)
|
decoder := nbt.NewDecoder(r)
|
||||||
for i := 0; i < int(NumberofBlockEntities); i++ {
|
for i := 0; i < int(nobe); i++ {
|
||||||
if err := decoder.Decode(&(*b)[i]); err != nil {
|
if err := decoder.Decode(&(*b)[i]); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
91
nbt/read.go
91
nbt/read.go
@ -5,6 +5,7 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
"math"
|
"math"
|
||||||
"reflect"
|
"reflect"
|
||||||
)
|
)
|
||||||
@ -16,46 +17,33 @@ func Unmarshal(data []byte, v interface{}) error {
|
|||||||
func (d *Decoder) Decode(v interface{}) error {
|
func (d *Decoder) Decode(v interface{}) error {
|
||||||
val := reflect.ValueOf(v)
|
val := reflect.ValueOf(v)
|
||||||
if val.Kind() != reflect.Ptr {
|
if val.Kind() != reflect.Ptr {
|
||||||
return errors.New("non-pointer passed to Unmarshal")
|
return errors.New("nbt: non-pointer passed to Unmarshal")
|
||||||
}
|
|
||||||
|
|
||||||
// check the head
|
|
||||||
compress, err := d.checkCompressed()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("check compressed fail: %v", err)
|
|
||||||
}
|
|
||||||
if compress != "" {
|
|
||||||
return fmt.Errorf("data may compressed with %s", compress)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//start read NBT
|
//start read NBT
|
||||||
tagType, tagName, err := d.readTag()
|
tagType, tagName, err := d.readTag()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return fmt.Errorf("nbt: %v", err)
|
||||||
}
|
}
|
||||||
return d.unmarshal(val.Elem(), tagType, tagName)
|
|
||||||
|
if c := d.checkCompressed(tagType); c != "" {
|
||||||
|
return fmt.Errorf("nbt: unknown Tag, maybe need %s", c)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = d.unmarshal(val.Elem(), tagType, tagName)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("nbt: %v", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// check the first byte and return if it use compress
|
// check the first byte and return if it use compress
|
||||||
func (d *Decoder) checkCompressed() (compress string, err error) {
|
func (d *Decoder) checkCompressed(head byte) (compress string) {
|
||||||
var head byte
|
if head == 0x1f { //gzip
|
||||||
head, err = d.r.ReadByte()
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if head <= 12 { //NBT
|
|
||||||
compress = ""
|
|
||||||
} else if head == 0x1f { //gzip
|
|
||||||
compress = "gzip"
|
compress = "gzip"
|
||||||
} else if head == 0x78 { //zlib
|
} else if head == 0x78 { //zlib
|
||||||
compress = "zlib"
|
compress = "zlib"
|
||||||
} else {
|
|
||||||
compress = "unknown"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
err = d.r.UnreadByte()
|
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -187,12 +175,15 @@ func (d *Decoder) unmarshal(val reflect.Value, tagType byte, tagName string) err
|
|||||||
}
|
}
|
||||||
|
|
||||||
case TagByteArray:
|
case TagByteArray:
|
||||||
var ba []byte
|
|
||||||
aryLen, err := d.readInt32()
|
aryLen, err := d.readInt32()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if ba, err = d.readNByte(int(aryLen)); err != nil {
|
if aryLen < 0 {
|
||||||
|
return errors.New("byte array len less than 0")
|
||||||
|
}
|
||||||
|
ba := make([]byte, aryLen)
|
||||||
|
if _, err = io.ReadFull(d.r, ba); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -366,6 +357,7 @@ func (d *Decoder) unmarshal(val reflect.Value, tagType byte, tagName string) err
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d *Decoder) rawRead(tagType byte) error {
|
func (d *Decoder) rawRead(tagType byte) error {
|
||||||
|
var buf [8]byte
|
||||||
switch tagType {
|
switch tagType {
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("unknown to read 0x%02x", tagType)
|
return fmt.Errorf("unknown to read 0x%02x", tagType)
|
||||||
@ -376,20 +368,21 @@ func (d *Decoder) rawRead(tagType byte) error {
|
|||||||
_, err := d.readString()
|
_, err := d.readString()
|
||||||
return err
|
return err
|
||||||
case TagShort:
|
case TagShort:
|
||||||
_, err := d.readNByte(2)
|
_, err := io.ReadFull(d.r, buf[:2])
|
||||||
return err
|
return err
|
||||||
case TagInt, TagFloat:
|
case TagInt, TagFloat:
|
||||||
_, err := d.readNByte(4)
|
_, err := io.ReadFull(d.r, buf[:4])
|
||||||
return err
|
return err
|
||||||
case TagLong, TagDouble:
|
case TagLong, TagDouble:
|
||||||
_, err := d.readNByte(8)
|
_, err := io.ReadFull(d.r, buf[:8])
|
||||||
return err
|
return err
|
||||||
case TagByteArray:
|
case TagByteArray:
|
||||||
aryLen, err := d.readInt32()
|
aryLen, err := d.readInt32()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if _, err = d.readNByte(int(aryLen)); err != nil {
|
|
||||||
|
if _, err = io.CopyN(ioutil.Discard, d.r, int64(aryLen)); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
case TagIntArray:
|
case TagIntArray:
|
||||||
@ -458,30 +451,22 @@ func (d *Decoder) readTag() (tagType byte, tagName string, err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Decoder) readNByte(n int) (buf []byte, err error) {
|
|
||||||
if n < 0 {
|
|
||||||
return nil, errors.New("read n byte cannot less than 0")
|
|
||||||
}
|
|
||||||
|
|
||||||
buf = make([]byte, n)
|
|
||||||
_, err = io.ReadFull(d.r, buf)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *Decoder) readInt16() (int16, error) {
|
func (d *Decoder) readInt16() (int16, error) {
|
||||||
data, err := d.readNByte(2)
|
var data [2]byte
|
||||||
|
_, err := io.ReadFull(d.r, data[:])
|
||||||
return int16(data[0])<<8 | int16(data[1]), err
|
return int16(data[0])<<8 | int16(data[1]), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Decoder) readInt32() (int32, error) {
|
func (d *Decoder) readInt32() (int32, error) {
|
||||||
data, err := d.readNByte(4)
|
var data [4]byte
|
||||||
|
_, err := io.ReadFull(d.r, data[:])
|
||||||
return int32(data[0])<<24 | int32(data[1])<<16 |
|
return int32(data[0])<<24 | int32(data[1])<<16 |
|
||||||
int32(data[2])<<8 | int32(data[3]), err
|
int32(data[2])<<8 | int32(data[3]), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Decoder) readInt64() (int64, error) {
|
func (d *Decoder) readInt64() (int64, error) {
|
||||||
data, err := d.readNByte(8)
|
var data [8]byte
|
||||||
|
_, err := io.ReadFull(d.r, data[:])
|
||||||
return int64(data[0])<<56 | int64(data[1])<<48 |
|
return int64(data[0])<<56 | int64(data[1])<<48 |
|
||||||
int64(data[2])<<40 | int64(data[3])<<32 |
|
int64(data[2])<<40 | int64(data[3])<<32 |
|
||||||
int64(data[4])<<24 | int64(data[5])<<16 |
|
int64(data[4])<<24 | int64(data[5])<<16 |
|
||||||
@ -492,7 +477,15 @@ func (d *Decoder) readString() (string, error) {
|
|||||||
length, err := d.readInt16()
|
length, err := d.readInt16()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
|
} else if length < 0 {
|
||||||
|
return "", errors.New("string length less than 0")
|
||||||
}
|
}
|
||||||
buf, err := d.readNByte(int(length))
|
|
||||||
return string(buf), err
|
var str string
|
||||||
|
if length > 0 {
|
||||||
|
buf := make([]byte, length)
|
||||||
|
_, err = io.ReadFull(d.r, buf)
|
||||||
|
str = string(buf)
|
||||||
|
}
|
||||||
|
return str, err
|
||||||
}
|
}
|
||||||
|
71
save/chunk.go
Normal file
71
save/chunk.go
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
package save
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"compress/gzip"
|
||||||
|
"compress/zlib"
|
||||||
|
"errors"
|
||||||
|
"github.com/Tnze/go-mc/nbt"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Column is 16* chunk
|
||||||
|
type Column struct {
|
||||||
|
DataVersion int
|
||||||
|
Level struct {
|
||||||
|
Heightmaps map[string][]int64
|
||||||
|
Structures struct {
|
||||||
|
References map[string][]int64
|
||||||
|
Starts map[string]struct {
|
||||||
|
ID string `nbt:"id"`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Entities
|
||||||
|
// LiquidTicks
|
||||||
|
// PostProcessing
|
||||||
|
Sections []Chunk
|
||||||
|
// TileEntities
|
||||||
|
// TileTicks
|
||||||
|
InhabitedTime int64
|
||||||
|
IsLightOn byte `nbt:"isLightOn"`
|
||||||
|
LastUpdate int64
|
||||||
|
Status string
|
||||||
|
PosX int32 `nbt:"xPos"`
|
||||||
|
PosY int32 `nbt:"yPos"`
|
||||||
|
Biomes []int32
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type Chunk struct {
|
||||||
|
Palette []Block
|
||||||
|
Y byte
|
||||||
|
BlockLight []byte
|
||||||
|
BlockStates []int64
|
||||||
|
SkyLight []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type Block struct {
|
||||||
|
Name string
|
||||||
|
Properties map[string]interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load read column data from []byte
|
||||||
|
func (c *Column) Load(data []byte) (err error) {
|
||||||
|
var r io.Reader = bytes.NewReader(data[1:])
|
||||||
|
|
||||||
|
switch data[0] {
|
||||||
|
default:
|
||||||
|
err = errors.New("unknown compression")
|
||||||
|
case 1:
|
||||||
|
r, err = gzip.NewReader(r)
|
||||||
|
case 2:
|
||||||
|
r, err = zlib.NewReader(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = nbt.NewDecoder(r).Decode(c)
|
||||||
|
return
|
||||||
|
}
|
53
save/chunk_test.go
Normal file
53
save/chunk_test.go
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
package save
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/Tnze/go-mc/save/region"
|
||||||
|
"math/rand"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestColumn(t *testing.T) {
|
||||||
|
var c Column
|
||||||
|
r, err := region.Open("testdata/region/r.0.0.mca")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer r.Close()
|
||||||
|
|
||||||
|
data, err := r.ReadSector(0, 0)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = c.Load(data)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Logf("%+v", c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkColumn_Load(b *testing.B) {
|
||||||
|
// Test how many time we load a chunk
|
||||||
|
var c Column
|
||||||
|
r, err := region.Open("testdata/region/r.-1.-1.mca")
|
||||||
|
if err != nil {
|
||||||
|
b.Fatal(err)
|
||||||
|
}
|
||||||
|
defer r.Close()
|
||||||
|
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
//x, y := (i%1024)/32, (i%1024)%32
|
||||||
|
x, y := rand.Intn(32), rand.Intn(32)
|
||||||
|
|
||||||
|
data, err := r.ReadSector(x, y)
|
||||||
|
if err != nil {
|
||||||
|
b.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = c.Load(data)
|
||||||
|
if err != nil {
|
||||||
|
b.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -8,6 +8,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Region contain 32*32 chunks in one .mca file
|
||||||
type Region struct {
|
type Region struct {
|
||||||
f *os.File
|
f *os.File
|
||||||
offsets [32][32]int32
|
offsets [32][32]int32
|
||||||
@ -26,9 +27,9 @@ func In(cx, cy int) (int, int) {
|
|||||||
return cx & 31, cy & 31
|
return cx & 31, cy & 31
|
||||||
}
|
}
|
||||||
|
|
||||||
// OpenRegion open a .mca file and read the head.
|
// Open open a .mca file and read the head.
|
||||||
// Close the Region after used.
|
// Close the Region after used.
|
||||||
func OpenRegion(name string) (r *Region, err error) {
|
func Open(name string) (r *Region, err error) {
|
||||||
r = new(Region)
|
r = new(Region)
|
||||||
r.sectors = make(map[int32]bool)
|
r.sectors = make(map[int32]bool)
|
||||||
|
|
||||||
@ -67,6 +68,7 @@ func OpenRegion(name string) (r *Region, err error) {
|
|||||||
return r, nil
|
return r, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close close the region file
|
||||||
func (r *Region) Close() error {
|
func (r *Region) Close() error {
|
||||||
return r.f.Close()
|
return r.f.Close()
|
||||||
}
|
}
|
||||||
@ -75,6 +77,7 @@ func sectorLoc(offset int32) (o, s int32) {
|
|||||||
return offset >> 8, offset & 0xFF
|
return offset >> 8, offset & 0xFF
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadSector find and read the Chunk data from region
|
||||||
func (r *Region) ReadSector(x, y int) (data []byte, err error) {
|
func (r *Region) ReadSector(x, y int) (data []byte, err error) {
|
||||||
offset, _ := sectorLoc(r.offsets[x][y])
|
offset, _ := sectorLoc(r.offsets[x][y])
|
||||||
|
|
||||||
@ -99,6 +102,7 @@ func (r *Region) ReadSector(x, y int) (data []byte, err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WriteSector write Chunk data into region file
|
||||||
func (r *Region) WriteSector(x, y int, data []byte) error {
|
func (r *Region) WriteSector(x, y int, data []byte) error {
|
||||||
need := int32(len(data)+4)/4096 + 1
|
need := int32(len(data)+4)/4096 + 1
|
||||||
n, now := sectorLoc(r.offsets[x][y])
|
n, now := sectorLoc(r.offsets[x][y])
|
||||||
@ -155,6 +159,11 @@ func (r *Region) WriteSector(x, y int, data []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ExistSector return if a sector is exist
|
||||||
|
func (r *Region) ExistSector(x, y int) bool {
|
||||||
|
return r.offsets[x][y] != 0
|
||||||
|
}
|
||||||
|
|
||||||
func (r *Region) findSpace(need int32) (n int32) {
|
func (r *Region) findSpace(need int32) (n int32) {
|
||||||
for i := int32(0); i < need; i++ {
|
for i := int32(0); i < need; i++ {
|
||||||
if r.sectors[n+i] {
|
if r.sectors[n+i] {
|
||||||
|
@ -22,7 +22,7 @@ func TestIn(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestReadRegion(t *testing.T) {
|
func TestReadRegion(t *testing.T) {
|
||||||
r, err := OpenRegion("../testdata/region/r.0.-1.mca")
|
r, err := Open("../testdata/region/r.0.-1.mca")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -77,3 +77,21 @@ func TestFindSpace(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCountChunks(t *testing.T) {
|
||||||
|
r, err := Open("../testdata/region/r.-1.-1.mca")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer r.Close()
|
||||||
|
|
||||||
|
var count int
|
||||||
|
for i := 0; i < 32; i++ {
|
||||||
|
for j := 0; j < 32; j++ {
|
||||||
|
if r.ExistSector(i, j) {
|
||||||
|
count++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t.Logf("chunk count: %d", count)
|
||||||
|
}
|
||||||
|
BIN
save/testdata/region/r.-1.-1.mca
vendored
BIN
save/testdata/region/r.-1.-1.mca
vendored
Binary file not shown.
Reference in New Issue
Block a user