Skip to content
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

LLM.int8() Refactoring: Part 1 #1401

Merged
merged 72 commits into from
Dec 5, 2024
Merged
Show file tree
Hide file tree
Changes from 64 commits
Commits
Show all changes
72 commits
Select commit Hold shift + click to select a range
0cc5c95
Start of int8 refactor: remove col32/col_ampere/col_turing transforms…
matthewdouglas Oct 7, 2024
0f2dc34
Fix unintended change
matthewdouglas Oct 8, 2024
50fe50e
New naive mm_dequant kernel for row-major; cleanup
matthewdouglas Oct 9, 2024
57e6427
fix
matthewdouglas Oct 9, 2024
ca372f2
int8 refactor: initial sparse decomp, cleanup
matthewdouglas Oct 14, 2024
510a880
Int8 refactoring: remove separate NO_CUBLASLT build; more cleanup
matthewdouglas Oct 14, 2024
0ab14fe
int8: inference optimizations, some cleanup
matthewdouglas Oct 18, 2024
fdf4745
int8: more tests passing, cleanup
matthewdouglas Oct 18, 2024
d231db7
int8 - more cleanup, most tests passing
matthewdouglas Oct 21, 2024
dfc4668
int8: specify CUDA stream for int8 ops
matthewdouglas Oct 22, 2024
01bf54e
perf: reduce overhead from getting cudaStream ptr
matthewdouglas Oct 24, 2024
32979b4
Mark some functions for deprecation.
matthewdouglas Oct 24, 2024
521da0c
int8 sparse decomp: small perf improvement
matthewdouglas Oct 30, 2024
c75eecd
Merge branch 'main' into int8
matthewdouglas Oct 30, 2024
217cf8e
update setup.py
matthewdouglas Oct 30, 2024
b9cb5c9
Update bitsandbytes/autograd/_functions.py
matthewdouglas Oct 31, 2024
6fa7905
Update bitsandbytes/functional.py
matthewdouglas Oct 31, 2024
e929df0
Update bitsandbytes/functional.py
matthewdouglas Oct 31, 2024
c7b31df
Update bitsandbytes/research/autograd/_functions.py
matthewdouglas Oct 31, 2024
57300e7
int8 - perf improvement for sparse decomposition inference; deprecate…
matthewdouglas Nov 4, 2024
0460d2e
int8 cleanup
matthewdouglas Nov 4, 2024
437a17e
Merge branch 'int8' of https://github.com/TimDettmers/bitsandbytes in…
matthewdouglas Nov 4, 2024
762daf4
Ignore ruff rule ISC001 (incompatible with formatter)
matthewdouglas Nov 4, 2024
875414e
add comment
matthewdouglas Nov 4, 2024
0aefeb0
int8 more cleanup
matthewdouglas Nov 4, 2024
bfb42d1
Update bitsandbytes/functional.py
matthewdouglas Nov 4, 2024
1ae2476
Merge branch 'int8' of https://github.com/TimDettmers/bitsandbytes in…
matthewdouglas Nov 4, 2024
bf002db
int8: rename / deprecate old fn signatures
matthewdouglas Nov 4, 2024
7f6fb60
Update bitsandbytes/functional.py
matthewdouglas Nov 4, 2024
135b336
type annotation
matthewdouglas Nov 4, 2024
5388877
format update
matthewdouglas Nov 4, 2024
be2e98f
Update bitsandbytes/research/autograd/_functions.py
matthewdouglas Nov 4, 2024
4c849bb
cleanup
matthewdouglas Nov 4, 2024
32a60c5
Merge branch 'int8' of https://github.com/TimDettmers/bitsandbytes in…
matthewdouglas Nov 4, 2024
b954474
Add comment to explain division optimization
matthewdouglas Nov 4, 2024
35dbb2e
more cleanup
matthewdouglas Nov 4, 2024
b36003f
Update bitsandbytes/functional.py
matthewdouglas Nov 5, 2024
03a1963
Update bitsandbytes/functional.py
matthewdouglas Nov 5, 2024
980279f
Update bitsandbytes/functional.py
matthewdouglas Nov 5, 2024
a72c463
cleanup
matthewdouglas Nov 5, 2024
b5d6135
Merge branch 'int8' of https://github.com/TimDettmers/bitsandbytes in…
matthewdouglas Nov 5, 2024
b1c4adc
Type annotations, cleanup
matthewdouglas Nov 5, 2024
ed922b8
remove unused kernels; improved type annotations
matthewdouglas Nov 5, 2024
a93b91f
small perf optimization for single-GPU systems
matthewdouglas Nov 5, 2024
4bced86
small perf optimization for single-GPU systems
matthewdouglas Nov 5, 2024
f61d8bc
update docstrings
matthewdouglas Nov 18, 2024
eed9c3c
Improve docs and tests
matthewdouglas Nov 18, 2024
6e0a4b3
Update docstring
matthewdouglas Nov 18, 2024
161c194
Update test
matthewdouglas Nov 18, 2024
0ac1452
Merge branch 'main' into int8
matthewdouglas Nov 19, 2024
e3051fa
add benchmarking script
matthewdouglas Nov 20, 2024
56abdc2
test cleanup: add deprecated marker, move benchmarks out
matthewdouglas Nov 20, 2024
df941ec
Add int8 dequant function; misc improvements
matthewdouglas Nov 25, 2024
73f02e8
int8 matmul fallback for inner dims not divisible by 4
matthewdouglas Nov 25, 2024
ebb6797
improve register usage of kInt8VectorQuant - especially for A100/H100
matthewdouglas Nov 27, 2024
196c8e0
disable fail-fast for package build
matthewdouglas Nov 27, 2024
fa6f597
maxwell compat
matthewdouglas Nov 27, 2024
498d8de
ptxas verbose
matthewdouglas Nov 27, 2024
a2ee1c4
docs update
matthewdouglas Nov 29, 2024
ac09570
Merge branch 'main' into int8
matthewdouglas Dec 2, 2024
15f1661
doc update
matthewdouglas Dec 2, 2024
5d536c6
backward fix
matthewdouglas Dec 2, 2024
5b2348b
Bugfix sparse decomp
matthewdouglas Dec 3, 2024
bbb7063
Int8 fix for PEFT OLoRA init
matthewdouglas Dec 3, 2024
d25ebb4
Fix test for deprecated spmm_coo
matthewdouglas Dec 3, 2024
3d595f1
test improvement
matthewdouglas Dec 3, 2024
03fcabd
doc update
matthewdouglas Dec 4, 2024
582bf22
typo
matthewdouglas Dec 4, 2024
1ae7c6b
doc cleanup
matthewdouglas Dec 4, 2024
213b10b
docs
matthewdouglas Dec 4, 2024
ca6fd44
add inference benchmark script
matthewdouglas Dec 4, 2024
b8c736b
Add benchmarks, doc update
matthewdouglas Dec 4, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
30 changes: 15 additions & 15 deletions .github/scripts/build-cuda.sh
Original file line number Diff line number Diff line change
Expand Up @@ -8,21 +8,21 @@ build_capability="50;52;60;61;70;75;80;86;89;90"
[[ "${cuda_version}" == 11.7.* ]] && build_capability=${build_capability%??????}
[[ "${cuda_version}" == 11.8.* ]] && build_capability=${build_capability%???}
[[ "${build_os}" = windows-* ]] && python3 -m pip install ninja
for NO_CUBLASLT in ON OFF; do
if [ "${build_os:0:6}" == ubuntu ]; then
image=nvidia/cuda:${cuda_version}-devel-ubuntu22.04
echo "Using image $image"
docker run --platform "linux/$build_arch" -i -w /src -v "$PWD:/src" "$image" sh -c \
"apt-get update \
&& DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends cmake \
&& cmake -DCOMPUTE_BACKEND=cuda -DCOMPUTE_CAPABILITY=\"${build_capability}\" -DNO_CUBLASLT=${NO_CUBLASLT} . \
&& cmake --build ."
else
pip install cmake==3.28.3
cmake -G Ninja -DCOMPUTE_BACKEND=cuda -DCOMPUTE_CAPABILITY="${build_capability}" -DNO_CUBLASLT=${NO_CUBLASLT} -DCMAKE_BUILD_TYPE=Release -S .
cmake --build . --config Release
fi
done

if [ "${build_os:0:6}" == ubuntu ]; then
image=nvidia/cuda:${cuda_version}-devel-ubuntu22.04
echo "Using image $image"
docker run --platform "linux/$build_arch" -i -w /src -v "$PWD:/src" "$image" sh -c \
"apt-get update \
&& DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends cmake \
&& cmake -DPTXAS_VERBOSE=1 -DCOMPUTE_BACKEND=cuda -DCOMPUTE_CAPABILITY=\"${build_capability}\" . \
&& cmake --build ."
else
pip install cmake==3.28.3
cmake -G Ninja -DCOMPUTE_BACKEND=cuda -DCOMPUTE_CAPABILITY="${build_capability}" -DCMAKE_BUILD_TYPE=Release -S .
cmake --build . --config Release
fi


output_dir="output/${build_os}/${build_arch}"
mkdir -p "${output_dir}"
Expand Down
1 change: 1 addition & 0 deletions .github/workflows/python-package.yml
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ jobs:
##
build-shared-libs-cuda:
strategy:
fail-fast: false
matrix:
os: [ubuntu-latest, windows-latest]
arch: [x86_64, aarch64]
Expand Down
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,11 @@ CMakeFiles/
bitsandbytes.dir/
Debug/
Release/
cmake-build-*/

# IDE local files
.vs/
.idea/

# Distribution / packaging
.Python
Expand Down
14 changes: 1 addition & 13 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
# For MSVC: `cmake -B build . && cmake --build build --config Release`
# You can also use the following options and variables
# - COMPUTE_BACKEND: Set to `cpu`, `cuda`, or `mps` to select the backend
# - NO_CUBLASLT: Default OFF, will skip building/linking CUBLASLT support
# - CUDA_VERSION: The expected CUDA version, for sanity checking. The actual version
# is whatever CMake finds on your path.
# - COMPUTE_CAPABILITY: Which GPU Arch/Compute codes to provide to NVCC.
Expand Down Expand Up @@ -47,10 +46,8 @@ if(${COMPUTE_BACKEND} STREQUAL "cuda")
if(APPLE)
message(FATAL_ERROR "CUDA is not supported on macOS" )
endif()
option(NO_CUBLASLT "Disable CUBLAS" OFF)
set(BUILD_CUDA ON)
set(BUILD_MPS OFF)
message(STATUS "NO_CUBLASLT := ${NO_CUBLASLT}")
elseif(${COMPUTE_BACKEND} STREQUAL "mps")
if(NOT APPLE)
message(FATAL_ERROR "MPS is only supported on macOS" )
Expand Down Expand Up @@ -166,9 +163,6 @@ if(BUILD_CUDA)
list(APPEND SRC_FILES ${CUDA_FILES})

string(APPEND BNB_OUTPUT_NAME "_cuda${CUDA_VERSION_SHORT}")
if(NO_CUBLASLT)
string(APPEND BNB_OUTPUT_NAME "_nocublaslt")
endif()
add_compile_definitions(BUILD_CUDA)
elseif(BUILD_MPS)
if(NOT APPLE)
Expand Down Expand Up @@ -212,13 +206,7 @@ target_include_directories(bitsandbytes PUBLIC csrc include)

if(BUILD_CUDA)
target_include_directories(bitsandbytes PUBLIC ${CMAKE_CUDA_TOOLKIT_INCLUDE_DIRECTORIES})
target_link_libraries(bitsandbytes PUBLIC CUDA::cudart CUDA::cublas CUDA::cusparse)
if(NO_CUBLASLT)
target_compile_definitions(bitsandbytes PUBLIC NO_CUBLASLT)
else()
target_link_libraries(bitsandbytes PUBLIC CUDA::cublasLt)
endif()

target_link_libraries(bitsandbytes PUBLIC CUDA::cudart CUDA::cublas CUDA::cublasLt CUDA::cusparse)
set_target_properties(bitsandbytes
PROPERTIES
CUDA_SEPARABLE_COMPILATION ON
Expand Down
68 changes: 68 additions & 0 deletions benchmarking/int8/int8_benchmark.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
"""
Basic benchmark for text generation.

Usage: python benchmarking/int8/int8_benchmark.py
"""

import time

import torch
from torch.profiler import ProfilerActivity, profile
from transformers import AutoModelForCausalLM, AutoTokenizer, BitsAndBytesConfig

MAX_NEW_TOKENS = 128
model_name = "meta-llama/Llama-3.1-8B"

text = "Below is a question. I need an answer.\n\nExplain machine learning: "
tokenizer = AutoTokenizer.from_pretrained(model_name)
input_ids = tokenizer([text] * 8, return_tensors="pt").input_ids.to(0)

model = AutoModelForCausalLM.from_pretrained(
model_name,
device_map="auto",
quantization_config=BitsAndBytesConfig(
load_in_8bit=True,
llm_int8_threshold=6.0,
),
attn_implementation="sdpa",
torch_dtype=torch.float16,
)

print(model)

# warmup
print("Warmup...")
for i in range(3):
generated_ids = model.generate(input_ids, max_new_tokens=MAX_NEW_TOKENS)

print("Profiler starting...")
with profile(
activities=[ProfilerActivity.CPU, ProfilerActivity.CUDA],
with_modules=True,
with_stack=True,
) as prof:
model.generate(input_ids, max_new_tokens=1)

print(
prof.key_averages().table(
sort_by="cpu_time_total",
max_name_column_width=50,
top_level_events_only=True,
row_limit=50,
)
)

torch.cuda.synchronize()


print("Generating...")
num = 0
time_1 = time.time()
for i in range(5):
generated_ids = model.generate(input_ids, max_new_tokens=MAX_NEW_TOKENS)
num += len(generated_ids[0])

print("=" * 40)
print(f"Example:\n{tokenizer.decode(generated_ids[0])}")
print("=" * 40)
print(f"Speed: {num/(time.time() - time_1)}token/s")
70 changes: 70 additions & 0 deletions benchmarking/int8/row_scale_benchmark.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
"""
Extracted from tests/test_functional.py

Note: This feature is currently unused! It is kept here for archival purposes.

Usage: pytest benchmarking/int8/row_scale_benchmark.py
"""

import time

import pytest
import torch

from bitsandbytes import functional as F

k = 20
torch.set_printoptions(precision=5, sci_mode=False, linewidth=120, edgeitems=20, threshold=10000)


@pytest.mark.parametrize(
("dim1", "dim4", "inner"),
[
pytest.param(1024, 12288 * 4, 12288, id="1024, 12288*4, 12288"),
pytest.param(2048, 4096 * 4, 4096, id="2048, 4096*4, 4096"),
],
)
@pytest.mark.skip("Row scale has some bugs for ampere")
@pytest.mark.benchmark
def test_row_scale_bench(dim1, dim4, inner):
formatB = F.get_special_format_str()
err1, err2, err3 = [], [], []
relerr1, relerr2 = [], []
scale = 1
A = torch.randn(dim1, inner, device="cuda").half()
B = torch.randn(dim4, inner, device="cuda").half()
torch.nn.init.xavier_uniform_(B)
# warmpup
for i in range(k):
C1 = torch.matmul(A, B.t())

torch.cuda.synchronize()
t0 = time.time()
for i in range(k):
C1 = torch.matmul(A, B.t())
torch.cuda.synchronize()
print("16", time.time() - t0)

C1a, C1b, stats1a, stats1b, coo_tensor = F.int8_double_quant(A)
CB, absmaxB = F.vectorwise_quant(B, quant_type="linear")
A2, SA = F.nvidia_transform(C1a, "col32")
B2, SB = F.nvidia_transform(CB, formatB)
A1, maxA = F.vectorwise_quant(A, dim=1)

c = 10.0 * inner * scale
row_scale = maxA / c
torch.cuda.synchronize()
t0 = time.time()
for i in range(k):
outC32 = F.int8_linear_matmul(A2, B2, dtype=torch.int8, row_scale=row_scale)
torch.cuda.synchronize()
print("row-wise", time.time() - t0)

C2a, C2b, stats2a, stats2b, coo_tensor = F.int8_double_quant(B)
B2, SB = F.nvidia_transform(C2a, formatB)
torch.cuda.synchronize()
t0 = time.time()
for i in range(k):
outC32 = F.int8_linear_matmul(A2, B2)
torch.cuda.synchronize()
print("vector-wise", time.time() - t0)
Loading
Loading