2022-03-25 17:31:05 +01:00
|
|
|
// Copyright 2022 The Ebiten Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package hlsl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/shaderir"
|
|
|
|
)
|
|
|
|
|
2024-11-24 10:30:34 +01:00
|
|
|
const boundaryInDWords = 4
|
2022-03-25 17:31:05 +01:00
|
|
|
|
2024-11-24 10:30:34 +01:00
|
|
|
func CalcUniformMemoryOffsetsInDWords(program *shaderir.Program) []int {
|
2022-03-25 17:31:05 +01:00
|
|
|
// https://docs.microsoft.com/en-us/windows/win32/direct3dhlsl/dx-graphics-hlsl-packing-rules
|
|
|
|
// https://github.com/microsoft/DirectXShaderCompiler/wiki/Buffer-Packing
|
|
|
|
|
2024-11-24 10:30:34 +01:00
|
|
|
var offsetsInDWords []int
|
|
|
|
var headInDWords int
|
2022-03-25 17:31:05 +01:00
|
|
|
|
|
|
|
align := func(x int) int {
|
|
|
|
if x == 0 {
|
|
|
|
return 0
|
|
|
|
}
|
2024-11-24 10:30:34 +01:00
|
|
|
return ((x-1)/boundaryInDWords + 1) * boundaryInDWords
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Reorder the variables with packoffset.
|
|
|
|
// See https://docs.microsoft.com/en-us/windows/win32/direct3dhlsl/dx-graphics-hlsl-variable-packoffset
|
2024-05-05 13:47:35 +02:00
|
|
|
for _, u := range program.Uniforms {
|
2022-03-25 17:31:05 +01:00
|
|
|
switch u.Main {
|
|
|
|
case shaderir.Float:
|
2024-11-24 10:30:34 +01:00
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 1
|
2022-11-12 10:38:15 +01:00
|
|
|
case shaderir.Int:
|
2024-11-24 10:30:34 +01:00
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 1
|
2022-11-20 07:36:07 +01:00
|
|
|
case shaderir.Vec2, shaderir.IVec2:
|
2024-11-24 10:30:34 +01:00
|
|
|
if headInDWords%boundaryInDWords >= 3 {
|
|
|
|
headInDWords = align(headInDWords)
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|
2024-11-24 10:30:34 +01:00
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 2
|
2022-11-20 07:36:07 +01:00
|
|
|
case shaderir.Vec3, shaderir.IVec3:
|
2024-11-24 10:30:34 +01:00
|
|
|
if headInDWords%boundaryInDWords >= 2 {
|
|
|
|
headInDWords = align(headInDWords)
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|
2024-11-24 10:30:34 +01:00
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 3
|
2022-11-20 07:36:07 +01:00
|
|
|
case shaderir.Vec4, shaderir.IVec4:
|
2024-11-24 10:30:34 +01:00
|
|
|
if headInDWords%boundaryInDWords >= 1 {
|
|
|
|
headInDWords = align(headInDWords)
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|
2024-11-24 10:30:34 +01:00
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 4
|
2022-03-25 17:31:05 +01:00
|
|
|
case shaderir.Mat2:
|
|
|
|
// For matrices, each column is aligned to the boundary.
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords = align(headInDWords)
|
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 6
|
2022-03-25 17:31:05 +01:00
|
|
|
case shaderir.Mat3:
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords = align(headInDWords)
|
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 11
|
2022-03-25 17:31:05 +01:00
|
|
|
case shaderir.Mat4:
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords = align(headInDWords)
|
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
|
|
|
headInDWords += 16
|
2022-03-25 17:31:05 +01:00
|
|
|
case shaderir.Array:
|
|
|
|
// Each array is 16-byte aligned.
|
|
|
|
// TODO: What if the array has 2 or more dimensions?
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords = align(headInDWords)
|
|
|
|
offsetsInDWords = append(offsetsInDWords, headInDWords)
|
2022-11-12 10:38:15 +01:00
|
|
|
n := u.Sub[0].Uint32Count()
|
2022-03-26 17:49:50 +01:00
|
|
|
switch u.Sub[0].Main {
|
|
|
|
case shaderir.Mat2:
|
|
|
|
n = 6
|
|
|
|
case shaderir.Mat3:
|
|
|
|
n = 11
|
|
|
|
case shaderir.Mat4:
|
|
|
|
n = 16
|
|
|
|
}
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords += (u.Length - 1) * align(n)
|
2022-03-25 17:31:05 +01:00
|
|
|
// The last element is not with a padding.
|
2024-11-24 10:30:34 +01:00
|
|
|
headInDWords += n
|
2022-03-25 17:31:05 +01:00
|
|
|
case shaderir.Struct:
|
|
|
|
// TODO: Implement this
|
|
|
|
panic("hlsl: offset for a struct is not implemented yet")
|
|
|
|
default:
|
2022-11-12 10:38:15 +01:00
|
|
|
panic(fmt.Sprintf("hlsl: unexpected type: %s", u.String()))
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-24 10:30:34 +01:00
|
|
|
return offsetsInDWords
|
2022-03-25 17:31:05 +01:00
|
|
|
}
|