Kernels & Functions¶
约 950 个字 98 行代码 1 张图片 预计阅读时间 4 分钟
修饰器 ¶
在 Taichi 中有两个重要的修饰器,ti.func
和ti.kernel
,分别代表 Taichi function 和 Taichi kernel,Taichi function 和 Taichi kernel 不由 Python 的解释器执行,而是由 Taichi 的 JIT 编译器执行计算,并部署到并行的多核 CPU 或 GPU 上
Tip
如果熟悉 CUDA 编程的话,ti.func
可以等效为 CUDA 中的__device__
,ti.kernel
可以等效为 CUDA 中的__global__
Example
Taichi kernel¶
Taichi kernel 是 Taichi 接管执行任务的入口,可以在程序的任何地方被调用
参数 ¶
一个 Taichi kernel 可以接受多个参数,但不意味着任意 Python 对象都可以传给 kernel,因为 Python 对象可以是动态的,并且可能包含 Taichi 的编译器无法识别的类型
Taichi kernel 支持的类型有这几种:scalar
, ti.types.matrix()
, ti.types.vector()
, ti.types.struct()
, ti.types.ndarray()
, 和ti.template()
其中scalar
, ti.types.matrix()
, ti.types.vector()
, 和ti.types.struct()
是值传递 , kernel 会接收参数的拷贝值;而ti.types.ndarray()
和ti.template()
是传引用值 , 在 kernel 中修改会影响原来的数据
可以使用ti.types.ndarray()
作为类型提示,将 NumPy 中的 ndarray 或 PyTorch 中的 tensor 传给 kernel,传的是引用值
Example
import numpy as np
import taichi as ti
ti.init(arch=ti.cpu)
x = np.array([1, 2, 3])
y = np.array([4, 5, 6])
@ti.kernel
def my_kernel(x: ti.types.ndarray(), y: ti.types.ndarray()):
# Taichi recognizes the shape of the array x and allows you to access it in a kernel
for i in range(x.shape[0]):
x[i] += y[i]
my_kernel(x, y)
print(x) # Prints [5, 7, 9]
返回值 ¶
Taichi kernel 最多允许有一个返回值,返回值类型可以是scalar
, ti.types.matrix()
, 或ti.types.vector()
,如果使用 LLVM-based 的后端(CPU/CUDAti.types.struct()
也可以使用
概括来说 Taichi kernel 的返回值,必须符合以下规则:
- kernel 的返回值必须声明类型
-
kernel 最多允许一个返回值
-
kernel 最多允许一个
return
语句 - 如果 kernel 的返回值是 vector 或 matrix,它不能包含超过 32 个元素,如果包含超过 32 个元素,kernel 仍将被编译,但会出现 warning
全局变量 ¶
在 Taichi 中,kernel 将全局变量视为编译时常量 (compile-time constants),这表示 kernel 在编译时会获取全局变量的当前值,并且之后不会跟踪它们的更改
Example
Taichi kernel 的要求总结如下:
- kernel 函数的参数必须有显式的 type annotation
- 不能在其他 kernel 或者 Taichi function 中调用,只能在 Python-scope 中调用
- 只能有一个 return 语句
- 只能返回一个变量(不像 python 可以 pack 成一个 tuple)
- 返回变量的元素个数不要超过 30(?)
- kernel 函数的输入变量有大小限制
- kernel 函数会将全局变量视为常量(不当指针)
Taichi function¶
Taichi function 是 kernel 的基本单位,只能从 Taichi kernel 或另一个 Taichi function 中被调用
所有的 Taichi function 是强制内联 (force-inlined)的,因此在被 Taichi function 调用是会完全展开,所以 Taichi function 的递归调用是不允许的
参数 ¶
大部分的类型都是支持的,和 Taichi kernel 的参数相比,以下几点需要注意:
- 不严格要求指定参数类型 ( 但仍然建议这样做 )
- 对参数的数量没有限制
返回值 ¶
大部分的类型都是支持的,和 Taichi kernel 的返回值相比,以下几点需要注意:
- 可以有多个返回值
- 不严格要求指定返回值类型 ( 但仍然建议这样做 )
- 最多只能有一个
return
语句
Taichi function 和 kernel 的区别 ¶
Taichi function 和 kernel 的区别如下表所示:
循环的并行运算 ¶
Taichi kernel 里最外层的任何 for 循环都会自动并行化
@ti.kernel
def fill():
total = 0
for i in range(10): # Parallelized
for j in range(5): # Serialized in each parallel thread
total += i * j
if total > 10:
for k in range(5): # Not parallelized because it is not at the outermost scope
Warning
并行计算的循环中不支持break
语句
通过ti.loop_config(serialize=True)
的设置可以禁用自动并行化
Taichi 的循环还有一个很好用的特性,使用ti.grouped(x)
将会把x
作为一个 1D 数组来遍历它的元素,而不管它的形状,从而避免了写多级循环,下面是一个例子 :
Example
import taichi as ti
ti.init(arch=ti.cpu)
x = ti.field(ti.f32, shape=(2, 2))
@ti.kernel
def loop_test():
count = 0
for i in ti.grouped(x):
print(i)
count += 1
print(f"all count: {count}")
if __name__ == "__main__":
loop_test()
输出结果如下:
ti.grouped(x) 不可以在 Python-scope 使用,只能在 Taichi-scope 使用