From 1b01cba81447d77268abbb5849297bad37402138 Mon Sep 17 00:00:00 2001
From: faiface <faiface@ksp.sk>
Date: Wed, 14 Dec 2016 16:24:31 +0100
Subject: [PATCH] replace reentrancy by enabled by more general binder

---
 pixelgl/shader.go  | 312 ++++++++++++---------------------------------
 pixelgl/texture.go |  29 ++---
 pixelgl/util.go    |  27 ++++
 pixelgl/vertex.go  |  76 +++++------
 4 files changed, 163 insertions(+), 281 deletions(-)
 create mode 100644 pixelgl/util.go

diff --git a/pixelgl/shader.go b/pixelgl/shader.go
index 533c564..42bf413 100644
--- a/pixelgl/shader.go
+++ b/pixelgl/shader.go
@@ -16,9 +16,8 @@ type UniformFormat map[string]Attr
 
 // Shader is an OpenGL shader program.
 type Shader struct {
-	enabled       bool
 	parent        Doer
-	program       uint32
+	program       binder
 	vertexFormat  VertexFormat
 	uniformFormat UniformFormat
 	uniforms      map[Attr]int32
@@ -29,7 +28,13 @@ type Shader struct {
 // Note that vertexShader and fragmentShader parameters must contain the source code, they're not filenames.
 func NewShader(parent Doer, vertexFormat VertexFormat, uniformFormat UniformFormat, vertexShader, fragmentShader string) (*Shader, error) {
 	shader := &Shader{
-		parent:        parent,
+		parent: parent,
+		program: binder{
+			restoreLoc: gl.CURRENT_PROGRAM,
+			bindFunc: func(obj uint32) {
+				gl.UseProgram(obj)
+			},
+		},
 		vertexFormat:  vertexFormat,
 		uniformFormat: uniformFormat,
 		uniforms:      make(map[Attr]int32),
@@ -86,31 +91,31 @@ func NewShader(parent Doer, vertexFormat VertexFormat, uniformFormat UniformForm
 
 			// shader program
 			{
-				shader.program = gl.CreateProgram()
-				gl.AttachShader(shader.program, vshader)
-				gl.AttachShader(shader.program, fshader)
-				gl.LinkProgram(shader.program)
+				shader.program.obj = gl.CreateProgram()
+				gl.AttachShader(shader.program.obj, vshader)
+				gl.AttachShader(shader.program.obj, fshader)
+				gl.LinkProgram(shader.program.obj)
 
 				var (
 					success int32
 					infoLog = make([]byte, 512)
 				)
-				gl.GetProgramiv(shader.program, gl.LINK_STATUS, &success)
+				gl.GetProgramiv(shader.program.obj, gl.LINK_STATUS, &success)
 				if success == 0 {
-					gl.GetProgramInfoLog(shader.program, int32(len(infoLog)), nil, &infoLog[0])
+					gl.GetProgramInfoLog(shader.program.obj, int32(len(infoLog)), nil, &infoLog[0])
 					return fmt.Errorf("error linking shader program: %s", string(infoLog))
 				}
 			}
 
 			// uniforms
 			for uname, utype := range uniformFormat {
-				ulocation := gl.GetUniformLocation(shader.program, gl.Str(uname+"\x00"))
+				ulocation := gl.GetUniformLocation(shader.program.obj, gl.Str(uname+"\x00"))
 				if ulocation == -1 {
-					gl.DeleteProgram(shader.program)
+					gl.DeleteProgram(shader.program.obj)
 					return fmt.Errorf("shader does not contain uniform '%s'", uname)
 				}
 				if _, ok := shader.uniforms[utype]; ok {
-					gl.DeleteProgram(shader.program)
+					gl.DeleteProgram(shader.program.obj)
 					return fmt.Errorf("failed to create shader: invalid uniform format: duplicate uniform attribute")
 				}
 				shader.uniforms[utype] = ulocation
@@ -130,14 +135,14 @@ func NewShader(parent Doer, vertexFormat VertexFormat, uniformFormat UniformForm
 func (s *Shader) Delete() {
 	s.parent.Do(func(ctx Context) {
 		DoNoBlock(func() {
-			gl.DeleteProgram(s.program)
+			gl.DeleteProgram(s.program.obj)
 		})
 	})
 }
 
 // ID returns an OpenGL identifier of a shader program.
 func (s *Shader) ID() uint32 {
-	return s.program
+	return s.program.obj
 }
 
 // VertexFormat returns the vertex attribute format of this shader. Do not change it.
@@ -150,245 +155,94 @@ func (s *Shader) UniformFormat() UniformFormat {
 	return s.uniformFormat
 }
 
-// SetUniformInt sets the value of an uniform attribute Attr{Purpose: purpose, Type: Int}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformInt(purpose AttrPurpose, value int32) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Int}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
-			gl.Uniform1i(s.uniforms[attr], value)
-		})
-	})
-	return true
-}
-
-// SetUniformFloat sets the value of an uniform attribute Attr{Purpose: purpose, Type: Float}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformFloat(purpose AttrPurpose, value float32) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Float}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
-			gl.Uniform1f(s.uniforms[attr], value)
-		})
-	})
-	return true
-}
-
-// SetUniformVec2 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Vec2}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformVec2(purpose AttrPurpose, value mgl32.Vec2) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Vec2}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
-			gl.Uniform2f(s.uniforms[attr], value[0], value[1])
-		})
-	})
-	return true
-}
-
-// SetUniformVec3 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Vec3}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformVec3(purpose AttrPurpose, value mgl32.Vec3) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Vec3}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
-			gl.Uniform3f(s.uniforms[attr], value[0], value[1], value[2])
-		})
-	})
-	return true
-}
-
-// SetUniformVec4 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Vec4}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformVec4(purpose AttrPurpose, value mgl32.Vec4) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Vec4}
+// SetUniformAttr sets the value of a uniform attribute of a shader.
+//
+// If the attribute does not exist, this method returns false.
+//
+// Supplied value must correspond to the type of the attribute. Correct types are these (right-hand is the type of the value):
+//   Attr{Type: Int}:   int32
+//   Attr{Type: Float}: float32
+//   Attr{Type: Vec2}:  mgl32.Vec2
+//   Attr{Type: Vec3}:  mgl32.Vec3
+//   Attr{Type: Vec4}:  mgl32.Vec4
+//   Attr{Type: Mat2}:  mgl32.Mat2
+//   Attr{Type: Mat23}: mgl32.Mat2x3
+//   Attr{Type: Mat24}: mgl32.Mat2x4
+//   Attr{Type: Mat3}:  mgl32.Mat3
+//   Attr{Type: Mat32}: mgl32.Mat3x2
+//   Attr{Type: Mat34}: mgl32.Mat3x4
+//   Attr{Type: Mat4}:  mgl32.Mat4
+//   Attr{Type: Mat42}: mgl32.Mat4x2
+//   Attr{Type: Mat43}: mgl32.Mat4x3
+// No other types are supported.
+func (s *Shader) SetUniformAttr(attr Attr, value interface{}) (ok bool) {
 	if _, ok := s.uniforms[attr]; !ok {
 		return false
 	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
-			gl.Uniform4f(s.uniforms[attr], value[0], value[1], value[2], value[3])
-		})
-	})
-	return true
-}
 
-// SetUniformMat2 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat2}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat2(purpose AttrPurpose, value mgl32.Mat2) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat2}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+	DoNoBlock(func() {
+		defer s.program.bind().restore()
+
+		switch attr.Type {
+		case Int:
+			value := value.(int32)
+			gl.Uniform1iv(s.uniforms[attr], 1, &value)
+		case Float:
+			value := value.(float32)
+			gl.Uniform1fv(s.uniforms[attr], 1, &value)
+		case Vec2:
+			value := value.(mgl32.Vec2)
+			gl.Uniform2fv(s.uniforms[attr], 1, &value[0])
+		case Vec3:
+			value := value.(mgl32.Vec3)
+			gl.Uniform3fv(s.uniforms[attr], 1, &value[0])
+		case Vec4:
+			value := value.(mgl32.Vec4)
+			gl.Uniform4fv(s.uniforms[attr], 1, &value[0])
+		case Mat2:
+			value := value.(mgl32.Mat2)
 			gl.UniformMatrix2fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat23 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat23}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat23(purpose AttrPurpose, value mgl32.Mat2x3) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat23}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat23:
+			value := value.(mgl32.Mat2x3)
 			gl.UniformMatrix2x3fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat24 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat24}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat24(purpose AttrPurpose, value mgl32.Mat2x4) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat24}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat24:
+			value := value.(mgl32.Mat2x4)
 			gl.UniformMatrix2x4fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat3 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat3}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat3(purpose AttrPurpose, value mgl32.Mat3) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat3}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat3:
+			value := value.(mgl32.Mat3)
 			gl.UniformMatrix3fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat32 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat32}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat32(purpose AttrPurpose, value mgl32.Mat3x2) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat32}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat32:
+			value := value.(mgl32.Mat3x2)
 			gl.UniformMatrix3x2fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat34 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat34}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat34(purpose AttrPurpose, value mgl32.Mat3x4) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat34}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat34:
+			value := value.(mgl32.Mat3x4)
 			gl.UniformMatrix3x4fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat4 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat4}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat4(purpose AttrPurpose, value mgl32.Mat4) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat4}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat4:
+			value := value.(mgl32.Mat4)
 			gl.UniformMatrix4fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat42 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat42}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat42(purpose AttrPurpose, value mgl32.Mat4x2) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat42}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat42:
+			value := value.(mgl32.Mat4x2)
 			gl.UniformMatrix4x2fv(s.uniforms[attr], 1, false, &value[0])
-		})
-	})
-	return true
-}
-
-// SetUniformMat43 sets the value of an uniform attribute Attr{Purpose: purpose, Type: Mat43}.
-//
-// Returns false if the attribute does not exist.
-func (s *Shader) SetUniformMat43(purpose AttrPurpose, value mgl32.Mat4x3) (ok bool) {
-	attr := Attr{Purpose: purpose, Type: Mat43}
-	if _, ok := s.uniforms[attr]; !ok {
-		return false
-	}
-	s.Do(func(Context) {
-		DoNoBlock(func() {
+		case Mat43:
+			value := value.(mgl32.Mat4x3)
 			gl.UniformMatrix4x3fv(s.uniforms[attr], 1, false, &value[0])
-		})
+		default:
+			panic("set uniform attr: invalid attribute type")
+		}
 	})
+
 	return true
 }
 
 // Do stars using a shader, executes sub, and stops using it.
 func (s *Shader) Do(sub func(Context)) {
 	s.parent.Do(func(ctx Context) {
-		if s.enabled {
-			sub(ctx.WithShader(s))
-			return
-		}
 		DoNoBlock(func() {
-			gl.UseProgram(s.program)
+			s.program.bind()
 		})
-		s.enabled = true
 		sub(ctx.WithShader(s))
-		s.enabled = false
 		DoNoBlock(func() {
-			gl.UseProgram(0)
+			s.program.restore()
 		})
 	})
 }
diff --git a/pixelgl/texture.go b/pixelgl/texture.go
index ad9a919..f443081 100644
--- a/pixelgl/texture.go
+++ b/pixelgl/texture.go
@@ -4,9 +4,8 @@ import "github.com/go-gl/gl/v3.3-core/gl"
 
 // Texture is an OpenGL texture.
 type Texture struct {
-	enabled       bool
 	parent        Doer
-	tex           uint32
+	tex           binder
 	width, height int
 }
 
@@ -15,14 +14,20 @@ type Texture struct {
 func NewTexture(parent Doer, width, height int, pixels []uint8) (*Texture, error) {
 	texture := &Texture{
 		parent: parent,
+		tex: binder{
+			restoreLoc: gl.TEXTURE_BINDING_2D,
+			bindFunc: func(obj uint32) {
+				gl.BindTexture(gl.TEXTURE_2D, obj)
+			},
+		},
 		width:  width,
 		height: height,
 	}
 
 	parent.Do(func(ctx Context) {
 		Do(func() {
-			gl.GenTextures(1, &texture.tex)
-			gl.BindTexture(gl.TEXTURE_2D, texture.tex)
+			gl.GenTextures(1, &texture.tex.obj)
+			defer texture.tex.bind().restore()
 
 			gl.TexImage2D(
 				gl.TEXTURE_2D,
@@ -37,8 +42,6 @@ func NewTexture(parent Doer, width, height int, pixels []uint8) (*Texture, error
 			)
 
 			gl.GenerateMipmap(gl.TEXTURE_2D)
-
-			gl.BindTexture(gl.TEXTURE_2D, 0)
 		})
 	})
 
@@ -49,14 +52,14 @@ func NewTexture(parent Doer, width, height int, pixels []uint8) (*Texture, error
 func (t *Texture) Delete() {
 	t.parent.Do(func(ctx Context) {
 		DoNoBlock(func() {
-			gl.DeleteTextures(1, &t.tex)
+			gl.DeleteTextures(1, &t.tex.obj)
 		})
 	})
 }
 
 // ID returns an OpenGL identifier of a texture.
 func (t *Texture) ID() uint32 {
-	return t.tex
+	return t.tex.obj
 }
 
 // Width returns the width of a texture in pixels.
@@ -72,18 +75,12 @@ func (t *Texture) Height() int {
 // Do bind a texture, executes sub, and unbinds the texture.
 func (t *Texture) Do(sub func(Context)) {
 	t.parent.Do(func(ctx Context) {
-		if t.enabled {
-			sub(ctx)
-			return
-		}
 		DoNoBlock(func() {
-			gl.BindTexture(gl.TEXTURE_2D, t.tex)
+			t.tex.bind()
 		})
-		t.enabled = true
 		sub(ctx)
-		t.enabled = false
 		DoNoBlock(func() {
-			gl.BindTexture(gl.TEXTURE_2D, 0)
+			t.tex.restore()
 		})
 	})
 }
diff --git a/pixelgl/util.go b/pixelgl/util.go
new file mode 100644
index 0000000..b74eda2
--- /dev/null
+++ b/pixelgl/util.go
@@ -0,0 +1,27 @@
+package pixelgl
+
+import "github.com/go-gl/gl/v3.3-core/gl"
+
+type binder struct {
+	restoreLoc uint32
+	bindFunc   func(uint32)
+
+	obj uint32
+
+	prev []uint32
+}
+
+func (b *binder) bind() *binder {
+	var prev int32
+	gl.GetIntegerv(b.restoreLoc, &prev)
+	b.prev = append(b.prev, uint32(prev))
+
+	b.bindFunc(b.obj)
+	return b
+}
+
+func (b *binder) restore() *binder {
+	b.bindFunc(b.prev[len(b.prev)-1])
+	b.prev = b.prev[:len(b.prev)-1]
+	return b
+}
diff --git a/pixelgl/vertex.go b/pixelgl/vertex.go
index 64c4d6e..19bf68a 100644
--- a/pixelgl/vertex.go
+++ b/pixelgl/vertex.go
@@ -43,9 +43,8 @@ const (
 // VertexArray is an OpenGL vertex array object that also holds it's own vertex buffer object.
 // From the user's points of view, VertexArray is an array of vertices that can be drawn.
 type VertexArray struct {
-	enabled             bool
 	parent              Doer
-	vao, vbo, ebo       uint32
+	vao, vbo, ebo       binder
 	vertexNum, indexNum int
 	format              VertexFormat
 	usage               VertexUsage
@@ -59,10 +58,28 @@ type VertexArray struct {
 // set the vertex attributes. Use indices to specify how you want to combine vertices into triangles.
 func NewVertexArray(parent Doer, format VertexFormat, usage VertexUsage, vertexNum int, indices []int) (*VertexArray, error) {
 	va := &VertexArray{
-		parent:    parent,
+		parent: parent,
+		vao: binder{
+			restoreLoc: gl.VERTEX_ARRAY_BINDING,
+			bindFunc: func(obj uint32) {
+				gl.BindVertexArray(obj)
+			},
+		},
+		vbo: binder{
+			restoreLoc: gl.ARRAY_BUFFER_BINDING,
+			bindFunc: func(obj uint32) {
+				gl.BindBuffer(gl.ARRAY_BUFFER, obj)
+			},
+		},
+		ebo: binder{
+			restoreLoc: gl.ELEMENT_ARRAY_BUFFER_BINDING,
+			bindFunc: func(obj uint32) {
+				gl.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, obj)
+			},
+		},
+		vertexNum: vertexNum,
 		format:    format,
 		usage:     usage,
-		vertexNum: vertexNum,
 		stride:    format.Size(),
 		attrs:     make(map[Attr]int),
 	}
@@ -83,16 +100,17 @@ func NewVertexArray(parent Doer, format VertexFormat, usage VertexUsage, vertexN
 
 	parent.Do(func(ctx Context) {
 		Do(func() {
-			gl.GenVertexArrays(1, &va.vao)
-			gl.BindVertexArray(va.vao)
+			gl.GenVertexArrays(1, &va.vao.obj)
+			va.vao.bind()
 
-			gl.GenBuffers(1, &va.vbo)
-			gl.BindBuffer(gl.ARRAY_BUFFER, va.vbo)
+			gl.GenBuffers(1, &va.vbo.obj)
+			defer va.vbo.bind().restore()
 
 			emptyData := make([]byte, vertexNum*va.stride)
 			gl.BufferData(gl.ARRAY_BUFFER, len(emptyData), gl.Ptr(emptyData), uint32(usage))
 
-			gl.GenBuffers(1, &va.ebo)
+			gl.GenBuffers(1, &va.ebo.obj)
+			defer va.ebo.bind().restore()
 
 			offset := 0
 			for i, attr := range format {
@@ -120,12 +138,7 @@ func NewVertexArray(parent Doer, format VertexFormat, usage VertexUsage, vertexN
 				offset += attr.Type.Size()
 			}
 
-			gl.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, va.ebo) // need to bind EBO, so that VAO registers it
-
-			gl.BindBuffer(gl.ARRAY_BUFFER, 0)
-			gl.BindVertexArray(0)
-
-			gl.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, 0)
+			va.vao.restore()
 		})
 	})
 
@@ -138,15 +151,15 @@ func NewVertexArray(parent Doer, format VertexFormat, usage VertexUsage, vertexN
 func (va *VertexArray) Delete() {
 	va.parent.Do(func(ctx Context) {
 		DoNoBlock(func() {
-			gl.DeleteVertexArrays(1, &va.vao)
-			gl.DeleteBuffers(1, &va.vbo)
+			gl.DeleteVertexArrays(1, &va.vao.obj)
+			gl.DeleteBuffers(1, &va.vbo.obj)
 		})
 	})
 }
 
 // ID returns an OpenGL identifier of a vertex array.
 func (va *VertexArray) ID() uint32 {
-	return va.vao
+	return va.vao.obj
 }
 
 // VertexNum returns the number of vertices in a vertex array.
@@ -184,9 +197,8 @@ func (va *VertexArray) SetIndices(indices []int) {
 	}
 	va.indexNum = len(indices32)
 	DoNoBlock(func() {
-		gl.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, va.ebo)
+		defer va.ebo.bind().restore()
 		gl.BufferData(gl.ELEMENT_ARRAY_BUFFER, 4*len(indices32), gl.Ptr(indices32), uint32(va.usage))
-		gl.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, 0)
 	})
 }
 
@@ -195,7 +207,7 @@ func (va *VertexArray) SetIndices(indices []int) {
 // If the vertex attribute does not exist, this method returns false. If the vertex is out of range,
 // this method panics.
 //
-// Supplied value must correspond to the type of the attribute. Correct types are these (righ-hand is the type of value):
+// Supplied value must correspond to the type of the attribute. Correct types are these (righ-hand is the type of the value):
 //   Attr{Type: Float}: float32
 //   Attr{Type: Vec2}:  mgl32.Vec2
 //   Attr{Type: Vec3}:  mgl32.Vec3
@@ -211,7 +223,7 @@ func (va *VertexArray) SetVertexAttr(vertex int, attr Attr, value interface{}) (
 	}
 
 	DoNoBlock(func() {
-		gl.BindBuffer(gl.ARRAY_BUFFER, va.vbo)
+		defer va.vbo.bind().restore()
 
 		offset := va.stride*vertex + va.attrs[attr]
 
@@ -228,9 +240,9 @@ func (va *VertexArray) SetVertexAttr(vertex int, attr Attr, value interface{}) (
 		case Vec4:
 			value := value.(mgl32.Vec4)
 			gl.BufferSubData(gl.ARRAY_BUFFER, offset, attr.Type.Size(), unsafe.Pointer(&value))
+		default:
+			panic("set vertex attr: invalid attribute type")
 		}
-
-		gl.BindBuffer(gl.ARRAY_BUFFER, 0)
 	})
 
 	return true
@@ -251,8 +263,8 @@ func (va *VertexArray) VertexAttr(vertex int, attr Attr) (value interface{}, ok
 		return nil, false
 	}
 
-	DoNoBlock(func() {
-		gl.BindBuffer(gl.ARRAY_BUFFER, va.vbo)
+	Do(func() {
+		defer va.vbo.bind().restore()
 
 		offset := va.stride*vertex + va.attrs[attr]
 
@@ -274,8 +286,6 @@ func (va *VertexArray) VertexAttr(vertex int, attr Attr) (value interface{}, ok
 			gl.GetBufferSubData(gl.ARRAY_BUFFER, offset, attr.Type.Size(), unsafe.Pointer(&data))
 			value = data
 		}
-
-		gl.BindBuffer(gl.ARRAY_BUFFER, 0)
 	})
 
 	return value, true
@@ -284,19 +294,13 @@ func (va *VertexArray) VertexAttr(vertex int, attr Attr) (value interface{}, ok
 // Do binds a vertex arrray and it's associated vertex buffer, executes sub, and unbinds the vertex array and it's vertex buffer.
 func (va *VertexArray) Do(sub func(Context)) {
 	va.parent.Do(func(ctx Context) {
-		if va.enabled {
-			sub(ctx)
-			return
-		}
 		DoNoBlock(func() {
-			gl.BindVertexArray(va.vao)
+			va.vao.bind()
 		})
-		va.enabled = true
 		sub(ctx)
-		va.enabled = false
 		DoNoBlock(func() {
 			gl.DrawElements(gl.TRIANGLES, int32(va.indexNum), gl.UNSIGNED_INT, gl.PtrOffset(0))
-			gl.BindVertexArray(0)
+			va.vao.restore()
 		})
 	})
 }
-- 
GitLab