-
Notifications
You must be signed in to change notification settings - Fork 70
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
How can we make KA fast on CPUs? #509
Comments
Does #436 speed things up for you? I haven't merged it since I haven't seen an impact on benchmarks. |
Let me run the benchmarks with that branch and check. EDIT: doesn't seem to help |
So the question would be where are the overheards coming from, so maybe you cana run a profile and compare the time being spend. You could also use static scheduling. Could you isolate a particular case where you are seeing these overheads? |
A MWE would be very useful, as the upcoming POCL CPU back-end may be interesting for performance, but hasn't been benchmarked. |
Oops this fell off my radar, I will create a self contained example this week |
(Independent of this) once POCL is ready and I can access a ci server with that, I can trigger the benchmark suite for NN primitives in LuxLib to generate results in https://luxdl.github.io/LuxLib.jl/benchmarks/ |
Good to know. OpenCL.jl's KA.jl integration is actually functional already, see JuliaGPU/OpenCL.jl#233, but not tagged yet. If you're feeling adventurous, you can test it out already: Just I expect issues with real applications though once testing this out, as the KA.jl and GPUArrays.jl test suites don't cover e.g. all array constructors or expected interfaces. Such feedback would be valuable! |
A toned down version of the 2d batchnorm routine from LuxLib using Statistics, LinearAlgebra
using KernelAbstractions
using OpenCL, pocl_jll
using Test
function batchnorm_looped(x::AbstractMatrix, scale::AbstractVector, bias::AbstractVector, μ::AbstractVector, σ²::AbstractVector)
y = similar(x)
@inbounds for j in axes(x, 2)
@simd ivdep for i in axes(x, 1)
y[i, j] = tanh(scale[i] * (x[i, j] - μ[i]) / sqrt(σ²[i] + 1e-5) + bias[i])
end
end
return y
end
@kernel function batchnorm_kernel_act!(y, @Const(act), @Const(scale), @Const(bias), @Const(x), @Const(μ), @Const(σ²))
i, j = @index(Global, NTuple)
y[i, j] = act(scale[i] * (x[i, j] - μ[i]) / sqrt(σ²[i] + 1e-5) + bias[i])
end
function batchnorm_ka(x::AbstractMatrix, scale::AbstractVector, bias::AbstractVector, μ::AbstractVector, σ²::AbstractVector)
y = similar(x)
backend = KernelAbstractions.get_backend(x)
kernel! = batchnorm_kernel_act!(backend)
kernel!(y, tanh, scale, bias, x, μ, σ²; ndrange=size(x))
KernelAbstractions.synchronize(backend)
return y
end
N, B = 32, 32
x = randn(Float32, N, B);
scale = randn(Float32, N);
bias = randn(Float32, N);
μ = rand(Float32, N);
σ² = rand(Float32, N);
x_cl = CLArray(x);
scale_cl = CLArray(scale);
bias_cl = CLArray(bias);
μ_cl = CLArray(μ);
σ²_cl = CLArray(σ²);
@test batchnorm_ka(x, scale, bias, μ, σ²) ≈ batchnorm_looped(x, scale, bias, μ, σ²)
@benchmark batchnorm_looped($x, $scale, $bias, $μ, $σ²)
@benchmark batchnorm_ka($x, $scale, $bias, $μ, $σ²)
@benchmark batchnorm_ka($x_cl, $scale_cl, $bias_cl, $μ_cl, $σ²_cl) julia> @benchmark batchnorm_looped($x, $scale, $bias, $μ, $σ²)
BenchmarkTools.Trial: 10000 samples with 1 evaluation.
Range (min … max): 15.391 μs … 50.510 μs ┊ GC (min … max): 0.00% … 0.00%
Time (median): 15.511 μs ┊ GC (median): 0.00%
Time (mean ± σ): 15.606 μs ± 645.765 ns ┊ GC (mean ± σ): 0.00% ± 0.00%
▂▆▇██▇▇▆▆▅▅▄▄▃▃▁▂▁▂▁▁▁▁▁▁▁▁▁▁ ▁ ▃
▆██████████████████████████████████████▆▇▆▆▆▅▄▅▄▅▃▃▃▄▄▄▃▅▁▃▃ █
15.4 μs Histogram: log(frequency) by time 16.6 μs <
Memory estimate: 4.08 KiB, allocs estimate: 3.
julia> @benchmark batchnorm_ka($x, $scale, $bias, $μ, $σ²)
BenchmarkTools.Trial: 10000 samples with 1 evaluation.
Range (min … max): 164.321 μs … 15.296 ms ┊ GC (min … max): 0.00% … 98.61%
Time (median): 166.691 μs ┊ GC (median): 0.00%
Time (mean ± σ): 181.080 μs ± 218.323 μs ┊ GC (mean ± σ): 3.55% ± 3.91%
▂█▁
▃▆███▃▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▂▁▂▁▂▂▂▄▄▄▄▃▂▂▃▄▄▃▃▂▂▂▂▂▂▂▂▂▂▂▂ ▃
164 μs Histogram: frequency by time 200 μs <
Memory estimate: 52.41 KiB, allocs estimate: 3084.
julia> @benchmark batchnorm_ka($x_cl, $scale_cl, $bias_cl, $μ_cl, $σ²_cl)
BenchmarkTools.Trial: 10000 samples with 1 evaluation.
Range (min … max): 78.521 μs … 3.324 ms ┊ GC (min … max): 0.00% … 0.00%
Time (median): 402.148 μs ┊ GC (median): 0.00%
Time (mean ± σ): 400.169 μs ± 186.617 μs ┊ GC (mean ± σ): 0.00% ± 0.00%
▁▇█▇▇▅▅▃▃▃▂▃▇▅▄▅▂▃▂▄▃▄▃▄▃▇▃▂ █▇▄▃▂▃▂▂▃█▇▂▃▇▄▃▆▆▃▃▅▇▅▆▅▂▂▃▆▅▃
▃████████████████████████████████████████████████████████████ █
78.5 μs Histogram: frequency by time 707 μs <
Memory estimate: 6.45 KiB, allocs estimate: 96. |
That's not too bad, I guess? The minimal performance at least is somewhat decent, but I'd be curious where the huge spread comes from. We haven't done any optimization of OpenCL.jl, so there's probably lots of low-hanging fruit there. Additionally, the PoCL build is not optimized: Anything but X86 will perform bad, and we're currently using the default pthread CPU driver while there's also OpenMP/oneTBB-based drivers that are known to perform better. |
Digging a bit further, for the KA CPU backend, if I pull out the activation the performance is much better @kernel inbounds=true function batchnorm_kernel_act!(y, @Const(scale), @Const(bias), @Const(x), @Const(μ), @Const(σ²))
i, j = @index(Global, NTuple)
res = scale[i] * (x[i, j] - μ[i]) / sqrt(σ²[i] + 1e-5) + bias[i]
y[i, j] = res
end
function batchnorm_ka(x::AbstractMatrix, scale::AbstractVector, bias::AbstractVector, μ::AbstractVector, σ²::AbstractVector)
y = similar(x)
backend = KernelAbstractions.get_backend(x)
kernel! = batchnorm_kernel_act!(backend)
kernel!(y, scale, bias, x, μ, σ²; ndrange=size(x))
@. y = tanh(y)
KernelAbstractions.synchronize(backend)
return y
end julia> @benchmark batchnorm_looped($x, $scale, $bias, $μ, $σ²)
BenchmarkTools.Trial: 10000 samples with 1 evaluation.
Range (min … max): 15.380 μs … 53.000 μs ┊ GC (min … max): 0.00% … 0.00%
Time (median): 15.510 μs ┊ GC (median): 0.00%
Time (mean ± σ): 15.586 μs ± 646.284 ns ┊ GC (mean ± σ): 0.00% ± 0.00%
▂▁▃▄███▇▇▆▅▅▆▃▃▂▃▁▁ ▂▁▁ ▁ ▁ ▁ ▂
▇████████████████████████████▇▇█▆███▇█▇██▇▇██▇▇██▇▇█▇▆▇▇▇▆▅▅ █
15.4 μs Histogram: log(frequency) by time 16.1 μs <
Memory estimate: 4.08 KiB, allocs estimate: 3.
julia> @benchmark batchnorm_ka($x, $scale, $bias, $μ, $σ²)
BenchmarkTools.Trial: 10000 samples with 1 evaluation.
Range (min … max): 11.250 μs … 50.540 μs ┊ GC (min … max): 0.00% … 0.00%
Time (median): 11.430 μs ┊ GC (median): 0.00%
Time (mean ± σ): 11.493 μs ± 555.961 ns ┊ GC (mean ± σ): 0.00% ± 0.00%
█ ▃ ▂▇ ▁
▁▁▃▃▂▃▃▅██▆████▅▄█▄▅▃▃▄▂▂▃▂▂▂▂▂▂▂▂▂▁▁▂▁▂▁▁▂▁▁▁▁▁▁▁▁▁▁▁▁▁▁▁▁▁ ▂
11.2 μs Histogram: frequency by time 12.1 μs <
Memory estimate: 4.41 KiB, allocs estimate: 12. Makes opencl slightly slower julia> @benchmark batchnorm_ka($x_cl, $scale_cl, $bias_cl, $μ_cl, $σ²_cl)
BenchmarkTools.Trial: 9522 samples with 1 evaluation.
Range (min … max): 99.260 μs … 1.161 ms ┊ GC (min … max): 0.00% … 0.00%
Time (median): 533.078 μs ┊ GC (median): 0.00%
Time (mean ± σ): 521.982 μs ± 241.771 μs ┊ GC (mean ± σ): 0.00% ± 0.00%
▂▅▅▄▃▁ ▂▁▁▁▁▂▁▁ ▂ ▃ ▂▁▁▂▂▁ ▄▁ ▄▂ ▁▃▄▂▁ ▂▄▃ ▃▃▅ ▃▃▂█▄▂▄
▄██████████████████▇█▆██████▇██▇███▆█████▇████████▇████████▇▆ ▆
99.3 μs Histogram: frequency by time 921 μs <
Memory estimate: 8.61 KiB, allocs estimate: 132. Even if I pass |
See LuxDL/LuxLib.jl#136 for some background context. The main motivation for me is to avoid code duplication between CPU and GPU versions. However, if you take a look at the benchmark comment on the PR (for
batchnorm
andgroupnorm)
you see somewhere between a 10x-40x slowdown between KA and the equivalent optimized loop version (note that it is simply using@simd
or@simd ivdep
and nothing like LoopVectorization).I think there are a couple of reasons for the slowdown:
@simd
annotations are missing (which causes slowdown even in the loop version if I remove the annotations)Potential solutions:
@simd
annotations (Make CPU loops simd & ivdep #436 seems to do this. not sure what is the status for that)The text was updated successfully, but these errors were encountered: