Только один VertexBuilder для их всех

This commit is contained in:
DBotThePony 2023-02-13 17:49:21 +07:00
parent 0052adf89a
commit 0fd5fb0be7
Signed by: DBot
GPG Key ID: DCC23B5715498507
18 changed files with 482 additions and 586 deletions

View File

@ -6,9 +6,7 @@ import ru.dbotthepony.kstarbound.client.gl.GLStateTracker
import ru.dbotthepony.kstarbound.client.gl.program.GLHardLightGeometryProgram
import ru.dbotthepony.kstarbound.client.gl.program.GLSoftLightGeometryProgram
import ru.dbotthepony.kstarbound.client.gl.vertex.GeometryType
import ru.dbotthepony.kstarbound.client.gl.vertex.StatefulVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.quad
import ru.dbotthepony.kstarbound.client.gl.vertex.shadowLine
import ru.dbotthepony.kstarbound.client.gl.vertex.StreamVertexBuilder
import ru.dbotthepony.kstarbound.client.render.ConfiguredStaticMesh
import ru.dbotthepony.kstarbound.client.render.entity.EntityRenderer
import ru.dbotthepony.kstarbound.client.render.GPULightRenderer
@ -191,13 +189,13 @@ class ClientChunk(world: ClientWorld, pos: ChunkPos) : Chunk<ClientWorld, Client
}
private inner class ShadowGeometryTracker(val x: Int, val y: Int) : GPULightRenderer.ShadowGeometryRenderer {
private val hardShadowGeometry by lazy(LazyThreadSafetyMode.NONE) { StatefulVertexBuilder(state, GPULightRenderer.SHADOW_FORMAT, GeometryType.LINES) }
private val hardShadowGeometry by lazy(LazyThreadSafetyMode.NONE) { StreamVertexBuilder(state, GPULightRenderer.SHADOW_FORMAT, GeometryType.LINES) }
private var hardShadowGeometryRev = -1
private val softShadowGeometry by lazy(LazyThreadSafetyMode.NONE) { StatefulVertexBuilder(state, GPULightRenderer.SHADOW_FORMAT_SOFT, GeometryType.QUADS_ALTERNATIVE) }
private val softShadowGeometry by lazy(LazyThreadSafetyMode.NONE) { StreamVertexBuilder(state, GPULightRenderer.SHADOW_FORMAT_SOFT, GeometryType.QUADS_ALTERNATIVE) }
private var softShadowGeometryRev = -1
private fun buildGeometry(builder: StatefulVertexBuilder, line: (StatefulVertexBuilder, Float, Float, Float, Float) -> Unit) {
builder.begin()
private fun buildGeometry(builder: StreamVertexBuilder, line: (StreamVertexBuilder, Float, Float, Float, Float) -> Unit) {
builder.builder.begin()
for (x in this.x * SHADOW_GEOMETRY_SQUARE_SIZE until (this.x + 1) * SHADOW_GEOMETRY_SQUARE_SIZE) {
for (y in this.y * SHADOW_GEOMETRY_SQUARE_SIZE until (this.y + 1) * SHADOW_GEOMETRY_SQUARE_SIZE) {
@ -228,14 +226,16 @@ class ClientChunk(world: ClientWorld, pos: ChunkPos) : Chunk<ClientWorld, Client
hardShadowGeometryRev = tileChangeset
buildGeometry(hardShadowGeometry) { it, x0, y0, x1, y1 ->
it.vertex().pushVec2f(x0, y0)
it.vertex().pushVec2f(x1, y1)
it.builder.vertex().pushVec2f(x0, y0)
it.builder.vertex().pushVec2f(x1, y1)
}
}
fun buildSoftGeometry() {
softShadowGeometryRev = tileChangeset
buildGeometry(softShadowGeometry, StatefulVertexBuilder::shadowLine)
buildGeometry(softShadowGeometry) { it, x0, y0, x1, y1 ->
it.builder.shadowLine(x0, y0, x1, y1)
}
}
override fun renderHardGeometry(
@ -398,14 +398,14 @@ class ClientChunk(world: ClientWorld, pos: ChunkPos) : Chunk<ClientWorld, Client
val builder = program.builder
for (type in types) {
builder.begin()
builder.builder.begin()
for (x in 0 until CHUNK_SIZE) {
for (y in 0 until CHUNK_SIZE) {
val state = getLiquid(x, y)
if (state != null && state.def === type) {
builder.quad(x.toFloat(), y.toFloat(), x + 1f, y + state.level)
builder.builder.quad(x.toFloat(), y.toFloat(), x + 1f, y + state.level)
}
}
}

View File

@ -3,7 +3,6 @@ package ru.dbotthepony.kstarbound.client.gl
import org.apache.logging.log4j.LogManager
import org.lwjgl.opengl.GL
import org.lwjgl.opengl.GL46.*
import ru.dbotthepony.kstarbound.Starbound
import ru.dbotthepony.kstarbound.api.ISBFileLocator
import ru.dbotthepony.kstarbound.client.freetype.FreeType
import ru.dbotthepony.kstarbound.client.freetype.InvalidArgumentException
@ -15,10 +14,9 @@ import ru.dbotthepony.kstarbound.client.gl.shader.GLTransformableProgram
import ru.dbotthepony.kstarbound.client.gl.vertex.GLAttributeList
import ru.dbotthepony.kstarbound.client.gl.vertex.StreamVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.GeometryType
import ru.dbotthepony.kstarbound.client.gl.vertex.quad
import ru.dbotthepony.kstarbound.client.gl.vertex.VertexBuilder
import ru.dbotthepony.kstarbound.client.render.Box2DRenderer
import ru.dbotthepony.kstarbound.client.render.Font
import ru.dbotthepony.kstarbound.client.render.TileRenderers
import ru.dbotthepony.kvector.api.IStruct4f
import ru.dbotthepony.kvector.matrix.Matrix4fStack
import ru.dbotthepony.kvector.util2d.AABB
@ -565,38 +563,21 @@ class GLStateTracker(val locator: ISBFileLocator) {
val programs = GLPrograms(this)
val flat2DLines = object : GLStreamBuilderList {
override val small by lazy {
return@lazy StreamVertexBuilder(this@GLStateTracker, GLAttributeList.VEC2F, GeometryType.LINES, 1024)
}
}
val flat2DTriangles = object : GLStreamBuilderList {
override val small by lazy {
return@lazy StreamVertexBuilder(this@GLStateTracker, GLAttributeList.VEC2F, GeometryType.TRIANGLES, 1024)
}
}
val flat2DTexturedQuads = object : GLStreamBuilderList {
override val small by lazy {
return@lazy StreamVertexBuilder(this@GLStateTracker, GLAttributeList.VERTEX_TEXTURE, GeometryType.QUADS, 1024)
}
}
val quadWireframe by lazy {
StreamVertexBuilder(this, GLAttributeList.VEC2F, GeometryType.QUADS_AS_LINES_WIREFRAME, 16384)
}
val flat2DLines by lazy { StreamVertexBuilder(this, GLAttributeList.VEC2F, GeometryType.LINES) }
val flat2DTriangles by lazy { StreamVertexBuilder(this, GLAttributeList.VEC2F, GeometryType.TRIANGLES) }
val flat2DTexturedQuads by lazy { StreamVertexBuilder(this, GLAttributeList.VERTEX_TEXTURE, GeometryType.QUADS) }
val quadWireframe by lazy { StreamVertexBuilder(this, GLAttributeList.VEC2F, GeometryType.QUADS_AS_LINES_WIREFRAME) }
val matrixStack = Matrix4fStack()
val freeType = FreeType()
val font = Font(this)
inline fun quadWireframe(color: Color = Color.WHITE, lambda: (StreamVertexBuilder) -> Unit) {
inline fun quadWireframe(color: Color = Color.WHITE, lambda: (VertexBuilder) -> Unit) {
val builder = quadWireframe
builder.begin()
lambda.invoke(builder)
builder.builder.begin()
lambda.invoke(builder.builder)
builder.upload()
programs.flat.use()
@ -606,11 +587,11 @@ class GLStateTracker(val locator: ISBFileLocator) {
builder.draw(GL_LINES)
}
inline fun quadColor(lambda: (StreamVertexBuilder) -> Unit) {
inline fun quadColor(lambda: (VertexBuilder) -> Unit) {
val builder = programs.flatColor.builder
builder.begin()
lambda.invoke(builder)
builder.builder.begin()
lambda.invoke(builder.builder)
builder.upload()
programs.flatColor.use()
@ -619,7 +600,7 @@ class GLStateTracker(val locator: ISBFileLocator) {
builder.draw(GL_TRIANGLES)
}
inline fun quadWireframe(value: AABB, color: Color = Color.WHITE, chain: (StreamVertexBuilder) -> Unit = {}) {
inline fun quadWireframe(value: AABB, color: Color = Color.WHITE, chain: (VertexBuilder) -> Unit = {}) {
quadWireframe(color) {
it.quad(value.mins.x.toFloat(), value.mins.y.toFloat(), value.maxs.x.toFloat(), value.maxs.y.toFloat())
chain(it)

View File

@ -9,7 +9,6 @@ import ru.dbotthepony.kstarbound.client.gl.vertex.GLAttributeList
import ru.dbotthepony.kstarbound.client.gl.vertex.StreamVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.GeometryType
import ru.dbotthepony.kstarbound.client.gl.vertex.QuadTransformers
import ru.dbotthepony.kstarbound.client.gl.vertex.quad
import ru.dbotthepony.kstarbound.client.render.GPULightRenderer
import ru.dbotthepony.kvector.vector.Color
import kotlin.properties.ReadOnlyProperty
@ -123,8 +122,8 @@ class GLColorQuadProgram(state: GLStateTracker) : GLInternalProgram(state, "scre
private val builder by lazy {
val builder = StreamVertexBuilder(state, FORMAT, GeometryType.QUADS, 1)
builder.begin()
builder.quad(-1f, -1f, 1f, 1f)
builder.builder.begin()
builder.builder.quad(-1f, -1f, 1f, 1f)
builder.upload()
builder
@ -173,8 +172,8 @@ class GLTextureQuadProgram(state: GLStateTracker) : GLInternalProgram(state, "sc
private val builder by lazy {
val builder = StreamVertexBuilder(state, FORMAT, GeometryType.QUADS, 1)
builder.begin()
builder.quad(-1f, -1f, 1f, 1f, QuadTransformers.uv())
builder.builder.begin()
builder.builder.quad(-1f, -1f, 1f, 1f, QuadTransformers.uv())
builder.upload()
builder
@ -201,8 +200,8 @@ class GLTextureBlurredQuadProgram(state: GLStateTracker) : GLInternalProgram(sta
private val builder by lazy {
val builder = StreamVertexBuilder(state, FORMAT, GeometryType.QUADS, 1)
builder.begin()
builder.quad(-1f, -1f, 1f, 1f, QuadTransformers.uv())
builder.builder.begin()
builder.builder.quad(-1f, -1f, 1f, 1f, QuadTransformers.uv())
builder.upload()
builder

View File

@ -1,308 +0,0 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import org.lwjgl.opengl.GL46
import org.lwjgl.opengl.GL46.GL_UNSIGNED_INT
import org.lwjgl.opengl.GL46.GL_UNSIGNED_SHORT
import org.lwjgl.opengl.GL46.GL_UNSIGNED_BYTE
import ru.dbotthepony.kstarbound.client.gl.GLType
import ru.dbotthepony.kstarbound.client.gl.VertexBufferObject
import ru.dbotthepony.kstarbound.client.gl.checkForGLError
import ru.dbotthepony.kstarbound.util.writeLEFloat
import ru.dbotthepony.kstarbound.util.writeLEInt
import ru.dbotthepony.kstarbound.util.writeLEShort
import ru.dbotthepony.kvector.util2d.AABB
import java.io.OutputStream
import kotlin.math.cos
import kotlin.math.sin
private fun put(type: Int, memory: OutputStream, value: Int) {
when (type) {
GL_UNSIGNED_SHORT -> memory.writeLEShort(value)
GL_UNSIGNED_BYTE -> memory.write(value)
else -> memory.writeLEInt(value)
}
}
/**
* Класс для построения геометрии для загрузки в память видеокарты.
*/
@Suppress("unchecked_cast")
abstract class AbstractVertexBuilder<out T : AbstractVertexBuilder<T>>(
val attributes: GLAttributeList,
val type: GeometryType,
) {
protected abstract val vertexMemory: OutputStream
protected abstract val elementMemory: OutputStream
/**
* [GL_UNSIGNED_BYTE], [GL_UNSIGNED_SHORT] или [GL_UNSIGNED_INT]
*
* Обязана быть статичным числом.
*
* Если нет, и его надо изменить, то это можно только делать внутри [ensureIndexCapacity].
*
* Подкласс обязан самостоятельно изменить тип существующих элементов внутри [elementMemory]
*/
abstract val elementIndexType: Int
protected abstract fun ensureIndexCapacity()
protected abstract fun resetMemory()
fun begin() {
inVertex = false
attributeIndex = 0
elementIndexOffset = 0
vertexCount = 0
indexCount = 0
resetMemory()
}
private var inVertex = false
private var attributeIndex = 0
protected var elementIndexOffset = 0
private set
var vertexCount = 0
private set
var indexCount = 0
private set
protected fun checkBounds() {
if (attributeIndex >= attributes.size) {
throw IndexOutOfBoundsException("Tried to add new attribute when already added all attributes")
}
}
fun expect(type: GLType) {
checkBounds()
if (attributes[attributeIndex].glType != type) {
throw IllegalStateException("Expected attribute type $type, got ${attributes[attributeIndex].name}[${attributes[attributeIndex].glType}] (at position $attributeIndex)")
}
}
fun expect(name: String) {
checkBounds()
if (attributes[attributeIndex].name != name) {
throw IllegalStateException("Expected attribute name $name, got ${attributes[attributeIndex].name}[${attributes[attributeIndex].glType}] (at position $attributeIndex)")
}
}
protected abstract fun doUpload(vbo: VertexBufferObject, ebo: VertexBufferObject, drawType: Int)
fun upload(vbo: VertexBufferObject, ebo: VertexBufferObject, drawType: Int = GL46.GL_DYNAMIC_DRAW) {
require(vbo.isArray) { "$vbo is not an array" }
require(ebo.isElementArray) { "$vbo is not an element array" }
end()
check(elementVertices == 0) { "Not fully built vertex element ($type requires ${type.elements} vertex points to be present, yet last strip has only $elementVertices elements)" }
doUpload(vbo, ebo, drawType)
checkForGLError()
}
private var elementVertices = 0
fun end() {
if (inVertex) {
inVertex = false
if (attributeIndex != attributes.size) {
throw IllegalStateException("Unfinished vertex, we are at $attributeIndex, while we have ${attributes.size} attributes")
}
vertexCount++
elementVertices++
if (elementVertices == type.elements) {
ensureIndexCapacity()
elementVertices = 0
val elementMemory = elementMemory
val elementIndexType = elementIndexType
for (index in type.indices) {
put(elementIndexType, elementMemory, index + elementIndexOffset)
}
elementIndexOffset += type.elements
indexCount += type.indices.size
}
}
}
fun vertex(): T {
end()
inVertex = true
attributeIndex = 0
return this as T
}
fun pushVec4f(x: Float, y: Float, z: Float, w: Float): T {
expect(GLType.VEC4F)
val memory = vertexMemory
memory.writeLEFloat(x)
memory.writeLEFloat(y)
memory.writeLEFloat(z)
memory.writeLEFloat(w)
attributeIndex++
return this as T
}
fun pushVec3f(x: Float, y: Float, z: Float): T {
expect(GLType.VEC3F)
val memory = vertexMemory
memory.writeLEFloat(x)
memory.writeLEFloat(y)
memory.writeLEFloat(z)
attributeIndex++
return this as T
}
fun pushVec2f(x: Float, y: Float): T {
expect(GLType.VEC2F)
val memory = vertexMemory
memory.writeLEFloat(x)
memory.writeLEFloat(y)
attributeIndex++
return this as T
}
fun push(value: Float): T {
expect(GLType.FLOAT)
vertexMemory.writeLEFloat(value)
attributeIndex++
return this as T
}
}
fun <T : AbstractVertexBuilder<T>> T.shadowLine(x0: Float, y0: Float, x1: Float, y1: Float): T {
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(0f, 0f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(0f, 1f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(1f, 1f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(1f, 0f)
return this
}
fun <T : AbstractVertexBuilder<T>> T.dfShadowLine(x0: Float, y0: Float, x1: Float, y1: Float): T {
shadowLine(x0, y0, x1, y1)
shadowLine(x1, y1, x0, y0)
return this
}
fun <T : AbstractVertexBuilder<T>> T.shadowQuad(x0: Float, y0: Float, x1: Float, y1: Float): T {
shadowLine(x0, y0, x1, y0)
shadowLine(x1, y0, x1, y1)
shadowLine(x1, y1, x0, y1)
shadowLine(x0, y1, x0, y0)
return this
}
// Помощники
fun <T : AbstractVertexBuilder<T>> T.quad(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): T {
check(type.elements == 4) { "Currently building $type" }
lambda(vertex().pushVec2f(x0, y0), 0).end()
lambda(vertex().pushVec2f(x1, y0), 1).end()
lambda(vertex().pushVec2f(x0, y1), 2).end()
lambda(vertex().pushVec2f(x1, y1), 3).end()
return this
}
fun <T : AbstractVertexBuilder<T>> T.quadRotated(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
x: Float,
y: Float,
angle: Double,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): T {
check(type.elements == 4) { "Currently building $type" }
val s = sin(angle).toFloat()
val c = cos(angle).toFloat()
lambda(vertex().pushVec2f(x + x0 * c - s * y0, y + s * x0 + c * y0), 0).end()
lambda(vertex().pushVec2f(x + x1 * c - s * y0, y + s * x1 + c * y0), 1).end()
lambda(vertex().pushVec2f(x + x0 * c - s * y1, y + s * x0 + c * y1), 2).end()
lambda(vertex().pushVec2f(x + x1 * c - s * y1, y + s * x1 + c * y1), 3).end()
return this
}
fun <T : AbstractVertexBuilder<T>> T.quad(aabb: AABB, lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM): T {
return quad(
aabb.mins.x.toFloat(),
aabb.mins.y.toFloat(),
aabb.maxs.x.toFloat(),
aabb.maxs.y.toFloat(),
lambda
)
}
fun <T : AbstractVertexBuilder<T>> T.quadZ(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
z: Float,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): T {
check(type.elements == 4) { "Currently building $type" }
lambda(vertex().pushVec3f(x0, y0, z), 0).end()
lambda(vertex().pushVec3f(x1, y0, z), 1).end()
lambda(vertex().pushVec3f(x0, y1, z), 2).end()
lambda(vertex().pushVec3f(x1, y1, z), 3).end()
return this
}
fun <T : AbstractVertexBuilder<T>> T.quadRotatedZ(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
z: Float,
x: Float,
y: Float,
angle: Double,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): T {
check(type.elements == 4) { "Currently building $type" }
val s = sin(angle).toFloat()
val c = cos(angle).toFloat()
lambda(vertex().pushVec3f(x + x0 * c - s * y0, y + s * x0 + c * y0, z), 0).end()
lambda(vertex().pushVec3f(x + x1 * c - s * y0, y + s * x1 + c * y0, z), 1).end()
lambda(vertex().pushVec3f(x + x0 * c - s * y1, y + s * x0 + c * y1, z), 2).end()
lambda(vertex().pushVec3f(x + x1 * c - s * y1, y + s * x1 + c * y1, z), 3).end()
return this
}

View File

@ -1,53 +0,0 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import org.lwjgl.opengl.GL46
import ru.dbotthepony.kstarbound.client.gl.VertexBufferObject
import ru.dbotthepony.kstarbound.util.ByteBufferOutputStream
/**
* Создаёт буфер для данных вне кучи, записывает данные напрямую в него,
* при [upload] загружает данные из буфера напрямую в память видеокарты.
*/
open class DirectVertexBuilder<T : DirectVertexBuilder<T>>(
attributes: GLAttributeList,
type: GeometryType,
val maxElements: Int,
) : AbstractVertexBuilder<DirectVertexBuilder<T>>(attributes, type) {
val maxIndexCount = maxElements * type.indices.size
val maxVertexCount = maxElements * type.elements
final override val elementIndexType: Int = when (maxIndexCount) {
// api performance issue 102: glDrawElements uses element index type 'GL_UNSIGNED_BYTE' that is not optimal for the current hardware configuration; consider using 'GL_UNSIGNED_SHORT' instead
// in 0 .. 255 -> GL_UNSIGNED_BYTE
in 0 .. 65535 -> GL46.GL_UNSIGNED_SHORT
else -> GL46.GL_UNSIGNED_INT
}
final override val vertexMemory = ByteBufferOutputStream.directLE(maxVertexCount * attributes.stride)
final override val elementMemory = ByteBufferOutputStream.directLE(maxIndexCount * (if (elementIndexType == GL46.GL_UNSIGNED_SHORT) 2 else 4))
override fun ensureIndexCapacity() {
if (vertexCount > maxVertexCount) {
throw IndexOutOfBoundsException("Vertex count overflow (can hold max of $maxElements elements, that's $maxVertexCount vertexes)")
}
}
override fun resetMemory() {
vertexMemory.position = 0
elementMemory.position = 0
}
override fun doUpload(vbo: VertexBufferObject, ebo: VertexBufferObject, drawType: Int) {
val vertexPos = vertexMemory.position
val elementPos = elementMemory.position
vertexMemory.position = 0
elementMemory.position = 0
vbo.bufferData(vertexMemory.buffer, drawType, length = vertexPos.toLong())
ebo.bufferData(elementMemory.buffer, drawType, length = elementPos.toLong())
vertexMemory.position = vertexPos
elementMemory.position = elementPos
}
}

View File

@ -1,6 +1,16 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
enum class GeometryType(val elements: Int, val indices: IntArray) {
enum class GeometryType(
/**
* Число вершин у одного элемента
*/
val elements: Int,
/**
* Индекс вершин одного элемента
*/
val indices: IntArray
) {
AS_IS(1, intArrayOf(0)),
LINES(2, intArrayOf(0, 1)),
TRIANGLES(3, intArrayOf(0, 1, 2)),

View File

@ -1,61 +0,0 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import it.unimi.dsi.fastutil.io.FastByteArrayInputStream
import it.unimi.dsi.fastutil.io.FastByteArrayOutputStream
import org.lwjgl.opengl.GL11.GL_UNSIGNED_INT
import org.lwjgl.opengl.GL11.GL_UNSIGNED_SHORT
import ru.dbotthepony.kstarbound.client.gl.VertexBufferObject
import ru.dbotthepony.kstarbound.util.readLEShort
import ru.dbotthepony.kstarbound.util.writeLEInt
import java.nio.ByteBuffer
/**
* Vertex Builder который хранит все данные на куче, при [upload] создаётся [ByteBuffer] вне кучи,
* в него записываются данные из кучи, данные загружаются на видеокарту.
*
* Полезен в случаях, когда размер данных для загрузки заранее не известен.
*/
open class HeapVertexBuilder<T : HeapVertexBuilder<T>>(
attributes: GLAttributeList,
type: GeometryType,
) : AbstractVertexBuilder<T>(attributes, type) {
final override val vertexMemory = FastByteArrayOutputStream()
final override val elementMemory = FastByteArrayOutputStream()
final override var elementIndexType: Int = GL_UNSIGNED_SHORT
private set
final override fun ensureIndexCapacity() {
if (elementIndexType == GL_UNSIGNED_SHORT && elementMemory.length / 2 + type.indices.size >= 30000) {
val backing = elementMemory.array
val copy = FastByteArrayInputStream(ByteArray(elementMemory.length) { backing[it] })
elementIndexType = GL_UNSIGNED_INT
val elementMemory = elementMemory
elementMemory.reset()
for (i in 0 until (copy.length / 2)) {
elementMemory.writeLEInt(copy.readLEShort())
}
}
}
final override fun resetMemory() {
vertexMemory.reset()
elementMemory.reset()
}
final override fun doUpload(vbo: VertexBufferObject, ebo: VertexBufferObject, drawType: Int) {
val vboMemory = ByteBuffer.allocateDirect(vertexMemory.length)
vboMemory.put(vertexMemory.array, 0, vertexMemory.length)
val eboMemory = ByteBuffer.allocateDirect(elementMemory.length)
eboMemory.put(elementMemory.array, 0, elementMemory.length)
vboMemory.position(0)
eboMemory.position(0)
vbo.bufferData(vboMemory, drawType)
ebo.bufferData(eboMemory, drawType)
}
}

View File

@ -1,6 +1,6 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
typealias QuadVertexTransformer = (AbstractVertexBuilder<*>, Int) -> AbstractVertexBuilder<*>
typealias QuadVertexTransformer = (VertexBuilder, Int) -> VertexBuilder
val EMPTY_VERTEX_TRANSFORM: QuadVertexTransformer = { it, _ -> it }

View File

@ -1,47 +0,0 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import org.lwjgl.opengl.GL46
import ru.dbotthepony.kstarbound.client.gl.GLStateTracker
import ru.dbotthepony.kstarbound.client.gl.checkForGLError
import java.io.Closeable
class StatefulVertexBuilder(
val state: GLStateTracker,
attributes: GLAttributeList,
type: GeometryType,
) : HeapVertexBuilder<StatefulVertexBuilder>(attributes, type), Closeable {
private val vao = state.newVAO()
private val vbo = state.newVBO()
private val ebo = state.newEBO()
init {
vao.bind()
vbo.bind()
ebo.bind()
attributes.apply(vao, true)
vao.unbind()
vbo.unbind()
ebo.unbind()
}
fun upload(drawType: Int = GL46.GL_STATIC_DRAW) {
upload(vbo, ebo, drawType)
}
fun bind() = vao.bind()
fun unbind() = vao.unbind()
fun draw(primitives: Int = GL46.GL_TRIANGLES) {
bind()
GL46.glDrawElements(primitives, indexCount, elementIndexType, 0L)
checkForGLError()
}
override fun close() {
vao.close()
vbo.close()
ebo.close()
}
}

View File

@ -1,21 +1,20 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import org.lwjgl.opengl.GL46
import ru.dbotthepony.kstarbound.PIXELS_IN_STARBOUND_UNITf
import ru.dbotthepony.kstarbound.client.gl.GLStateTracker
import ru.dbotthepony.kstarbound.client.gl.GLTexture2D
import ru.dbotthepony.kstarbound.client.gl.checkForGLError
import java.io.Closeable
/**
* Класс-помощник для быстрого построения, загрузки и отрисовки геометрии
* Быстрое наполнение буфера вершинами, загрузка в память видеокарты, и отрисовка
*/
class StreamVertexBuilder(
val state: GLStateTracker,
attributes: GLAttributeList,
type: GeometryType,
maxElements: Int,
) : DirectVertexBuilder<StreamVertexBuilder>(attributes, type, maxElements), Closeable {
initialCapacity: Int = 64,
) : Closeable {
val builder = VertexBuilder(attributes, type, initialCapacity)
private val vao = state.newVAO()
private val vbo = state.newVBO()
private val ebo = state.newEBO()
@ -33,7 +32,7 @@ class StreamVertexBuilder(
}
fun upload(drawType: Int = GL46.GL_DYNAMIC_DRAW) {
upload(vbo, ebo, drawType)
builder.upload(vbo, ebo, drawType)
}
fun bind() = vao.bind()
@ -41,7 +40,7 @@ class StreamVertexBuilder(
fun draw(primitives: Int = GL46.GL_TRIANGLES) {
bind()
GL46.glDrawElements(primitives, indexCount, elementIndexType, 0L)
GL46.glDrawElements(primitives, builder.indexCount, builder.indexType, 0L)
checkForGLError()
}
@ -52,9 +51,9 @@ class StreamVertexBuilder(
}
fun singleSprite(x: Float, y: Float, width: Float, height: Float, z: Float = 5f, angle: Double = 0.0, transformer: QuadVertexTransformer) {
begin()
builder.begin()
quadRotatedZ(x, y, width, height, z, 0f, 0f, angle, transformer)
builder.quadRotatedZ(x, y, width, height, z, 0f, 0f, angle, transformer)
upload()
draw()

View File

@ -0,0 +1,407 @@
package ru.dbotthepony.kstarbound.client.gl.vertex
import org.lwjgl.opengl.GL46
import org.lwjgl.opengl.GL46.GL_UNSIGNED_INT
import org.lwjgl.opengl.GL46.GL_UNSIGNED_SHORT
import org.lwjgl.opengl.GL46.GL_UNSIGNED_BYTE
import ru.dbotthepony.kstarbound.client.gl.GLType
import ru.dbotthepony.kstarbound.client.gl.VertexBufferObject
import ru.dbotthepony.kvector.util2d.AABB
import java.nio.ByteBuffer
import java.nio.ByteOrder
import kotlin.math.cos
import kotlin.math.sin
private fun interface IndexWriter {
fun write(buffer: ByteBuffer, value: Int)
}
private fun writer(type: Int): IndexWriter {
return when (type) {
GL_UNSIGNED_SHORT -> IndexWriter { it, value -> it.putShort(value.toShort()) }
GL_UNSIGNED_BYTE -> IndexWriter { it, value -> it.put(value.toByte()) }
else -> IndexWriter { it, value -> it.putInt(value) }
}
}
private fun chooseIndexType(capacity: Int): Int {
return when (capacity) {
// api performance issue 102: glDrawElements uses element index type 'GL_UNSIGNED_BYTE' that is not optimal for the current hardware configuration; consider using 'GL_UNSIGNED_SHORT' instead
// in 0 .. 255 -> GL_UNSIGNED_BYTE
in 0 .. 65535 -> GL_UNSIGNED_SHORT
else -> GL_UNSIGNED_INT
}
}
private fun indexSize(type: Int): Int {
return when (type) {
GL_UNSIGNED_BYTE -> 1
GL_UNSIGNED_SHORT -> 2
GL_UNSIGNED_INT -> 4
else -> throw IllegalStateException()
}
}
/**
* Создаёт буферы вне кучи для загрузки геометрии в них, с набором аттрибутов [attributes], типом [type] и указанным начальным размером.
*
* По мере необходимости, буферы могут быть увеличены в размерах.
*
* Загрузка в память видеокарты происходит напрямую из буферов, через метод [upload]
*/
@Suppress("unchecked_cast")
class VertexBuilder(
val attributes: GLAttributeList,
val type: GeometryType,
initialCapacity: Int = 64
) {
/**
* [GL_UNSIGNED_BYTE], [GL_UNSIGNED_SHORT] или [GL_UNSIGNED_INT]
*/
var indexType: Int = chooseIndexType(initialCapacity)
private set
var indexSize: Int = indexSize(indexType)
private set
/**
* В элементах, не вершинах
*/
private var capacity = initialCapacity
private var vertexMemory = ByteBuffer.allocateDirect(capacity * attributes.stride * type.elements).also { it.order(ByteOrder.LITTLE_ENDIAN) }
private var indexMemory = ByteBuffer.allocateDirect(capacity * indexSize * type.indices.size).also { it.order(ByteOrder.LITTLE_ENDIAN) }
private var indexWriter = writer(indexType)
private var inVertex = false
private var attributeIndex = 0
private var elementIndexOffset = 0
private var elementVertices = 0
var vertexCount = 0
private set
var indexCount = 0
private set
var elementCount = 0
private set
fun begin() {
inVertex = false
attributeIndex = 0
elementIndexOffset = 0
vertexCount = 0
indexCount = 0
elementCount = 0
vertexMemory.position(0)
indexMemory.position(0)
}
private fun checkBounds() {
if (attributeIndex >= attributes.size) {
throw IndexOutOfBoundsException("Tried to add new attribute when already added all attributes")
}
}
fun expect(type: GLType) {
checkBounds()
if (attributes[attributeIndex].glType != type) {
throw IllegalStateException("Expected attribute type $type, got ${attributes[attributeIndex].name}[${attributes[attributeIndex].glType}] (at position $attributeIndex)")
}
}
fun expect(name: String) {
checkBounds()
if (attributes[attributeIndex].name != name) {
throw IllegalStateException("Expected attribute name $name, got ${attributes[attributeIndex].name}[${attributes[attributeIndex].glType}] (at position $attributeIndex)")
}
}
fun upload(vbo: VertexBufferObject, ebo: VertexBufferObject, drawType: Int = GL46.GL_DYNAMIC_DRAW) {
require(vbo.isArray) { "$vbo is not an array" }
require(ebo.isElementArray) { "$ebo is not an element array" }
end()
check(elementVertices == 0) { "Not fully built vertex element ($type requires ${type.elements} vertex points to be present, yet last strip has only $elementVertices elements)" }
val vertexPos = vertexMemory.position()
val elementPos = indexMemory.position()
vertexMemory.position(0)
indexMemory.position(0)
vbo.bufferData(vertexMemory, drawType, length = vertexPos.toLong())
ebo.bufferData(indexMemory, drawType, length = elementPos.toLong())
vertexMemory.position(vertexPos)
indexMemory.position(elementPos)
}
fun end() {
if (inVertex) {
inVertex = false
if (attributeIndex != attributes.size) {
throw IllegalStateException("Unfinished vertex, we are at $attributeIndex, while we have ${attributes.size} attributes")
}
vertexCount++
elementVertices++
if (elementVertices == type.elements) {
elementCount++
elementVertices = 0
for (index in type.indices) {
indexWriter.write(indexMemory, index + elementIndexOffset)
}
elementIndexOffset += type.elements
indexCount += type.indices.size
if (capacity <= elementCount) {
check(this.vertexMemory.position() == this.vertexMemory.capacity()) { "${this.vertexMemory.position()} != ${this.vertexMemory.capacity()}" }
check(this.indexMemory.position() == this.indexMemory.capacity()) { "${this.indexMemory.position()} != ${this.indexMemory.capacity()}" }
val capacity = capacity * 2
val indexType = chooseIndexType(capacity)
val indexSize = indexSize(indexType)
val vertexMemory = ByteBuffer.allocateDirect(capacity * attributes.stride * type.elements)
vertexMemory.order(ByteOrder.LITTLE_ENDIAN)
this.vertexMemory.position(0)
vertexMemory.put(this.vertexMemory)
val indexMemory = ByteBuffer.allocateDirect(capacity * indexSize * type.indices.size)
indexMemory.order(ByteOrder.LITTLE_ENDIAN)
if (indexType != this.indexType) {
when (this.indexType) {
GL_UNSIGNED_BYTE -> {
when (indexType) {
GL_UNSIGNED_BYTE -> throw IllegalArgumentException()
GL_UNSIGNED_SHORT -> {
this.indexMemory.position(0)
for (i in 0 until this.capacity) {
indexMemory.putShort(this.indexMemory.get().toShort())
}
}
GL_UNSIGNED_INT -> {
this.indexMemory.position(0)
for (i in 0 until this.capacity) {
indexMemory.putInt(this.indexMemory.get().toInt())
}
}
}
}
GL_UNSIGNED_SHORT -> {
when (indexType) {
GL_UNSIGNED_BYTE -> throw IllegalArgumentException()
GL_UNSIGNED_SHORT -> throw IllegalArgumentException()
GL_UNSIGNED_INT -> {
this.indexMemory.position(0)
for (i in 0 until this.capacity) {
indexMemory.putInt(this.indexMemory.getShort().toInt())
}
}
}
}
else -> throw IllegalArgumentException()
}
} else {
this.indexMemory.position(0)
indexMemory.put(this.indexMemory)
}
this.capacity = capacity
this.vertexMemory = vertexMemory
this.indexMemory = indexMemory
this.indexType = indexType
this.indexSize = indexSize
this.indexWriter = writer(indexType)
}
}
}
}
fun vertex(): VertexBuilder {
end()
inVertex = true
attributeIndex = 0
return this
}
fun pushVec4f(x: Float, y: Float, z: Float, w: Float): VertexBuilder {
expect(GLType.VEC4F)
val memory = vertexMemory
memory.putFloat(x)
memory.putFloat(y)
memory.putFloat(z)
memory.putFloat(w)
attributeIndex++
return this
}
fun pushVec3f(x: Float, y: Float, z: Float): VertexBuilder {
expect(GLType.VEC3F)
val memory = vertexMemory
memory.putFloat(x)
memory.putFloat(y)
memory.putFloat(z)
attributeIndex++
return this
}
fun pushVec2f(x: Float, y: Float): VertexBuilder {
expect(GLType.VEC2F)
val memory = vertexMemory
memory.putFloat(x)
memory.putFloat(y)
attributeIndex++
return this
}
fun push(value: Float): VertexBuilder {
expect(GLType.FLOAT)
vertexMemory.putFloat(value)
attributeIndex++
return this
}
fun shadowLine(x0: Float, y0: Float, x1: Float, y1: Float): VertexBuilder {
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(0f, 0f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(0f, 1f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(1f, 1f)
vertex()
pushVec4f(x0, y0, x1, y1)
pushVec2f(1f, 0f)
return this
}
fun dfShadowLine(x0: Float, y0: Float, x1: Float, y1: Float): VertexBuilder {
shadowLine(x0, y0, x1, y1)
shadowLine(x1, y1, x0, y0)
return this
}
fun shadowQuad(x0: Float, y0: Float, x1: Float, y1: Float): VertexBuilder {
shadowLine(x0, y0, x1, y0)
shadowLine(x1, y0, x1, y1)
shadowLine(x1, y1, x0, y1)
shadowLine(x0, y1, x0, y0)
return this
}
// Помощники
fun quad(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): VertexBuilder {
check(type.elements == 4) { "Currently building $type" }
lambda(vertex().pushVec2f(x0, y0), 0).end()
lambda(vertex().pushVec2f(x1, y0), 1).end()
lambda(vertex().pushVec2f(x0, y1), 2).end()
lambda(vertex().pushVec2f(x1, y1), 3).end()
return this
}
fun quadRotated(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
x: Float,
y: Float,
angle: Double,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): VertexBuilder {
check(type.elements == 4) { "Currently building $type" }
val s = sin(angle).toFloat()
val c = cos(angle).toFloat()
lambda(vertex().pushVec2f(x + x0 * c - s * y0, y + s * x0 + c * y0), 0).end()
lambda(vertex().pushVec2f(x + x1 * c - s * y0, y + s * x1 + c * y0), 1).end()
lambda(vertex().pushVec2f(x + x0 * c - s * y1, y + s * x0 + c * y1), 2).end()
lambda(vertex().pushVec2f(x + x1 * c - s * y1, y + s * x1 + c * y1), 3).end()
return this
}
fun quad(aabb: AABB, lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM): VertexBuilder {
return quad(
aabb.mins.x.toFloat(),
aabb.mins.y.toFloat(),
aabb.maxs.x.toFloat(),
aabb.maxs.y.toFloat(),
lambda
)
}
fun quadZ(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
z: Float,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): VertexBuilder {
check(type.elements == 4) { "Currently building $type" }
lambda(vertex().pushVec3f(x0, y0, z), 0).end()
lambda(vertex().pushVec3f(x1, y0, z), 1).end()
lambda(vertex().pushVec3f(x0, y1, z), 2).end()
lambda(vertex().pushVec3f(x1, y1, z), 3).end()
return this
}
fun quadRotatedZ(
x0: Float,
y0: Float,
x1: Float,
y1: Float,
z: Float,
x: Float,
y: Float,
angle: Double,
lambda: QuadVertexTransformer = EMPTY_VERTEX_TRANSFORM
): VertexBuilder {
check(type.elements == 4) { "Currently building $type" }
val s = sin(angle).toFloat()
val c = cos(angle).toFloat()
lambda(vertex().pushVec3f(x + x0 * c - s * y0, y + s * x0 + c * y0, z), 0).end()
lambda(vertex().pushVec3f(x + x1 * c - s * y0, y + s * x1 + c * y0, z), 1).end()
lambda(vertex().pushVec3f(x + x0 * c - s * y1, y + s * x0 + c * y1, z), 2).end()
lambda(vertex().pushVec3f(x + x1 * c - s * y1, y + s * x1 + c * y1, z), 3).end()
return this
}
}

View File

@ -20,15 +20,15 @@ class Box2DRenderer(val state: GLStateTracker) : IDebugDraw {
override fun drawPolygon(vertices: List<Vector2d>, color: Color) {
require(vertices.size > 1) { "Vertex list had only ${vertices.size} namings in it" }
val builder = state.flat2DLines.small
val builder = state.flat2DLines
builder.begin()
builder.builder.begin()
for (i in vertices.indices) {
val current = vertices[i]
val next = vertices[(i + 1) % vertices.size]
builder.vertex().pushVec2f(current.x.toFloat(), current.y.toFloat())
builder.vertex().pushVec2f(next.x.toFloat(), next.y.toFloat())
builder.builder.vertex().pushVec2f(current.x.toFloat(), current.y.toFloat())
builder.builder.vertex().pushVec2f(next.x.toFloat(), next.y.toFloat())
}
builder.upload()
@ -43,18 +43,18 @@ class Box2DRenderer(val state: GLStateTracker) : IDebugDraw {
private fun drawSolid(vertices: List<Vector2d>, color: Color) {
require(vertices.size >= 3) { "Vertex list had only ${vertices.size} namings in it" }
val builder = state.flat2DTriangles.small
val builder = state.flat2DTriangles
builder.begin()
builder.builder.begin()
val zero = vertices[0]
for (i in 1 until vertices.size) {
val current = vertices[i]
val next = vertices[(i + 1) % vertices.size]
builder.vertex().pushVec2f(zero.x.toFloat(), zero.y.toFloat())
builder.vertex().pushVec2f(current.x.toFloat(), current.y.toFloat())
builder.vertex().pushVec2f(next.x.toFloat(), next.y.toFloat())
builder.builder.vertex().pushVec2f(zero.x.toFloat(), zero.y.toFloat())
builder.builder.vertex().pushVec2f(current.x.toFloat(), current.y.toFloat())
builder.builder.vertex().pushVec2f(next.x.toFloat(), next.y.toFloat())
}
builder.upload()

View File

@ -4,7 +4,7 @@ import org.lwjgl.opengl.GL46.*
import ru.dbotthepony.kstarbound.client.gl.program.GLShaderProgram
import ru.dbotthepony.kstarbound.client.gl.VertexArrayObject
import ru.dbotthepony.kstarbound.client.gl.checkForGLError
import ru.dbotthepony.kstarbound.client.gl.vertex.AbstractVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.VertexBuilder
import ru.dbotthepony.kvector.api.IFloatMatrix
import ru.dbotthepony.kvector.matrix.Matrix4fStack
@ -47,11 +47,11 @@ class ConfiguredStaticMesh(
) : AutoCloseable {
private var onClose = {}
constructor(programState: ConfiguredShaderProgram, builder: AbstractVertexBuilder<*>) : this(
constructor(programState: ConfiguredShaderProgram, builder: VertexBuilder) : this(
programState,
builder.indexCount,
programState.program.state.newVAO(),
builder.elementIndexType,
builder.indexType,
) {
val vbo = programState.program.state.newVBO()
val ebo = programState.program.state.newEBO()

View File

@ -7,10 +7,9 @@ import ru.dbotthepony.kstarbound.client.freetype.LoadFlag
import ru.dbotthepony.kstarbound.client.gl.*
import ru.dbotthepony.kstarbound.client.freetype.struct.FT_Pixel_Mode
import ru.dbotthepony.kstarbound.client.gl.vertex.GLAttributeList
import ru.dbotthepony.kstarbound.client.gl.vertex.HeapVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.QuadTransformers
import ru.dbotthepony.kstarbound.client.gl.vertex.GeometryType
import ru.dbotthepony.kstarbound.client.gl.vertex.quad
import ru.dbotthepony.kstarbound.client.gl.vertex.VertexBuilder
import ru.dbotthepony.kvector.matrix.Matrix4fStack
import ru.dbotthepony.kvector.vector.Color
@ -308,14 +307,14 @@ class Font(
ebo.bind()
vbo.bind()
val builder = HeapVertexBuilder(GLAttributeList.VERTEX_2D_TEXTURE, GeometryType.QUADS)
val builder = VertexBuilder(GLAttributeList.VERTEX_2D_TEXTURE, GeometryType.QUADS)
builder.quad(0f, 0f, width, height, QuadTransformers.uv())
builder.upload(vbo, ebo, GL_STATIC_DRAW)
builder.attributes.apply(vao, true)
indexCount = builder.indexCount
elementIndexType = builder.elementIndexType
elementIndexType = builder.indexType
vao.unbind()
ebo.unbind()

View File

@ -16,7 +16,6 @@ import ru.dbotthepony.kstarbound.client.gl.vertex.GLAttributeList
import ru.dbotthepony.kstarbound.client.gl.vertex.GeometryType
import ru.dbotthepony.kstarbound.client.gl.vertex.QuadTransformers
import ru.dbotthepony.kstarbound.client.gl.vertex.StreamVertexBuilder
import ru.dbotthepony.kstarbound.client.gl.vertex.quad
import ru.dbotthepony.kvector.matrix.Matrix4fStack
import ru.dbotthepony.kvector.vector.Color
import ru.dbotthepony.kvector.vector.nfloat.Vector2f
@ -182,8 +181,8 @@ class GPULightRenderer(val state: GLStateTracker) {
// Свет
val builder = state.programs.light.builder
builder.begin()
builder.quad(position.x - radius, position.y - radius, position.x + radius, position.y + radius, QuadTransformers.uv())
builder.builder.begin()
builder.builder.quad(position.x - radius, position.y - radius, position.x + radius, position.y + radius, QuadTransformers.uv())
builder.upload()
state.blendFunc = BLEND_MODE_INV
@ -248,8 +247,8 @@ class GPULightRenderer(val state: GLStateTracker) {
// Свет
val builder = state.programs.light.builder
builder.begin()
builder.quad(position.x - radius, position.y - radius, position.x + radius, position.y + radius, QuadTransformers.uv())
builder.builder.begin()
builder.builder.quad(position.x - radius, position.y - radius, position.x + radius, position.y + radius, QuadTransformers.uv())
builder.upload()
state.blendFunc = BLEND_MODE_SOFT

View File

@ -18,7 +18,7 @@ import kotlin.collections.HashMap
data class TileLayer(
val bakedProgramState: ConfiguredShaderProgram,
val vertexBuilder: AbstractVertexBuilder<*>,
val vertexBuilder: VertexBuilder,
val zPos: Int
)
@ -30,7 +30,7 @@ class TileLayerList {
*
* Если такого слоя нет, вызывается [compute] и создаётся новый [TileLayer], затем возвращается результат [compute].
*/
fun computeIfAbsent(program: ConfiguredShaderProgram, zLevel: Int, compute: () -> AbstractVertexBuilder<*>): AbstractVertexBuilder<*> {
fun computeIfAbsent(program: ConfiguredShaderProgram, zLevel: Int, compute: () -> VertexBuilder): VertexBuilder {
return layers.computeIfAbsent(program) { Int2ObjectAVLTreeMap() }.computeIfAbsent(zLevel, Int2ObjectFunction {
return@Int2ObjectFunction TileLayer(program, compute.invoke(), zLevel)
}).vertexBuilder
@ -169,7 +169,7 @@ private enum class TileRenderTesselateResult {
HALT
}
private fun vertexTextureBuilder() = HeapVertexBuilder(GLAttributeList.TILE, GeometryType.QUADS)
private fun vertexTextureBuilder() = VertexBuilder(GLAttributeList.TILE, GeometryType.QUADS)
private class TileEqualityTester(val definition: TileDefinition) : EqualityRuleTester {
override fun test(thisTile: ITileState, otherTile: ITileState): Boolean {
@ -199,7 +199,7 @@ class TileRenderer(val renderers: TileRenderers, val def: IRenderableTile) {
val bakedBackgroundProgramState = renderers.background(texture)
// private var notifiedDepth = false
private fun tesselateAt(self: ITileState, piece: RenderPiece, getter: ITileChunk, builder: AbstractVertexBuilder<*>, pos: Vector2i, offset: Vector2i = Vector2i.ZERO, isModifier: Boolean) {
private fun tesselateAt(self: ITileState, piece: RenderPiece, getter: ITileChunk, builder: VertexBuilder, pos: Vector2i, offset: Vector2i = Vector2i.ZERO, isModifier: Boolean) {
val fx = pos.x.toFloat()
val fy = pos.y.toFloat()
@ -246,7 +246,7 @@ class TileRenderer(val renderers: TileRenderers, val def: IRenderableTile) {
getter: ITileChunk,
layers: TileLayerList,
pos: Vector2i,
thisBuilder: AbstractVertexBuilder<*>,
thisBuilder: VertexBuilder,
background: Boolean,
isModifier: Boolean,
): TileRenderTesselateResult {

View File

@ -3,8 +3,6 @@ package ru.dbotthepony.kstarbound.client.render.entity
import ru.dbotthepony.kstarbound.PIXELS_IN_STARBOUND_UNITf
import ru.dbotthepony.kstarbound.client.ClientChunk
import ru.dbotthepony.kstarbound.client.gl.GLStateTracker
import ru.dbotthepony.kstarbound.client.gl.vertex.QuadTransformers
import ru.dbotthepony.kstarbound.client.gl.vertex.quadRotatedZ
import ru.dbotthepony.kstarbound.client.render.bind
import ru.dbotthepony.kstarbound.world.entities.ItemEntity
import ru.dbotthepony.kvector.matrix.Matrix4fStack
@ -25,7 +23,7 @@ class ItemRenderer(state: GLStateTracker, entity: ItemEntity, chunk: ClientChunk
for (texture in textures) {
texture.bind()
state.flat2DTexturedQuads.small.singleSprite(texture.width / PIXELS_IN_STARBOUND_UNITf, texture.height / PIXELS_IN_STARBOUND_UNITf, entity.movement.angle, texture.transformer)
state.flat2DTexturedQuads.singleSprite(texture.width / PIXELS_IN_STARBOUND_UNITf, texture.height / PIXELS_IN_STARBOUND_UNITf, entity.movement.angle, texture.transformer)
}
}
}

View File

@ -1,27 +0,0 @@
package ru.dbotthepony.kstarbound.util
import java.io.OutputStream
import java.nio.ByteBuffer
import java.nio.ByteOrder
class ByteBufferOutputStream(val buffer: ByteBuffer) : OutputStream() {
override fun write(b: Int) {
buffer.put(b.toByte())
}
override fun write(b: ByteArray) {
buffer.put(b)
}
override fun write(b: ByteArray, off: Int, len: Int) {
buffer.put(b, off, len)
}
var position: Int
get() = buffer.position()
set(value) { buffer.position(value) }
companion object {
fun directLE(capacity: Int) = ByteBufferOutputStream(ByteBuffer.allocateDirect(capacity).also { it.order(ByteOrder.LITTLE_ENDIAN) })
}
}