Compare commits
	
		
			363 Commits
		
	
	
		
			v0.5.6
			...
			brucemacd/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 04950140ec | ||
|   | e7019c9455 | ||
|   | d98bfe7e70 | ||
|   | 6747099d71 | ||
|   | ccc8c6777b | ||
|   | dbb149e6f7 | ||
|   | a807985e59 | ||
|   | 8643c4d5bf | ||
|   | b0c3aba590 | ||
|   | 19c0c25de8 | ||
|   | 2f723ac2d6 | ||
|   | 249fbbe52f | ||
|   | c38680b8a1 | ||
|   | 16fca86c4a | ||
|   | 0f3f9e353d | ||
|   | 6bd0a983cd | ||
|   | 1861fbdeb5 | ||
|   | 3b96a93672 | ||
|   | e53b3cbd0c | ||
|   | b51e0f397c | ||
|   | b42970063d | ||
|   | 493385eb3e | ||
|   | 9876c9faa4 | ||
|   | 4e415029b3 | ||
|   | e172f095ba | ||
|   | c001b98087 | ||
|   | 23fc8e92eb | ||
|   | 4059a297a6 | ||
|   | 66b2539238 | ||
|   | ef27d52e79 | ||
|   | b2a465296d | ||
|   | 5d097277ef | ||
|   | 071a9872cb | ||
|   | 0bd0454ea7 | ||
|   | 01aa788722 | ||
|   | ead27aa9fe | ||
|   | b816ff86c9 | ||
|   | e5d84fb90b | ||
|   | dd66712e31 | ||
|   | f66216e399 | ||
|   | f4f0992b6e | ||
|   | 1feff61977 | ||
|   | 5e0b904e88 | ||
|   | 131f0355a5 | ||
|   | ce929984a3 | ||
|   | 4b34930a31 | ||
|   | 74bd09652d | ||
|   | fb6252d786 | ||
|   | c794fef2f2 | ||
|   | 00ebda8cc4 | ||
|   | d14ce75b95 | ||
|   | 2d6eac9084 | ||
|   | 3ed7ad3ab3 | ||
|   | 6d1103048e | ||
|   | 0ff28758b3 | ||
|   | d3e9ca3eda | ||
|   | 0fbfcf3c9c | ||
|   | 0c220935bd | ||
|   | ffbfe833da | ||
|   | 42a14f7f63 | ||
|   | f8c3dbe5b5 | ||
|   | b078dd157c | ||
|   | 2ddacd7516 | ||
|   | da0e345200 | ||
|   | df94175a0f | ||
|   | 61a8825216 | ||
|   | 021dcf089d | ||
|   | bf24498b1e | ||
|   | 95e271d98f | ||
|   | 364629b8d6 | ||
|   | 108fe02165 | ||
|   | 4561fff36e | ||
|   | 50b5962042 | ||
|   | e27e4a3c1b | ||
|   | 088514bbd4 | ||
|   | 2c8b484643 | ||
|   | 8294676150 | ||
|   | ef378ad673 | ||
|   | 2d2247e59e | ||
|   | 7bf793a600 | ||
|   | 282bfaaa95 | ||
|   | 9679f40146 | ||
|   | 3892c3a703 | ||
|   | 4e320b8b90 | ||
|   | eb2b22b042 | ||
|   | 4ea4d2b189 | ||
|   | 8d76fa23ef | ||
|   | 74b44fdf8f | ||
|   | 65b88c544f | ||
|   | a422ba39c9 | ||
|   | d2ec22371e | ||
|   | 033cec232a | ||
|   | 543240fb5f | ||
|   | 4bed739259 | ||
|   | 80c7ce381b | ||
|   | ccfd41c4f0 | ||
|   | 3e102b7dad | ||
|   | ec46f3286c | ||
|   | 5e2e0b46b1 | ||
|   | 45a13b1dec | ||
|   | 5c0b663969 | ||
|   | 30d7a59ba8 | ||
|   | 4aeb67ef4c | ||
|   | 3ba91634c1 | ||
|   | 1b7433b71e | ||
|   | a70820daa0 | ||
|   | 6b45b1d6b4 | ||
|   | 85ab552028 | ||
|   | b3af953a55 | ||
|   | ad4e0bf3be | ||
|   | aee28501b5 | ||
|   | 83f0ec8269 | ||
|   | c6b6938b3a | ||
|   | fb4664fcec | ||
|   | 20e3593863 | ||
|   | 63a394068c | ||
|   | ab39e08eb9 | ||
|   | 11bfa62796 | ||
|   | f63e62e546 | ||
|   | 65b0f329d1 | ||
|   | 06007c0a18 | ||
|   | a8e83a7654 | ||
|   | 475005504e | ||
|   | 2c40c4d35e | ||
|   | e95278932b | ||
|   | 9d2a20a763 | ||
|   | 2e54d72fc3 | ||
|   | 6b32a2d549 | ||
|   | c5cbe4fc2a | ||
|   | f888912870 | ||
|   | 9e4642e9b3 | ||
|   | 6b0486c216 | ||
|   | d368c039f0 | ||
|   | 9b54267e69 | ||
|   | 46bb0169c4 | ||
|   | 8934324b72 | ||
|   | 0e886595bf | ||
|   | c62861f4fa | ||
|   | 0df1800436 | ||
|   | 631fecc6d9 | ||
|   | 4346c2409d | ||
|   | 4b037a97dc | ||
|   | 5f74d1fd47 | ||
|   | 4dcf80167a | ||
|   | 26a26998fb | ||
|   | 9926eae015 | ||
|   | 8585b7b151 | ||
|   | 7e34f4fbfa | ||
|   | fe776293f7 | ||
|   | d8a5d96b98 | ||
|   | 757668c42f | ||
|   | 96ec8afd09 | ||
|   | e093db92c4 | ||
|   | a1cda80bcb | ||
|   | 4614fafae0 | ||
|   | 4100ed7bdd | ||
|   | f52b2615ef | ||
|   | 25f9b152f9 | ||
|   | 6da8b6a879 | ||
|   | 0daaaef8c9 | ||
|   | 98272fbd58 | ||
|   | b27e8f3f10 | ||
|   | 45df786f09 | ||
|   | daaf42e4a4 | ||
|   | 2dc60d4620 | ||
|   | b5312f30e8 | ||
|   | 26c2e0bd35 | ||
|   | bf920883d5 | ||
|   | 58b9ec1f6b | ||
|   | 7bae7fa5ce | ||
|   | 764e199d67 | ||
|   | bfce55db3d | ||
|   | bab6f34dc0 | ||
|   | 0682dae027 | ||
|   | 1f6986e919 | ||
|   | 4289c74359 | ||
|   | 25248f4bd5 | ||
|   | a7e63b82be | ||
|   | b70fc4d51e | ||
|   | e2252d0fc6 | ||
|   | cae5d4d4ea | ||
|   | 05a01fdecb | ||
|   | 8fe6f69f28 | ||
|   | 1fdb351c37 | ||
|   | 7a01ad7614 | ||
|   | 55ab9f371a | ||
|   | fefbf8f74b | ||
|   | b428ddd796 | ||
|   | ba7d31240e | ||
|   | d25efe3954 | ||
|   | 36dfb906bb | ||
|   | a6f0f908b9 | ||
|   | 3b1ddb2b3a | ||
|   | 1579c4f06d | ||
|   | 3519dd1c6e | ||
|   | e41c4cbea7 | ||
|   | ee048b76d4 | ||
|   | af68d60a58 | ||
|   | 21aa666a1e | ||
|   | ee141cc821 | ||
|   | 55e5776c44 | ||
|   | 854a9195f3 | ||
|   | 96a97adf9b | ||
|   | e75c6126e9 | ||
|   | cda6f5c66c | ||
|   | bebb6823c0 | ||
|   | 31e472baa4 | ||
|   | 657685e85d | ||
|   | a14912858e | ||
|   | eed11ded30 | ||
|   | b42aba40ed | ||
|   | 25885e5335 | ||
|   | 98d44fa39d | ||
|   | 2099e2d267 | ||
|   | 0c1041ad85 | ||
|   | c245b0406f | ||
|   | 8b194b7520 | ||
|   | 3e8b8a1933 | ||
|   | 41dc280491 | ||
|   | 53d2990d9b | ||
|   | e185c08ad9 | ||
|   | 2412adf42b | ||
|   | be2ac1ed93 | ||
|   | dc13813a03 | ||
|   | d6af13efed | ||
|   | a59f665235 | ||
|   | 688925aca9 | ||
|   | 76e903cf9d | ||
|   | a5272130c4 | ||
|   | d7d7e99662 | ||
|   | 2db96c18e7 | ||
|   | e12af460ed | ||
|   | 3ad4bc8afe | ||
|   | 0d694793f2 | ||
|   | e91ae3d47d | ||
|   | 6ecd7f64ba | ||
|   | 888855675e | ||
|   | b16367b4b2 | ||
|   | a499390648 | ||
|   | 4df98f3eb5 | ||
|   | 348b3e0983 | ||
|   | 0b7e1676eb | ||
|   | 314573bfe8 | ||
|   | 4604b10306 | ||
|   | 8c13cfa4dd | ||
|   | 7cfd4aee4d | ||
|   | 68bac1e0a6 | ||
|   | f53f4198c3 | ||
|   | 2192a28eed | ||
|   | 5d81c1a184 | ||
|   | 5c5535c064 | ||
|   | e5bcc51ae1 | ||
|   | bd6a7d5e64 | ||
|   | 14b5a9a150 | ||
|   | ba9ec3d05e | ||
|   | 7c168b08c9 | ||
|   | 3d4cc7833c | ||
|   | 351a85d9ea | ||
|   | bda4ef6c56 | ||
|   | 1e438b237c | ||
|   | d721a02e7d | ||
|   | 778603a818 | ||
|   | 3c874df46e | ||
|   | d2eb226c91 | ||
|   | e13e7c8d94 | ||
|   | 78f403ff45 | ||
|   | 5f8c03189e | ||
|   | 08a299e1d0 | ||
|   | 7b5d916a9a | ||
|   | 33ad61b112 | ||
|   | 716e365615 | ||
|   | 3b4424ff98 | ||
|   | f9c7ead160 | ||
|   | 5930aaeb1a | ||
|   | faf67db089 | ||
|   | 0667baddc6 | ||
|   | d006e1e09b | ||
|   | df2680b4b9 | ||
|   | 010313bb63 | ||
|   | 5296f487a8 | ||
|   | f05774b04c | ||
|   | 6600bd7d91 | ||
|   | ed443a0393 | ||
|   | 6945617af5 | ||
|   | 7916f55009 | ||
|   | d650ad398f | ||
|   | d223f3b697 | ||
|   | 60830695c2 | ||
|   | 01d9a46854 | ||
|   | d773b7d671 | ||
|   | 4d4463b2bd | ||
|   | 0e38297f87 | ||
|   | 7e13f568dc | ||
|   | 58245413f4 | ||
|   | 8cf16063a5 | ||
|   | 3a4449e2f1 | ||
|   | 10d59d5f90 | ||
|   | a4f69a0191 | ||
|   | 82658c3eec | ||
|   | 378d6e1e6a | ||
|   | afa55bc70c | ||
|   | 49df03da9a | ||
|   | 0189bdd0b7 | ||
|   | f4711da7bd | ||
|   | 38117fba83 | ||
|   | 1f766c36fb | ||
|   | 484a99e428 | ||
|   | ec6121c331 | ||
|   | b86c0a1500 | ||
|   | 7e402ebb8c | ||
|   | b901a712c6 | ||
|   | abb8dd57f8 | ||
|   | a400df48c0 | ||
|   | 6ab4ba4c26 | ||
|   | e8d4eb3e68 | ||
|   | ae7e368f75 | ||
|   | 31acd1ebf9 | ||
|   | 9a4757ae66 | ||
|   | 7814019708 | ||
|   | b698f9a0d8 | ||
|   | 32285a6d19 | ||
|   | 1c198977ec | ||
|   | 330b6c50b0 | ||
|   | 928911bc68 | ||
|   | 5b446cc815 | ||
|   | 451c1596af | ||
|   | 932bded12f | ||
|   | 070ad913ac | ||
|   | 8d8b9f83ae | ||
|   | f00d359a67 | ||
|   | 291def6adb | ||
|   | cd3fbf1c49 | ||
|   | c852b8e021 | ||
|   | d8932c55e7 | ||
|   | 63f0269f7f | ||
|   | 4759ecae19 | ||
|   | 65b7ecac7b | ||
|   | f9d2d89135 | ||
|   | 669dc31cf3 | ||
|   | d4d338c224 | ||
|   | bfdeffc375 | ||
|   | e806184023 | ||
|   | 50566113ac | ||
|   | ad22ace439 | ||
|   | f4321a421c | ||
|   | 475333d533 | ||
|   | 39fd89308c | ||
|   | 548a9f56a6 | ||
|   | 3f0cb36bdb | ||
|   | bea1f1fac6 | ||
|   | 5d75d837ef | ||
|   | 711648c9bb | ||
|   | dcfb7a105c | ||
|   | 2ef3c803a1 | ||
|   | 453e4d090b | ||
|   | ca2f9843c8 | ||
|   | 294b6f5a22 | ||
|   | 7bb356c680 | ||
|   | 021817e59a | ||
|   | a420a453b4 | ||
|   | 42cf4db601 | ||
|   | 93a8daf285 | ||
|   | a041b4df7c | 
| @@ -3,7 +3,9 @@ ollama | ||||
| app | ||||
| macapp | ||||
| dist | ||||
| build | ||||
| .env | ||||
| .cache | ||||
| test_data | ||||
| llama/build | ||||
| .git | ||||
|  | ||||
|   | ||||
							
								
								
									
										13
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -7,5 +7,18 @@ llama/**/*.cuh linguist-vendored | ||||
| llama/**/*.m linguist-vendored | ||||
| llama/**/*.metal linguist-vendored | ||||
|  | ||||
| ml/backend/**/*.c linguist-vendored | ||||
| ml/backend/**/*.h linguist-vendored | ||||
| ml/backend/**/*.cpp linguist-vendored | ||||
| ml/backend/**/*.hpp linguist-vendored | ||||
| ml/backend/**/*.cu linguist-vendored | ||||
| ml/backend/**/*.cuh linguist-vendored | ||||
| ml/backend/**/*.m linguist-vendored | ||||
| ml/backend/**/*.metal linguist-vendored | ||||
| ml/backend/**/CMakeLists.txt linguist-vendored | ||||
|  | ||||
| llama/build-info.cpp linguist-generated | ||||
| ml/backend/ggml/ggml/src/ggml-metal/ggml-metal-embed.s linguist-generated | ||||
|  | ||||
| * text=auto | ||||
| *.go text eol=lf | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/10_bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/ISSUE_TEMPLATE/10_bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,6 +9,14 @@ body: | ||||
|       description: What happened? What did you expect to happen? | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: logs | ||||
|     attributes: | ||||
|       label: Relevant log output | ||||
|       description: Please copy and paste any relevant log output. See [Troubleshooting Guide](https://github.com/ollama/ollama/blob/main/docs/troubleshooting.md#how-to-troubleshoot-issues) for details. | ||||
|       render: shell | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: dropdown | ||||
|     id: os | ||||
|     attributes: | ||||
|   | ||||
							
								
								
									
										1027
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1027
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										456
									
								
								.github/workflows/test.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										456
									
								
								.github/workflows/test.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,5 @@ | ||||
| name: test | ||||
|  | ||||
| env: | ||||
|   ROCM_WINDOWS_URL: https://download.amd.com/developer/eula/rocm-hub/AMD-Software-PRO-Edition-24.Q3-WinSvr2022-For-HIP.exe | ||||
|   MSYS2_URL: https://github.com/msys2/msys2-installer/releases/download/2024-07-27/msys2-x86_64-20240727.exe | ||||
|   CUDA_12_WINDOWS_URL: https://developer.download.nvidia.com/compute/cuda/12.4.0/local_installers/cuda_12.4.0_551.61_windows.exe | ||||
|   CUDA_12_WINDOWS_VER: 12.4 | ||||
|  | ||||
| concurrency: | ||||
|   # For PRs, later CI runs preempt previous ones. e.g. a force push on a PR | ||||
|   # cancels running CI jobs and starts all new ones. | ||||
| @@ -27,7 +21,7 @@ jobs: | ||||
|   changes: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       RUNNERS: ${{ steps.changes.outputs.RUNNERS }} | ||||
|       changed: ${{ steps.changes.outputs.changed }} | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
| @@ -35,309 +29,213 @@ jobs: | ||||
|       - id: changes | ||||
|         run: | | ||||
|           changed() { | ||||
|             git diff-tree -r --no-commit-id --name-only \ | ||||
|               $(git merge-base ${{ github.event.pull_request.base.sha }} ${{ github.event.pull_request.head.sha }}) \ | ||||
|               ${{ github.event.pull_request.head.sha }} \ | ||||
|             local BASE=${{ github.event.pull_request.base.sha }} | ||||
|             local HEAD=${{ github.event.pull_request.head.sha }} | ||||
|             local MERGE_BASE=$(git merge-base $BASE $HEAD) | ||||
|             git diff-tree -r --no-commit-id --name-only "$MERGE_BASE" "$HEAD" \ | ||||
|               | xargs python3 -c "import sys; from pathlib import Path; print(any(Path(x).match(glob) for x in sys.argv[1:] for glob in '$*'.split(' ')))" | ||||
|           } | ||||
|  | ||||
|           { | ||||
|             echo RUNNERS=$(changed 'llama/**') | ||||
|           } >>$GITHUB_OUTPUT | ||||
|           echo changed=$(changed 'llama/llama.cpp/**' 'ml/backend/ggml/ggml/**') | tee -a $GITHUB_OUTPUT | ||||
|  | ||||
|   runners-linux-cuda: | ||||
|   linux: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|     if: needs.changes.outputs.changed == 'True' | ||||
|     strategy: | ||||
|       matrix: | ||||
|         cuda-version: | ||||
|           - '11.8.0' | ||||
|         include: | ||||
|           - preset: CPU | ||||
|           - preset: CUDA | ||||
|             container: nvidia/cuda:11.8.0-devel-ubuntu22.04 | ||||
|             flags: '-DCMAKE_CUDA_ARCHITECTURES=87' | ||||
|           - preset: ROCm | ||||
|             container: rocm/dev-ubuntu-22.04:6.1.2 | ||||
|             extra-packages: rocm-libs | ||||
|             flags: '-DAMDGPU_TARGETS=gfx1010 -DCMAKE_PREFIX_PATH=/opt/rocm' | ||||
|     runs-on: linux | ||||
|     container: nvidia/cuda:${{ matrix.cuda-version }}-devel-ubuntu20.04 | ||||
|     container: ${{ matrix.container }} | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - run: | | ||||
|           apt-get update && apt-get install -y git build-essential curl | ||||
|           [ -n "${{ matrix.container }}" ] || sudo=sudo | ||||
|           $sudo apt-get update | ||||
|           $sudo apt-get install -y cmake ccache ${{ matrix.extra-packages }} | ||||
|         env: | ||||
|           DEBIAN_FRONTEND: noninteractive | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-go@v4 | ||||
|       - uses: actions/cache@v4 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - run: go get ./... | ||||
|           path: /github/home/.cache/ccache | ||||
|           key: ccache-${{ runner.os }}-${{ runner.arch }}-${{ matrix.preset }} | ||||
|       - run: | | ||||
|           git config --global --add safe.directory /__w/ollama/ollama | ||||
|           cores=$(grep '^core id' /proc/cpuinfo |sort -u|wc -l) | ||||
|           make -j $cores cuda_v11 | ||||
|   runners-linux-rocm: | ||||
|           cmake --preset ${{ matrix.preset }} ${{ matrix.flags }} | ||||
|           cmake --build --preset ${{ matrix.preset }} --parallel | ||||
|  | ||||
|   windows: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|     if: needs.changes.outputs.changed == 'True' | ||||
|     strategy: | ||||
|       matrix: | ||||
|         rocm-version: | ||||
|           - '6.1.2' | ||||
|     runs-on: linux | ||||
|     container: rocm/dev-ubuntu-20.04:${{ matrix.rocm-version }} | ||||
|     steps: | ||||
|       - run: | | ||||
|           apt-get update && apt-get install -y git build-essential curl rocm-libs | ||||
|         env: | ||||
|           DEBIAN_FRONTEND: noninteractive | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-go@v4 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - run: go get ./... | ||||
|       - run: | | ||||
|           git config --global --add safe.directory /__w/ollama/ollama | ||||
|           cores=$(grep '^core id' /proc/cpuinfo |sort -u|wc -l) | ||||
|           make -j $cores rocm | ||||
|  | ||||
|   # ROCm generation step | ||||
|   runners-windows-rocm: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|         include: | ||||
|           - preset: CPU | ||||
|           - preset: CUDA | ||||
|             install: https://developer.download.nvidia.com/compute/cuda/11.3.1/local_installers/cuda_11.3.1_465.89_win10.exe | ||||
|             flags: '-DCMAKE_CUDA_ARCHITECTURES=80' | ||||
|           - preset: ROCm | ||||
|             install: https://download.amd.com/developer/eula/rocm-hub/AMD-Software-PRO-Edition-24.Q4-WinSvr2022-For-HIP.exe | ||||
|             flags: '-DAMDGPU_TARGETS=gfx1010' | ||||
|     runs-on: windows | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-go@v5 | ||||
|       - run: | | ||||
|           choco install -y --no-progress ccache ninja | ||||
|           ccache -o cache_dir=${{ github.workspace }}\.ccache | ||||
|       - if: matrix.preset == 'CUDA' || matrix.preset == 'ROCm' | ||||
|         id: cache-install | ||||
|         uses: actions/cache/restore@v4 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - name: Set make jobs default | ||||
|         run: | | ||||
|           echo "MAKEFLAGS=--jobs=$((Get-ComputerInfo -Property CsProcessors).CsProcessors.NumberOfCores)" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|  | ||||
|       # ROCM installation steps | ||||
|       - name: 'Cache ROCm installer' | ||||
|         id: cache-rocm | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: rocm-install.exe | ||||
|           key: ${{ env.ROCM_WINDOWS_URL }} | ||||
|       - name: 'Conditionally Download ROCm' | ||||
|         if: steps.cache-rocm.outputs.cache-hit != 'true' | ||||
|           path: | | ||||
|             C:\Program Files\NVIDIA GPU Computing Toolkit\CUDA | ||||
|             C:\Program Files\AMD\ROCm | ||||
|           key: ${{ matrix.install }} | ||||
|       - if: matrix.preset == 'CUDA' | ||||
|         name: Install CUDA ${{ matrix.cuda-version }} | ||||
|         run: | | ||||
|           $ErrorActionPreference = "Stop" | ||||
|           Invoke-WebRequest -Uri "${env:ROCM_WINDOWS_URL}" -OutFile "rocm-install.exe" | ||||
|       - name: 'Install ROCm' | ||||
|         run: | | ||||
|           Start-Process "rocm-install.exe" -ArgumentList '-install' -NoNewWindow -Wait | ||||
|       - name: 'Verify ROCm' | ||||
|         run: | | ||||
|           & 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' --version | ||||
|           echo "HIP_PATH=$(Resolve-Path 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' | split-path | split-path | select -first 1)" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|           if ("${{ steps.cache-install.outputs.cache-hit }}" -ne 'true') { | ||||
|             Invoke-WebRequest -Uri "${{ matrix.install }}" -OutFile "install.exe" | ||||
|             Start-Process -FilePath .\install.exe -ArgumentList (@("-s", "cudart_11.3", "nvcc_11.3", "cublas_11.3", "cublas_dev_11.3")) -NoNewWindow -Wait | ||||
|           } | ||||
|  | ||||
|       - name: Add msys paths | ||||
|         run: | | ||||
|           echo "c:\msys64\usr\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "C:\msys64\clang64\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|       - name: Install msys2 tools | ||||
|         run: | | ||||
|           Start-Process "c:\msys64\usr\bin\pacman.exe" -ArgumentList @("-S", "--noconfirm", "mingw-w64-clang-x86_64-gcc-compat", "mingw-w64-clang-x86_64-clang") -NoNewWindow -Wait | ||||
|  | ||||
|       - name: make rocm runner | ||||
|         run: | | ||||
|           import-module 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Microsoft.VisualStudio.DevShell.dll' | ||||
|           Enter-VsDevShell -vsinstallpath 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise' -skipautomaticlocation -DevCmdArguments '-arch=x64 -no_logo' | ||||
|           if (!(gcc --version | select-string -quiet clang)) { throw "wrong gcc compiler detected - must be clang" } | ||||
|           make -C llama print-HIP_PATH print-HIP_LIB_DIR | ||||
|           make rocm | ||||
|  | ||||
|   # CUDA generation step | ||||
|   runners-windows-cuda: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|     runs-on: windows | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - name: Set make jobs default | ||||
|         run: | | ||||
|           echo "MAKEFLAGS=--jobs=$((Get-ComputerInfo -Property CsProcessors).CsProcessors.NumberOfCores)" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|  | ||||
|       # CUDA installation steps | ||||
|       - name: 'Cache CUDA installer' | ||||
|         id: cache-cuda | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: cuda-install.exe | ||||
|           key: ${{ env.CUDA_12_WINDOWS_URL }} | ||||
|       - name: 'Conditionally Download CUDA' | ||||
|         if: steps.cache-cuda.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           $ErrorActionPreference = "Stop" | ||||
|           Invoke-WebRequest -Uri "${env:CUDA_12_WINDOWS_URL}" -OutFile "cuda-install.exe" | ||||
|       - name: 'Install CUDA' | ||||
|         run: | | ||||
|           $subpackages = @("cudart", "nvcc", "cublas", "cublas_dev") | foreach-object {"${_}_${{ env.CUDA_12_WINDOWS_VER }}"} | ||||
|           Start-Process "cuda-install.exe" -ArgumentList (@("-s") + $subpackages) -NoNewWindow -Wait | ||||
|       - name: 'Verify CUDA' | ||||
|         run: | | ||||
|           & (resolve-path "c:\Program Files\NVIDIA*\CUDA\v*\bin\nvcc.exe")[0] --version | ||||
|           $cudaPath=((resolve-path "c:\Program Files\NVIDIA*\CUDA\v*\bin\nvcc.exe")[0].path | split-path | split-path) | ||||
|           $cudaVer=($cudaPath | split-path -leaf ) -replace 'v(\d+).(\d+)', '$1_$2'  | ||||
|           $cudaPath = (Resolve-Path "C:\Program Files\NVIDIA GPU Computing Toolkit\CUDA\*").path | ||||
|           echo "$cudaPath\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "CUDA_PATH=$cudaPath" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|           echo "CUDA_PATH_V${cudaVer}=$cudaPath" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|           echo "CUDA_PATH_VX_Y=CUDA_PATH_V${cudaVer}" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | ||||
|       - if: matrix.preset == 'ROCm' | ||||
|         name: Install ROCm ${{ matrix.rocm-version }} | ||||
|         run: | | ||||
|           $ErrorActionPreference = "Stop" | ||||
|           if ("${{ steps.cache-install.outputs.cache-hit }}" -ne 'true') { | ||||
|             Invoke-WebRequest -Uri "${{ matrix.install }}" -OutFile "install.exe" | ||||
|             Start-Process -FilePath .\install.exe -ArgumentList '-install' -NoNewWindow -Wait | ||||
|           } | ||||
|  | ||||
|       - name: Add msys paths | ||||
|         run: | | ||||
|           echo "c:\msys64\usr\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "C:\msys64\clang64\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|       - name: Install msys2 tools | ||||
|         run: | | ||||
|           Start-Process "c:\msys64\usr\bin\pacman.exe" -ArgumentList @("-S", "--noconfirm", "mingw-w64-clang-x86_64-gcc-compat", "mingw-w64-clang-x86_64-clang") -NoNewWindow -Wait | ||||
|       - name: make cuda runner | ||||
|         run: | | ||||
|           import-module 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Microsoft.VisualStudio.DevShell.dll' | ||||
|           Enter-VsDevShell -vsinstallpath 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise' -skipautomaticlocation -DevCmdArguments '-arch=x64 -no_logo' | ||||
|           if (!(gcc --version | select-string -quiet clang)) { throw "wrong gcc compiler detected - must be clang" } | ||||
|           make cuda_v$(($env:CUDA_PATH | split-path -leaf) -replace 'v(\d+).*', '$1') | ||||
|  | ||||
|   runners-cpu: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest, macos-latest, windows-2019] | ||||
|         arch: [amd64, arm64] | ||||
|         exclude: | ||||
|           - os: ubuntu-latest | ||||
|             arch: arm64 | ||||
|           - os: windows-2019 | ||||
|             arch: arm64 | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     env: | ||||
|       GOARCH: ${{ matrix.arch }} | ||||
|       ARCH: ${{ matrix.arch }} | ||||
|       CGO_ENABLED: '1' | ||||
|     steps: | ||||
|           $hipPath = (Resolve-Path "C:\Program Files\AMD\ROCm\*").path | ||||
|           echo "$hipPath\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "CC=$hipPath\bin\clang.exe" | Out-File -FilePath $env:GITHUB_ENV -Append | ||||
|           echo "CXX=$hipPath\bin\clang++.exe" | Out-File -FilePath $env:GITHUB_ENV -Append | ||||
|       - if: ${{ !cancelled() && steps.cache-install.outputs.cache-hit != 'true' }} | ||||
|         uses: actions/cache/save@v4 | ||||
|         with: | ||||
|           path: | | ||||
|             C:\Program Files\NVIDIA GPU Computing Toolkit\CUDA | ||||
|             C:\Program Files\AMD\ROCm | ||||
|           key: ${{ matrix.install }} | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-go@v5 | ||||
|       - uses: actions/cache@v4 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - name: Add msys paths | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           echo "c:\msys64\usr\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "C:\msys64\clang64\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|       - name: Install msys2 tools | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           Start-Process "c:\msys64\usr\bin\pacman.exe" -ArgumentList @("-S", "--noconfirm", "mingw-w64-clang-x86_64-gcc-compat", "mingw-w64-clang-x86_64-clang") -NoNewWindow -Wait | ||||
|       - name: 'Build Windows Go Runners' | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           $gopath=(get-command go).source | split-path -parent | ||||
|           $gccpath=(get-command gcc).source | split-path -parent | ||||
|           import-module 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Microsoft.VisualStudio.DevShell.dll' | ||||
|           Enter-VsDevShell -vsinstallpath 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise' -skipautomaticlocation -DevCmdArguments '-arch=x64 -no_logo' | ||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" | ||||
|           $env:PATH="$gopath;$gccpath;$env:PATH" | ||||
|           echo $env:PATH | ||||
|           if (!(gcc --version | select-string -quiet clang)) { throw "wrong gcc compiler detected - must be clang" } | ||||
|           make -j 4 | ||||
|       - name: 'Build Unix Go Runners' | ||||
|         if: ${{ ! startsWith(matrix.os, 'windows-') }} | ||||
|         run: make -j 4 | ||||
|       - run: go build . | ||||
|  | ||||
|   lint: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest, macos-latest, windows-2019] | ||||
|         arch: [amd64, arm64] | ||||
|         exclude: | ||||
|           - os: ubuntu-latest | ||||
|             arch: arm64 | ||||
|           - os: windows-2019 | ||||
|             arch: arm64 | ||||
|           - os: macos-latest | ||||
|             arch: amd64 | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     env: | ||||
|       GOARCH: ${{ matrix.arch }} | ||||
|       CGO_ENABLED: '1' | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           submodules: recursive | ||||
|       - name: Add msys paths | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           echo "c:\msys64\usr\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "C:\msys64\clang64\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|       - name: Install msys2 tools | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           Start-Process "c:\msys64\usr\bin\pacman.exe" -ArgumentList @("-S", "--noconfirm", "mingw-w64-clang-x86_64-gcc-compat", "mingw-w64-clang-x86_64-clang") -NoNewWindow -Wait | ||||
|       - uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: false | ||||
|           path: ${{ github.workspace }}\.ccache | ||||
|           key: ccache-${{ runner.os }}-${{ runner.arch }}-${{ matrix.preset }} | ||||
|       - run: | | ||||
|           case ${{ matrix.arch }} in | ||||
|             amd64) echo ARCH=x86_64 ;; | ||||
|             arm64) echo ARCH=arm64 ;; | ||||
|           esac >>$GITHUB_ENV | ||||
|         shell: bash | ||||
|       - uses: golangci/golangci-lint-action@v6 | ||||
|         with: | ||||
|           args: --timeout 10m0s -v | ||||
|   test: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest, macos-latest, windows-2019] | ||||
|         arch: [amd64] | ||||
|         exclude: | ||||
|           - os: ubuntu-latest | ||||
|             arch: arm64 | ||||
|           - os: windows-2019 | ||||
|             arch: arm64 | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     env: | ||||
|       GOARCH: ${{ matrix.arch }} | ||||
|       CGO_ENABLED: '1' | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           submodules: recursive | ||||
|       - name: Add msys paths | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           echo "c:\msys64\usr\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|           echo "C:\msys64\clang64\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||
|       - name: Install msys2 tools | ||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} | ||||
|         run: | | ||||
|           Start-Process "c:\msys64\usr\bin\pacman.exe" -ArgumentList @("-S", "--noconfirm", "mingw-w64-clang-x86_64-gcc-compat", "mingw-w64-clang-x86_64-clang") -NoNewWindow -Wait | ||||
|       - uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version-file: go.mod | ||||
|           cache: true | ||||
|       - run: | | ||||
|           case ${{ matrix.arch }} in | ||||
|             amd64) echo ARCH=amd64 ;; | ||||
|             arm64) echo ARCH=arm64 ;; | ||||
|           esac >>$GITHUB_ENV | ||||
|         shell: bash | ||||
|       - run: go test ./... | ||||
|           Import-Module 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Microsoft.VisualStudio.DevShell.dll' | ||||
|           Enter-VsDevShell -VsInstallPath 'C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise' -SkipAutomaticLocation  -DevCmdArguments '-arch=x64 -no_logo' | ||||
|           cmake --preset "${{ matrix.preset }}" ${{ matrix.flags }} | ||||
|           cmake --build --parallel --preset "${{ matrix.preset }}" | ||||
|         env: | ||||
|           CMAKE_GENERATOR: Ninja | ||||
|  | ||||
|   patches: | ||||
|     needs: [changes] | ||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} | ||||
|   go_mod_tidy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - name: check that 'go mod tidy' is clean | ||||
|         run: go mod tidy --diff || (echo "Please run 'go mod tidy'." && exit 1) | ||||
|  | ||||
|   test: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest, macos-latest, windows-latest] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     env: | ||||
|       CGO_ENABLED: '1' | ||||
|       GOEXPERIMENT: 'synctest' | ||||
|     steps: | ||||
|       - name: checkout | ||||
|         uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # 4.2.2 | ||||
|  | ||||
|       - name: cache restore | ||||
|         uses: actions/cache/restore@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | ||||
|         with: | ||||
|           submodules: recursive | ||||
|       - name: Verify patches carry all the changes | ||||
|           # Note: unlike the other setups, this is only grabbing the mod download | ||||
|           # cache, rather than the whole mod directory, as the download cache | ||||
|           # contains zips that can be unpacked in parallel faster than they can be | ||||
|           # fetched and extracted by tar | ||||
|           path: | | ||||
|             ~/.cache/go-build | ||||
|             ~/go/pkg/mod/cache | ||||
|             ~\AppData\Local\go-build | ||||
|           # NOTE: The -3- here should be incremented when the scheme of data to be | ||||
|           # cached changes (e.g. path above changes). | ||||
|           key: ${{ github.job }}-${{ runner.os }}-${{ matrix.goarch }}-${{ matrix.buildflags }}-go-3-${{ hashFiles('**/go.sum') }}-${{ github.run_id }} | ||||
|           restore-keys: | | ||||
|             ${{ github.job }}-${{ runner.os }}-${{ matrix.goarch }}-${{ matrix.buildflags }}-go-3-${{ hashFiles('**/go.sum') }} | ||||
|             ${{ github.job }}-${{ runner.os }}-${{ matrix.goarch }}-${{ matrix.buildflags }}-go-3- | ||||
|  | ||||
|       - name: Setup Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           # The caching strategy of setup-go is less than ideal, and wastes | ||||
|           # time by not saving artifacts due to small failures like the linter | ||||
|           # complaining, etc. This means subsequent have to rebuild their world | ||||
|           # again until all checks pass. For instance, if you mispell a word, | ||||
|           # you're punished until you fix it. This is more hostile than | ||||
|           # helpful. | ||||
|           cache: false | ||||
|  | ||||
|           go-version-file: go.mod | ||||
|  | ||||
|       # It is tempting to run this in a platform independent way, but the past | ||||
|       # shows this codebase will see introductions of platform specific code | ||||
|       # generation, and so we need to check this per platform to ensure we | ||||
|       # don't abuse go generate on specific platforms. | ||||
|       - name: check that 'go generate' is clean | ||||
|         if: always() | ||||
|         run: | | ||||
|           make apply-patches sync && git diff --compact-summary --exit-code llama | ||||
|           go generate ./... | ||||
|           git diff --name-only --exit-code || (echo "Please run 'go generate ./...'." && exit 1) | ||||
|  | ||||
|       - name: go test | ||||
|         if: always() | ||||
|         run: go test -count=1 -benchtime=1x ./... | ||||
|  | ||||
|       # TODO(bmizerany): replace this heavy tool with just the | ||||
|       # tools/checks/binaries we want and then make them all run in parallel | ||||
|       # across jobs, not on a single tiny vm on Github Actions. | ||||
|       - uses: golangci/golangci-lint-action@v6 | ||||
|         with: | ||||
|           args: --timeout 10m0s -v | ||||
|  | ||||
|       - name: cache save | ||||
|         # Always save the cache, even if the job fails. The artifacts produced | ||||
|         # during the building of test binaries are not all for naught. They can | ||||
|         # be used to speed up subsequent runs. | ||||
|         if: always() | ||||
|  | ||||
|         uses: actions/cache/save@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | ||||
|         with: | ||||
|           # Note: unlike the other setups, this is only grabbing the mod download | ||||
|           # cache, rather than the whole mod directory, as the download cache | ||||
|           # contains zips that can be unpacked in parallel faster than they can be | ||||
|           # fetched and extracted by tar | ||||
|           path: | | ||||
|             ~/.cache/go-build | ||||
|             ~/go/pkg/mod/cache | ||||
|             ~\AppData\Local\go-build | ||||
|           # NOTE: The -3- here should be incremented when the scheme of data to be | ||||
|           # cached changes (e.g. path above changes). | ||||
|           key: ${{ github.job }}-${{ runner.os }}-${{ matrix.goarch }}-${{ matrix.buildflags }}-go-3-${{ hashFiles('**/go.sum') }}-${{ github.run_id }} | ||||
|  | ||||
|   patches: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - name: Verify patches apply cleanly and do not change files | ||||
|         run: | | ||||
|           make -f Makefile.sync clean sync | ||||
|           git diff --compact-summary --exit-code | ||||
|   | ||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -4,12 +4,13 @@ | ||||
| .venv | ||||
| .swp | ||||
| dist | ||||
| ollama | ||||
| build | ||||
| .cache | ||||
| *.exe | ||||
| .idea | ||||
| test_data | ||||
| *.crt | ||||
| llama/build | ||||
| __debug_bin* | ||||
| llama/vendor | ||||
| llama/build | ||||
| llama/vendor | ||||
| /ollama | ||||
|   | ||||
| @@ -6,8 +6,6 @@ linters: | ||||
|     - bidichk | ||||
|     - bodyclose | ||||
|     - containedctx | ||||
|     - contextcheck | ||||
|     - errcheck | ||||
|     - gocheckcompilerdirectives | ||||
|     - gofmt | ||||
|     - gofumpt | ||||
| @@ -23,10 +21,11 @@ linters: | ||||
|     - staticcheck | ||||
|     - tenv | ||||
|     - unconvert | ||||
|     - unused | ||||
|     - usestdlibvars | ||||
|     - wastedassign | ||||
|     - whitespace | ||||
|   disable: | ||||
|     - usestdlibvars | ||||
|     - errcheck | ||||
| linters-settings: | ||||
|   staticcheck: | ||||
|     checks: | ||||
| @@ -39,5 +38,4 @@ severity: | ||||
|         - gofmt | ||||
|         - goimports | ||||
|         - intrange | ||||
|         - usestdlibvars | ||||
|       severity: info | ||||
|   | ||||
							
								
								
									
										132
									
								
								CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| cmake_minimum_required(VERSION 3.21) | ||||
|  | ||||
| project(Ollama C CXX) | ||||
|  | ||||
| include(CheckLanguage) | ||||
|  | ||||
| find_package(Threads REQUIRED) | ||||
|  | ||||
| set(CMAKE_BUILD_TYPE Release) | ||||
| set(BUILD_SHARED_LIBS ON) | ||||
|  | ||||
| set(CMAKE_CXX_STANDARD 17) | ||||
| set(CMAKE_CXX_STANDARD_REQUIRED ON) | ||||
| set(CMAKE_CXX_EXTENSIONS OFF) | ||||
|  | ||||
| set(GGML_BUILD ON) | ||||
| set(GGML_SHARED ON) | ||||
| set(GGML_CCACHE ON) | ||||
| set(GGML_BACKEND_DL ON) | ||||
| set(GGML_BACKEND_SHARED ON) | ||||
| set(GGML_SCHED_MAX_COPIES 4) | ||||
|  | ||||
| set(GGML_LLAMAFILE ON) | ||||
| set(GGML_CUDA_PEER_MAX_BATCH_SIZE 128) | ||||
| set(GGML_CUDA_GRAPHS ON) | ||||
| set(GGML_CUDA_FA ON) | ||||
|  | ||||
| if((CMAKE_OSX_ARCHITECTURES AND NOT CMAKE_OSX_ARCHITECTURES MATCHES "arm64") | ||||
|     OR (NOT CMAKE_OSX_ARCHITECTURES AND NOT CMAKE_SYSTEM_PROCESSOR MATCHES "arm|aarch64|ARM64|ARMv[0-9]+")) | ||||
|     set(GGML_CPU_ALL_VARIANTS ON) | ||||
| endif() | ||||
|  | ||||
| if (CMAKE_OSX_ARCHITECTURES MATCHES "x86_64") | ||||
|     set(CMAKE_BUILD_RPATH "@loader_path") | ||||
|     set(CMAKE_INSTALL_RPATH "@loader_path") | ||||
| endif() | ||||
|  | ||||
| set(OLLAMA_BUILD_DIR ${CMAKE_BINARY_DIR}/lib/ollama) | ||||
| set(OLLAMA_INSTALL_DIR ${CMAKE_INSTALL_PREFIX}/lib/ollama) | ||||
|  | ||||
| set(CMAKE_RUNTIME_OUTPUT_DIRECTORY         ${OLLAMA_BUILD_DIR}) | ||||
| set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG   ${OLLAMA_BUILD_DIR}) | ||||
| set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${OLLAMA_BUILD_DIR}) | ||||
| set(CMAKE_LIBRARY_OUTPUT_DIRECTORY         ${OLLAMA_BUILD_DIR}) | ||||
| set(CMAKE_LIBRARY_OUTPUT_DIRECTORY_DEBUG   ${OLLAMA_BUILD_DIR}) | ||||
| set(CMAKE_LIBRARY_OUTPUT_DIRECTORY_RELEASE ${OLLAMA_BUILD_DIR}) | ||||
|  | ||||
| include_directories(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src) | ||||
| include_directories(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src/include) | ||||
| include_directories(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src/ggml-cpu) | ||||
| include_directories(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src/ggml-cpu/amx) | ||||
|  | ||||
| set(GGML_CPU ON) | ||||
| add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src) | ||||
| set_property(TARGET ggml PROPERTY EXCLUDE_FROM_ALL TRUE) | ||||
|  | ||||
| get_target_property(CPU_VARIANTS ggml-cpu MANUALLY_ADDED_DEPENDENCIES) | ||||
| if(NOT CPU_VARIANTS) | ||||
|     set(CPU_VARIANTS "ggml-cpu") | ||||
| endif() | ||||
|  | ||||
| install(TARGETS ggml-base ${CPU_VARIANTS} | ||||
|     RUNTIME_DEPENDENCIES | ||||
|         PRE_EXCLUDE_REGEXES ".*" | ||||
|     RUNTIME DESTINATION ${OLLAMA_INSTALL_DIR} COMPONENT CPU | ||||
|     LIBRARY DESTINATION ${OLLAMA_INSTALL_DIR} COMPONENT CPU | ||||
|     FRAMEWORK DESTINATION ${OLLAMA_INSTALL_DIR} COMPONENT CPU | ||||
| ) | ||||
|  | ||||
| check_language(CUDA) | ||||
| if(CMAKE_CUDA_COMPILER) | ||||
|     if(CMAKE_VERSION VERSION_GREATER_EQUAL "3.24" AND NOT CMAKE_CUDA_ARCHITECTURES) | ||||
|         set(CMAKE_CUDA_ARCHITECTURES "native") | ||||
|     endif() | ||||
|  | ||||
|     find_package(CUDAToolkit) | ||||
|     add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src/ggml-cuda) | ||||
|     set(OLLAMA_CUDA_INSTALL_DIR ${OLLAMA_INSTALL_DIR}/cuda_v${CUDAToolkit_VERSION_MAJOR}) | ||||
|     install(TARGETS ggml-cuda | ||||
|         RUNTIME_DEPENDENCIES | ||||
|             DIRECTORIES ${CUDAToolkit_BIN_DIR} ${CUDAToolkit_LIBRARY_DIR} | ||||
|             PRE_INCLUDE_REGEXES cublas cublasLt cudart | ||||
|             PRE_EXCLUDE_REGEXES ".*" | ||||
|         RUNTIME DESTINATION ${OLLAMA_CUDA_INSTALL_DIR} COMPONENT CUDA | ||||
|         LIBRARY DESTINATION ${OLLAMA_CUDA_INSTALL_DIR} COMPONENT CUDA | ||||
|     ) | ||||
| endif() | ||||
|  | ||||
| set(WINDOWS_AMDGPU_TARGETS_EXCLUDE_REGEX "^gfx(906|908|90a|1200|1201):xnack[+-]$" | ||||
|     CACHE STRING | ||||
|     "Regular expression describing AMDGPU_TARGETS not supported on Windows. Override to force building these targets. Default \"^gfx(906|908|90a|1200|1201):xnack[+-]$\"." | ||||
| ) | ||||
|  | ||||
| check_language(HIP) | ||||
| if(CMAKE_HIP_COMPILER) | ||||
|     set(HIP_PLATFORM "amd") | ||||
|  | ||||
|     find_package(hip REQUIRED) | ||||
|     if(NOT AMDGPU_TARGETS) | ||||
|         list(FILTER AMDGPU_TARGETS INCLUDE REGEX "^gfx(900|94[012]|101[02]|1030|110[012]|120[01])$") | ||||
|     elseif(WIN32 AND WINDOWS_AMDGPU_TARGETS_EXCLUDE_REGEX) | ||||
|         list(FILTER AMDGPU_TARGETS EXCLUDE REGEX ${WINDOWS_AMDGPU_TARGETS_EXCLUDE_REGEX}) | ||||
|     endif() | ||||
|  | ||||
|     if(AMDGPU_TARGETS) | ||||
|         add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/ml/backend/ggml/ggml/src/ggml-hip) | ||||
|  | ||||
|         if (WIN32) | ||||
|             target_compile_definitions(ggml-hip PRIVATE GGML_CUDA_NO_PEER_COPY) | ||||
|         endif() | ||||
|  | ||||
|         target_compile_definitions(ggml-hip PRIVATE GGML_HIP_NO_VMM) | ||||
|  | ||||
|         set(OLLAMA_HIP_INSTALL_DIR ${OLLAMA_INSTALL_DIR}/rocm) | ||||
|         install(TARGETS ggml-hip | ||||
|             RUNTIME_DEPENDENCIES | ||||
|                 DIRECTORIES ${HIP_BIN_INSTALL_DIR} ${HIP_LIB_INSTALL_DIR} | ||||
|                 PRE_INCLUDE_REGEXES hipblas rocblas amdhip64 rocsolver amd_comgr hsa-runtime64 rocsparse tinfo rocprofiler-register drm drm_amdgpu numa elf | ||||
|                 PRE_EXCLUDE_REGEXES ".*" | ||||
|                 POST_EXCLUDE_REGEXES "system32" | ||||
|             RUNTIME DESTINATION ${OLLAMA_HIP_INSTALL_DIR} COMPONENT HIP | ||||
|             LIBRARY DESTINATION ${OLLAMA_HIP_INSTALL_DIR} COMPONENT HIP | ||||
|         ) | ||||
|  | ||||
|         foreach(HIP_LIB_BIN_INSTALL_DIR IN ITEMS ${HIP_BIN_INSTALL_DIR} ${HIP_LIB_INSTALL_DIR}) | ||||
|             if(EXISTS ${HIP_LIB_BIN_INSTALL_DIR}/rocblas) | ||||
|                 install(DIRECTORY ${HIP_LIB_BIN_INSTALL_DIR}/rocblas DESTINATION ${OLLAMA_HIP_INSTALL_DIR} COMPONENT HIP) | ||||
|                 break() | ||||
|             endif() | ||||
|         endforeach() | ||||
|     endif() | ||||
| endif() | ||||
							
								
								
									
										110
									
								
								CMakePresets.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								CMakePresets.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| { | ||||
|   "version": 3, | ||||
|   "configurePresets": [ | ||||
|     { | ||||
|       "name": "Default", | ||||
|       "binaryDir": "${sourceDir}/build", | ||||
|       "installDir": "${sourceDir}/dist", | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_BUILD_TYPE": "Release" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "CPU", | ||||
|       "inherits": [ "Default" ] | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA", | ||||
|       "inherits": [ "Default" ] | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA 11", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_CUDA_ARCHITECTURES": "50;52;53;60;61;70;75;80;86" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA 12", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_CUDA_ARCHITECTURES": "50;60;61;70;75;80;86;87;89;90;90a;120" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "JetPack 5", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_CUDA_ARCHITECTURES": "72;87" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "JetPack 6", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_CUDA_ARCHITECTURES": "87" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "ROCm", | ||||
|       "inherits": [ "Default" ], | ||||
|       "cacheVariables": { | ||||
|         "CMAKE_HIP_PLATFORM": "amd" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "name": "ROCm 6", | ||||
|       "inherits": [ "ROCm" ], | ||||
|       "cacheVariables": { | ||||
|         "AMDGPU_TARGETS": "gfx900;gfx940;gfx941;gfx942;gfx1010;gfx1012;gfx1030;gfx1100;gfx1101;gfx1102;gfx1151;gfx1200;gfx1201;gfx906:xnack-;gfx908:xnack-;gfx90a:xnack+;gfx90a:xnack-" | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "buildPresets": [ | ||||
|     { | ||||
|       "name": "Default", | ||||
|       "configurePreset": "Default", | ||||
|       "configuration": "Release" | ||||
|     }, | ||||
|     { | ||||
|       "name": "CPU", | ||||
|       "configurePreset": "Default", | ||||
|       "targets": [ "ggml-cpu" ] | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA", | ||||
|       "configurePreset": "CUDA", | ||||
|       "targets": [ "ggml-cuda" ] | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA 11", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "configurePreset": "CUDA 11" | ||||
|     }, | ||||
|     { | ||||
|       "name": "CUDA 12", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "configurePreset": "CUDA 12" | ||||
|     }, | ||||
|     { | ||||
|       "name": "JetPack 5", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "configurePreset": "JetPack 5" | ||||
|     }, | ||||
|     { | ||||
|       "name": "JetPack 6", | ||||
|       "inherits": [ "CUDA" ], | ||||
|       "configurePreset": "JetPack 6" | ||||
|     }, | ||||
|     { | ||||
|       "name": "ROCm", | ||||
|       "configurePreset": "ROCm", | ||||
|       "targets": [ "ggml-hip" ] | ||||
|     }, | ||||
|     { | ||||
|       "name": "ROCm 6", | ||||
|       "inherits": [ "ROCm" ], | ||||
|       "configurePreset": "ROCm 6" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| @@ -6,8 +6,6 @@ Thank you for your interest in contributing to Ollama! Here are a few guidelines | ||||
|  | ||||
| See the [development documentation](./docs/development.md) for instructions on how to build and run Ollama locally. | ||||
|  | ||||
| ## Pull requests | ||||
|  | ||||
| ### Ideal issues | ||||
|  | ||||
| * [Bugs](https://github.com/ollama/ollama/issues?q=is%3Aissue+is%3Aopen+label%3Abug): issues where Ollama stops working or where it results in an unexpected error. | ||||
| @@ -26,11 +24,64 @@ See the [development documentation](./docs/development.md) for instructions on h | ||||
| * Changes that add significant friction to the user experience | ||||
| * Changes that create a large future maintenance burden for maintainers and contributors | ||||
|  | ||||
| ### Best practices | ||||
| ## Proposing a (non-trivial) change | ||||
|  | ||||
| * Commit messages: please leave both a title and a description in your commit messages. The title should be a short summary of the changes, with a leading word that explains the section of the code being changed (e.g. `api: fix parsing of prompt field`) . In the description, leave a short 2-3 sentences that explain more about the change and its impact. | ||||
| * Tests: please add test coverage to changes where possible. | ||||
| * Minimize dependencies: avoid adding new dependencies unless absolutely necessary. | ||||
| > By "non-trivial", we mean a change that is not a bug fix or small | ||||
| > documentation update. If you are unsure, please ask us on our [Discord | ||||
| > server](https://discord.gg/ollama). | ||||
|  | ||||
| Before opening a non-trivial Pull Request, please open an issue to discuss the change and | ||||
| get feedback from the maintainers. This helps us understand the context of the | ||||
| change and how it fits into Ollama's roadmap and prevents us from duplicating | ||||
| work or you from spending time on a change that we may not be able to accept. | ||||
|  | ||||
| Tips for proposals: | ||||
|  | ||||
| * Explain the problem you are trying to solve, not what you are trying to do. | ||||
| * Explain why the change is important. | ||||
| * Explain how the change will be used. | ||||
| * Explain how the change will be tested. | ||||
|  | ||||
| Additionally, for bonus points: Provide draft documentation you would expect to | ||||
| see if the change were accepted. | ||||
|  | ||||
| ## Pull requests | ||||
|  | ||||
| **Commit messages** | ||||
|  | ||||
| The title should look like: | ||||
|  | ||||
|     <package>: <short description> | ||||
|  | ||||
| The package is the most affected Go package. If the change does not affect Go | ||||
| code, then use the directory name instead. Changes to a single well-known | ||||
| file in the root directory may use the file name. | ||||
|  | ||||
| The short description should start with a lowercase letter and be a | ||||
| continuation of the sentence: | ||||
|  | ||||
|       "This changes Ollama to..." | ||||
|  | ||||
| Examples: | ||||
|  | ||||
|       llm/backend/mlx: support the llama architecture | ||||
|       CONTRIBUTING: provide clairity on good commit messages, and bad | ||||
|  | ||||
| Bad Examples: | ||||
|  | ||||
|       feat: add more emoji | ||||
|       fix: was not using famous web framework | ||||
|       chore: generify code | ||||
|  | ||||
| **Tests** | ||||
|  | ||||
| Please include tests. Strive to test behavior, not implementation. | ||||
|  | ||||
| **New dependencies** | ||||
|  | ||||
| Dependencies should be added sparingly. If you are adding a new dependency, | ||||
| please explain why it is necessary and what other ways you attempted that | ||||
| did not work without it. | ||||
|  | ||||
| ## Need help? | ||||
|  | ||||
|   | ||||
							
								
								
									
										284
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										284
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,201 +1,131 @@ | ||||
| ARG GOLANG_VERSION=1.22.8 | ||||
| ARG CUDA_VERSION_11=11.3.1 | ||||
| ARG CUDA_VERSION_12=12.4.0 | ||||
| ARG ROCM_VERSION=6.1.2 | ||||
| ARG JETPACK_6=r36.2.0 | ||||
| ARG JETPACK_5=r35.4.1 | ||||
| # vim: filetype=dockerfile | ||||
|  | ||||
| ### To create a local image for building linux binaries on mac or windows with efficient incremental builds | ||||
| # | ||||
| # docker build --platform linux/amd64 -t builder-amd64 -f Dockerfile --target unified-builder-amd64 . | ||||
| # docker run --platform linux/amd64 --rm -it -v $(pwd):/go/src/github.com/ollama/ollama/ builder-amd64 | ||||
| # | ||||
| ### Then incremental builds will be much faster in this container | ||||
| # | ||||
| # make -j 10 dist | ||||
| # | ||||
| FROM --platform=linux/amd64 rocm/dev-centos-7:${ROCM_VERSION}-complete AS unified-builder-amd64 | ||||
| ARG GOLANG_VERSION | ||||
| ARG CUDA_VERSION_11 | ||||
| ARG CUDA_VERSION_12 | ||||
| COPY ./scripts/rh_linux_deps.sh / | ||||
| ENV PATH /opt/rh/devtoolset-10/root/usr/bin:/usr/local/cuda/bin:$PATH | ||||
| ENV LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/usr/local/cuda/lib64 | ||||
| RUN GOLANG_VERSION=${GOLANG_VERSION} sh /rh_linux_deps.sh | ||||
| RUN yum-config-manager --add-repo https://developer.download.nvidia.com/compute/cuda/repos/rhel7/x86_64/cuda-rhel7.repo && \ | ||||
|     dnf clean all && \ | ||||
|     dnf install -y \ | ||||
|     zsh \ | ||||
|     cuda-toolkit-$(echo ${CUDA_VERSION_11} | cut -f1-2 -d. | sed -e "s/\./-/g") \ | ||||
|     cuda-toolkit-$(echo ${CUDA_VERSION_12} | cut -f1-2 -d. | sed -e "s/\./-/g") | ||||
| # TODO intel oneapi goes here... | ||||
| ENV GOARCH amd64 | ||||
| ENV CGO_ENABLED 1 | ||||
| WORKDIR /go/src/github.com/ollama/ollama/ | ||||
| ENTRYPOINT [ "zsh" ] | ||||
| ARG FLAVOR=${TARGETARCH} | ||||
|  | ||||
| ### To create a local image for building linux binaries on mac or linux/arm64 with efficient incremental builds | ||||
| # Note: this does not contain jetson variants | ||||
| # | ||||
| # docker build --platform linux/arm64 -t builder-arm64 -f Dockerfile --target unified-builder-arm64 . | ||||
| # docker run --platform linux/arm64 --rm -it -v $(pwd):/go/src/github.com/ollama/ollama/ builder-arm64 | ||||
| # | ||||
| FROM --platform=linux/arm64 rockylinux:8 AS unified-builder-arm64 | ||||
| ARG GOLANG_VERSION | ||||
| ARG CUDA_VERSION_11 | ||||
| ARG CUDA_VERSION_12 | ||||
| COPY ./scripts/rh_linux_deps.sh / | ||||
| RUN GOLANG_VERSION=${GOLANG_VERSION} sh /rh_linux_deps.sh | ||||
| RUN yum-config-manager --add-repo https://developer.download.nvidia.com/compute/cuda/repos/rhel8/sbsa/cuda-rhel8.repo && \ | ||||
|     dnf config-manager --set-enabled appstream && \ | ||||
|     dnf clean all && \ | ||||
|     dnf install -y \ | ||||
|     zsh \ | ||||
|     cuda-toolkit-$(echo ${CUDA_VERSION_11} | cut -f1-2 -d. | sed -e "s/\./-/g") \ | ||||
|     cuda-toolkit-$(echo ${CUDA_VERSION_12} | cut -f1-2 -d. | sed -e "s/\./-/g") | ||||
| ENV PATH /opt/rh/gcc-toolset-10/root/usr/bin:$PATH:/usr/local/cuda/bin | ||||
| ENV LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/usr/local/cuda/lib64 | ||||
| ENV LIBRARY_PATH=/usr/local/cuda/lib64/stubs:/opt/amdgpu/lib64 | ||||
| ENV GOARCH arm64 | ||||
| ENV CGO_ENABLED 1 | ||||
| WORKDIR /go/src/github.com/ollama/ollama/ | ||||
| ENTRYPOINT [ "zsh" ] | ||||
| ARG ROCMVERSION=6.3.3 | ||||
| ARG JETPACK5VERSION=r35.4.1 | ||||
| ARG JETPACK6VERSION=r36.4.0 | ||||
| ARG CMAKEVERSION=3.31.2 | ||||
|  | ||||
| FROM --platform=linux/amd64 unified-builder-amd64 AS build-amd64 | ||||
| COPY . . | ||||
| ARG OLLAMA_SKIP_CUDA_GENERATE | ||||
| ARG OLLAMA_SKIP_ROCM_GENERATE | ||||
| ARG OLLAMA_FAST_BUILD | ||||
| ARG VERSION | ||||
| ARG CUSTOM_CPU_FLAGS | ||||
| # CUDA v11 requires gcc v10.  v10.3 has regressions, so the rockylinux 8.5 AppStream has the latest compatible version | ||||
| FROM --platform=linux/amd64 rocm/dev-almalinux-8:${ROCMVERSION}-complete AS base-amd64 | ||||
| RUN yum install -y yum-utils \ | ||||
|     && yum-config-manager --add-repo https://dl.rockylinux.org/vault/rocky/8.5/AppStream/\$basearch/os/ \ | ||||
|     && rpm --import https://dl.rockylinux.org/pub/rocky/RPM-GPG-KEY-Rocky-8 \ | ||||
|     && dnf install -y yum-utils ccache gcc-toolset-10-gcc-10.2.1-8.2.el8 gcc-toolset-10-gcc-c++-10.2.1-8.2.el8 gcc-toolset-10-binutils-2.35-11.el8 \ | ||||
|     && yum-config-manager --add-repo https://developer.download.nvidia.com/compute/cuda/repos/rhel8/x86_64/cuda-rhel8.repo | ||||
| ENV PATH=/opt/rh/gcc-toolset-10/root/usr/bin:$PATH | ||||
|  | ||||
| FROM --platform=linux/arm64 almalinux:8 AS base-arm64 | ||||
| # install epel-release for ccache | ||||
| RUN yum install -y yum-utils epel-release \ | ||||
|     && dnf install -y clang ccache \ | ||||
|     && yum-config-manager --add-repo https://developer.download.nvidia.com/compute/cuda/repos/rhel8/sbsa/cuda-rhel8.repo | ||||
| ENV CC=clang CXX=clang++ | ||||
|  | ||||
| FROM base-${TARGETARCH} AS base | ||||
| ARG CMAKEVERSION | ||||
| RUN curl -fsSL https://github.com/Kitware/CMake/releases/download/v${CMAKEVERSION}/cmake-${CMAKEVERSION}-linux-$(uname -m).tar.gz | tar xz -C /usr/local --strip-components 1 | ||||
| COPY CMakeLists.txt CMakePresets.json . | ||||
| COPY ml/backend/ggml/ggml ml/backend/ggml/ggml | ||||
| ENV LDFLAGS=-s | ||||
|  | ||||
| FROM base AS cpu | ||||
| RUN dnf install -y gcc-toolset-11-gcc gcc-toolset-11-gcc-c++ | ||||
| ENV PATH=/opt/rh/gcc-toolset-11/root/usr/bin:$PATH | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     if grep "^flags" /proc/cpuinfo|grep avx>/dev/null; then \ | ||||
|         make -j $(nproc) dist ; \ | ||||
|     else \ | ||||
|         make -j 5 dist ; \ | ||||
|     fi | ||||
| RUN cd dist/linux-$GOARCH && \ | ||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH.tgz | ||||
| RUN if [ -z ${OLLAMA_SKIP_ROCM_GENERATE} ] ; then \ | ||||
|     cd dist/linux-$GOARCH-rocm && \ | ||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH-rocm.tgz ;\ | ||||
|     fi | ||||
|     cmake --preset 'CPU' \ | ||||
|         && cmake --build --parallel --preset 'CPU' \ | ||||
|         && cmake --install build --component CPU --strip --parallel 8 | ||||
|  | ||||
| # Jetsons need to be built in discrete stages | ||||
| FROM --platform=linux/arm64 nvcr.io/nvidia/l4t-jetpack:${JETPACK_5} AS runners-jetpack5-arm64 | ||||
| ARG GOLANG_VERSION | ||||
| RUN apt-get update && apt-get install -y git curl ccache && \ | ||||
|     curl -s -L https://dl.google.com/go/go${GOLANG_VERSION}.linux-arm64.tar.gz | tar xz -C /usr/local && \ | ||||
|     ln -s /usr/local/go/bin/go /usr/local/bin/go && \ | ||||
|     ln -s /usr/local/go/bin/gofmt /usr/local/bin/gofmt && \ | ||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* | ||||
| WORKDIR /go/src/github.com/ollama/ollama/ | ||||
| COPY . . | ||||
| ARG CGO_CFLAGS | ||||
| ENV GOARCH arm64 | ||||
| ARG VERSION | ||||
| FROM base AS cuda-11 | ||||
| ARG CUDA11VERSION=11.3 | ||||
| RUN dnf install -y cuda-toolkit-${CUDA11VERSION//./-} | ||||
| ENV PATH=/usr/local/cuda-11/bin:$PATH | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     make -j 5 dist_cuda_v11 \ | ||||
|         CUDA_ARCHITECTURES="72;87" \ | ||||
|         GPU_RUNNER_VARIANT=_jetpack5 \ | ||||
|         DIST_LIB_DIR=/go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack5/lib/ollama \ | ||||
|         DIST_GPU_RUNNER_DEPS_DIR=/go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack5/lib/ollama/cuda_jetpack5 | ||||
|     cmake --preset 'CUDA 11' \ | ||||
|         && cmake --build --parallel --preset 'CUDA 11' \ | ||||
|         && cmake --install build --component CUDA --strip --parallel 8 | ||||
|  | ||||
| FROM --platform=linux/arm64 nvcr.io/nvidia/l4t-jetpack:${JETPACK_6} AS runners-jetpack6-arm64 | ||||
| ARG GOLANG_VERSION | ||||
| RUN apt-get update && apt-get install -y git curl ccache && \ | ||||
|     curl -s -L https://dl.google.com/go/go${GOLANG_VERSION}.linux-arm64.tar.gz | tar xz -C /usr/local && \ | ||||
|     ln -s /usr/local/go/bin/go /usr/local/bin/go && \ | ||||
|     ln -s /usr/local/go/bin/gofmt /usr/local/bin/gofmt && \ | ||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* | ||||
| WORKDIR /go/src/github.com/ollama/ollama/ | ||||
| COPY . . | ||||
| ARG CGO_CFLAGS | ||||
| ENV GOARCH arm64 | ||||
| ARG VERSION | ||||
| FROM base AS cuda-12 | ||||
| ARG CUDA12VERSION=12.8 | ||||
| RUN dnf install -y cuda-toolkit-${CUDA12VERSION//./-} | ||||
| ENV PATH=/usr/local/cuda-12/bin:$PATH | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     make -j 5 dist_cuda_v12 \ | ||||
|         CUDA_ARCHITECTURES="87" \ | ||||
|         GPU_RUNNER_VARIANT=_jetpack6 \ | ||||
|         DIST_LIB_DIR=/go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack6/lib/ollama \ | ||||
|         DIST_GPU_RUNNER_DEPS_DIR=/go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack6/lib/ollama/cuda_jetpack6 | ||||
|     cmake --preset 'CUDA 12' \ | ||||
|         && cmake --build --parallel --preset 'CUDA 12' \ | ||||
|         && cmake --install build --component CUDA --strip --parallel 8 | ||||
|  | ||||
| FROM --platform=linux/arm64 unified-builder-arm64 AS build-arm64 | ||||
| COPY . . | ||||
| ARG OLLAMA_SKIP_CUDA_GENERATE | ||||
| ARG OLLAMA_FAST_BUILD | ||||
| ARG VERSION | ||||
| FROM base AS rocm-6 | ||||
| ENV PATH=/opt/rocm/hcc/bin:/opt/rocm/hip/bin:/opt/rocm/bin:/opt/rocm/hcc/bin:$PATH | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     make -j 5 dist | ||||
| COPY --from=runners-jetpack5-arm64 /go/src/github.com/ollama/ollama/dist/ dist/ | ||||
| COPY --from=runners-jetpack6-arm64 /go/src/github.com/ollama/ollama/dist/ dist/ | ||||
| RUN cd dist/linux-$GOARCH && \ | ||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH.tgz | ||||
| RUN cd dist/linux-$GOARCH-jetpack5 && \ | ||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH-jetpack5.tgz | ||||
| RUN cd dist/linux-$GOARCH-jetpack6 && \ | ||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH-jetpack6.tgz | ||||
|     cmake --preset 'ROCm 6' \ | ||||
|         && cmake --build --parallel --preset 'ROCm 6' \ | ||||
|         && cmake --install build --component HIP --strip --parallel 8 | ||||
|  | ||||
| FROM --platform=linux/amd64 scratch AS dist-amd64 | ||||
| COPY --from=build-amd64 /go/src/github.com/ollama/ollama/dist/ollama-linux-*.tgz / | ||||
| FROM --platform=linux/arm64 scratch AS dist-arm64 | ||||
| COPY --from=build-arm64 /go/src/github.com/ollama/ollama/dist/ollama-linux-*.tgz / | ||||
| FROM dist-$TARGETARCH AS dist | ||||
| FROM --platform=linux/arm64 nvcr.io/nvidia/l4t-jetpack:${JETPACK5VERSION} AS jetpack-5 | ||||
| ARG CMAKEVERSION | ||||
| RUN apt-get update && apt-get install -y curl ccache \ | ||||
|     && curl -fsSL https://github.com/Kitware/CMake/releases/download/v${CMAKEVERSION}/cmake-${CMAKEVERSION}-linux-$(uname -m).tar.gz | tar xz -C /usr/local --strip-components 1 | ||||
| COPY CMakeLists.txt CMakePresets.json . | ||||
| COPY ml/backend/ggml/ggml ml/backend/ggml/ggml | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     cmake --preset 'JetPack 5' \ | ||||
|         && cmake --build --parallel --preset 'JetPack 5' \ | ||||
|         && cmake --install build --component CUDA --strip --parallel 8 | ||||
|  | ||||
| FROM --platform=linux/arm64 nvcr.io/nvidia/l4t-jetpack:${JETPACK6VERSION} AS jetpack-6 | ||||
| ARG CMAKEVERSION | ||||
| RUN apt-get update && apt-get install -y curl ccache \ | ||||
|     && curl -fsSL https://github.com/Kitware/CMake/releases/download/v${CMAKEVERSION}/cmake-${CMAKEVERSION}-linux-$(uname -m).tar.gz | tar xz -C /usr/local --strip-components 1 | ||||
| COPY CMakeLists.txt CMakePresets.json . | ||||
| COPY ml/backend/ggml/ggml ml/backend/ggml/ggml | ||||
| RUN --mount=type=cache,target=/root/.ccache \ | ||||
|     cmake --preset 'JetPack 6' \ | ||||
|         && cmake --build --parallel --preset 'JetPack 6' \ | ||||
|         && cmake --install build --component CUDA --strip --parallel 8 | ||||
|  | ||||
| # For amd64 container images, filter out cuda/rocm to minimize size | ||||
| FROM build-amd64 AS runners-cuda-amd64 | ||||
| RUN rm -rf \ | ||||
|     ./dist/linux-amd64/lib/ollama/libggml_hipblas.so \ | ||||
|     ./dist/linux-amd64/lib/ollama/runners/rocm* | ||||
| FROM base AS build | ||||
| WORKDIR /go/src/github.com/ollama/ollama | ||||
| COPY go.mod go.sum . | ||||
| RUN curl -fsSL https://golang.org/dl/go$(awk '/^go/ { print $2 }' go.mod).linux-$(case $(uname -m) in x86_64) echo amd64 ;; aarch64) echo arm64 ;; esac).tar.gz | tar xz -C /usr/local | ||||
| ENV PATH=/usr/local/go/bin:$PATH | ||||
| RUN go mod download | ||||
| COPY . . | ||||
| ARG GOFLAGS="'-ldflags=-w -s'" | ||||
| ENV CGO_ENABLED=1 | ||||
| RUN --mount=type=cache,target=/root/.cache/go-build \ | ||||
|     go build -trimpath -buildmode=pie -o /bin/ollama . | ||||
|  | ||||
| FROM build-amd64 AS runners-rocm-amd64 | ||||
| RUN rm -rf \ | ||||
|     ./dist/linux-amd64/lib/ollama/libggml_cuda*.so \ | ||||
|     ./dist/linux-amd64/lib/ollama/libcu*.so* \ | ||||
|     ./dist/linux-amd64/lib/ollama/runners/cuda* | ||||
| FROM --platform=linux/amd64 scratch AS amd64 | ||||
| COPY --from=cuda-11 dist/lib/ollama/cuda_v11 /lib/ollama/cuda_v11 | ||||
| COPY --from=cuda-12 dist/lib/ollama/cuda_v12 /lib/ollama/cuda_v12 | ||||
|  | ||||
| FROM --platform=linux/amd64 ubuntu:22.04 AS runtime-amd64 | ||||
| RUN apt-get update && \ | ||||
|     apt-get install -y ca-certificates && \ | ||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* | ||||
| COPY --from=build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/bin/ /bin/ | ||||
| COPY --from=runners-cuda-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ | ||||
| FROM --platform=linux/arm64 scratch AS arm64 | ||||
| COPY --from=cuda-11 dist/lib/ollama/cuda_v11 /lib/ollama/cuda_v11 | ||||
| COPY --from=cuda-12 dist/lib/ollama/cuda_v12 /lib/ollama/cuda_v12 | ||||
| COPY --from=jetpack-5 dist/lib/ollama/cuda_v11 lib/ollama/cuda_jetpack5 | ||||
| COPY --from=jetpack-6 dist/lib/ollama/cuda_v12 lib/ollama/cuda_jetpack6 | ||||
|  | ||||
| FROM --platform=linux/arm64 ubuntu:22.04 AS runtime-arm64 | ||||
| RUN apt-get update && \ | ||||
|     apt-get install -y ca-certificates && \ | ||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* | ||||
| COPY --from=build-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/bin/ /bin/ | ||||
| COPY --from=build-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/lib/ /lib/ | ||||
| COPY --from=runners-jetpack5-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack5/lib/ /lib/ | ||||
| COPY --from=runners-jetpack6-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64-jetpack6/lib/ /lib/ | ||||
| FROM scratch AS rocm | ||||
| COPY --from=rocm-6 dist/lib/ollama/rocm /lib/ollama/rocm | ||||
|  | ||||
| FROM ${FLAVOR} AS archive | ||||
| COPY --from=cpu dist/lib/ollama /lib/ollama | ||||
| COPY --from=build /bin/ollama /bin/ollama | ||||
|  | ||||
| # ROCm libraries larger so we keep it distinct from the CPU/CUDA image | ||||
| FROM --platform=linux/amd64 ubuntu:22.04 AS runtime-rocm | ||||
| # Frontload the rocm libraries which are large, and rarely change to increase chance of a common layer | ||||
| # across releases | ||||
| COPY --from=build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64-rocm/lib/ /lib/ | ||||
| RUN apt-get update && \ | ||||
|     apt-get install -y ca-certificates && \ | ||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* | ||||
| COPY --from=build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/bin/ /bin/ | ||||
| COPY --from=runners-rocm-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ | ||||
|  | ||||
| EXPOSE 11434 | ||||
| ENV OLLAMA_HOST 0.0.0.0 | ||||
|  | ||||
| ENTRYPOINT ["/bin/ollama"] | ||||
| CMD ["serve"] | ||||
|  | ||||
| FROM runtime-$TARGETARCH | ||||
| EXPOSE 11434 | ||||
| ENV OLLAMA_HOST 0.0.0.0 | ||||
| FROM ubuntu:20.04 | ||||
| RUN apt-get update \ | ||||
|     && apt-get install -y ca-certificates \ | ||||
|     && apt-get clean \ | ||||
|     && rm -rf /var/lib/apt/lists/* | ||||
| COPY --from=archive /bin /usr/bin | ||||
| ENV PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin | ||||
| COPY --from=archive /lib/ollama /usr/lib/ollama | ||||
| ENV LD_LIBRARY_PATH=/usr/local/nvidia/lib:/usr/local/nvidia/lib64 | ||||
| ENV NVIDIA_DRIVER_CAPABILITIES=compute,utility | ||||
| ENV NVIDIA_VISIBLE_DEVICES=all | ||||
|  | ||||
| ENV OLLAMA_HOST=0.0.0.0:11434 | ||||
| EXPOSE 11434 | ||||
| ENTRYPOINT ["/bin/ollama"] | ||||
| CMD ["serve"] | ||||
|   | ||||
							
								
								
									
										103
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										103
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,103 +0,0 @@ | ||||
| # top level makefile for Ollama | ||||
| include make/common-defs.make | ||||
|  | ||||
|  | ||||
| # Determine which if any GPU runners we should build | ||||
| include make/cuda-v11-defs.make | ||||
| include make/cuda-v12-defs.make | ||||
| include make/rocm-defs.make | ||||
|  | ||||
| ifeq ($(CUSTOM_CPU_FLAGS),) | ||||
| ifeq ($(ARCH),amd64) | ||||
| 	RUNNER_TARGETS=cpu | ||||
| endif | ||||
| # Without CUSTOM_CPU_FLAGS we default to build both v11 and v12 if present | ||||
| ifeq ($(OLLAMA_SKIP_CUDA_GENERATE),) | ||||
| ifneq ($(CUDA_11_COMPILER),) | ||||
| 	RUNNER_TARGETS += cuda_v11 | ||||
| endif | ||||
| ifneq ($(CUDA_12_COMPILER),) | ||||
| 	RUNNER_TARGETS += cuda_v12 | ||||
| endif | ||||
| endif | ||||
| else # CUSTOM_CPU_FLAGS is set, we'll build only the latest cuda version detected | ||||
| ifneq ($(CUDA_12_COMPILER),) | ||||
| 	RUNNER_TARGETS += cuda_v12 | ||||
| else ifneq ($(CUDA_11_COMPILER),) | ||||
| 	RUNNER_TARGETS += cuda_v11 | ||||
| endif | ||||
| endif | ||||
|  | ||||
| ifeq ($(OLLAMA_SKIP_ROCM_GENERATE),) | ||||
| ifneq ($(HIP_COMPILER),) | ||||
| 	RUNNER_TARGETS += rocm | ||||
| endif | ||||
| endif | ||||
|  | ||||
|  | ||||
| all: runners exe | ||||
|  | ||||
| dist: $(addprefix dist_, $(RUNNER_TARGETS)) dist_exe | ||||
|  | ||||
| dist_%: | ||||
| 	@$(MAKE) --no-print-directory -f make/Makefile.$* dist | ||||
|  | ||||
| runners: $(RUNNER_TARGETS) | ||||
|  | ||||
| $(RUNNER_TARGETS): | ||||
| 	@$(MAKE) --no-print-directory -f make/Makefile.$@ | ||||
|  | ||||
| exe dist_exe: | ||||
| 	@$(MAKE) --no-print-directory -f make/Makefile.ollama $@ | ||||
|  | ||||
| help-sync apply-patches create-patches sync sync-clean: | ||||
| 	@$(MAKE) --no-print-directory -f make/Makefile.sync $@ | ||||
|  | ||||
| test integration lint: | ||||
| 	@$(MAKE) --no-print-directory -f make/Makefile.test $@ | ||||
|  | ||||
| clean: | ||||
| 	rm -rf $(BUILD_DIR) $(DIST_LIB_DIR) $(OLLAMA_EXE) $(DIST_OLLAMA_EXE) | ||||
| 	go clean -cache | ||||
|  | ||||
| help: | ||||
| 	@echo "The following make targets will help you build Ollama" | ||||
| 	@echo "" | ||||
| 	@echo "	make all   		# (default target) Build Ollama llm subprocess runners, and the primary ollama executable" | ||||
| 	@echo "	make runners		# Build Ollama llm subprocess runners; after you may use 'go build .' to build the primary ollama exectuable" | ||||
| 	@echo "	make <runner>		# Build specific runners. Enabled: '$(RUNNER_TARGETS)'" | ||||
| 	@echo "	make dist		# Build the runners and primary ollama executable for distribution" | ||||
| 	@echo "	make help-sync 		# Help information on vendor update targets" | ||||
| 	@echo "	make help-runners 	# Help information on runner targets" | ||||
| 	@echo "" | ||||
| 	@echo "The following make targets will help you test Ollama" | ||||
| 	@echo "" | ||||
| 	@echo "	make test   		# Run unit tests" | ||||
| 	@echo "	make integration	# Run integration tests.  You must 'make all' first" | ||||
| 	@echo "	make lint   		# Run lint and style tests" | ||||
| 	@echo "" | ||||
| 	@echo "For more information see 'docs/development.md'" | ||||
| 	@echo "" | ||||
|  | ||||
|  | ||||
| help-runners: | ||||
| 	@echo "The following runners will be built based on discovered GPU libraries: '$(RUNNER_TARGETS)'" | ||||
| 	@echo "" | ||||
| 	@echo "GPU Runner CPU Flags: '$(GPU_RUNNER_CPU_FLAGS)'  (Override with CUSTOM_CPU_FLAGS)" | ||||
| 	@echo "" | ||||
| 	@echo "# CUDA_PATH sets the location where CUDA toolkits are present" | ||||
| 	@echo "CUDA_PATH=$(CUDA_PATH)" | ||||
| 	@echo "	CUDA_11_PATH=$(CUDA_11_PATH)" | ||||
| 	@echo "	CUDA_11_COMPILER=$(CUDA_11_COMPILER)" | ||||
| 	@echo "	CUDA_12_PATH=$(CUDA_12_PATH)" | ||||
| 	@echo "	CUDA_12_COMPILER=$(CUDA_12_COMPILER)" | ||||
| 	@echo "" | ||||
| 	@echo "# HIP_PATH sets the location where the ROCm toolkit is present" | ||||
| 	@echo "HIP_PATH=$(HIP_PATH)" | ||||
| 	@echo "	HIP_COMPILER=$(HIP_COMPILER)" | ||||
|  | ||||
| .PHONY: all exe dist help help-sync help-runners test integration lint runners clean $(RUNNER_TARGETS) | ||||
|  | ||||
| # Handy debugging for make variables | ||||
| print-%: | ||||
| 	@echo '$*=$($*)' | ||||
							
								
								
									
										60
									
								
								Makefile.sync
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								Makefile.sync
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| UPSTREAM=https://github.com/ggerganov/llama.cpp.git | ||||
| WORKDIR=llama/vendor | ||||
| FETCH_HEAD=d7cfe1ffe0f435d0048a6058d529daf76e072d9c | ||||
|  | ||||
| .PHONY: help | ||||
| help: | ||||
| 	@echo "Available targets:" | ||||
| 	@echo "    sync                 Sync with upstream repositories" | ||||
| 	@echo "    checkout             Checkout upstream repository" | ||||
| 	@echo "    apply-patches        Apply patches to local repository" | ||||
| 	@echo "    format-patches       Format patches from local repository" | ||||
| 	@echo "    clean                Clean local repository" | ||||
| 	@echo | ||||
| 	@echo "Example:" | ||||
| 	@echo "    make -f $(lastword $(MAKEFILE_LIST)) clean sync" | ||||
|  | ||||
| .PHONY: sync | ||||
| sync: llama/build-info.cpp llama/llama.cpp ml/backend/ggml/ggml apply-patches | ||||
|  | ||||
| .PHONY: llama/build-info.cpp | ||||
| llama/build-info.cpp: llama/build-info.cpp.in | ||||
| 	sed -e 's|@FETCH_HEAD@|$(FETCH_HEAD)|' $< > $@ | ||||
|  | ||||
| .PHONY: llama/llama.cpp | ||||
| llama/llama.cpp: llama/vendor/ apply-patches | ||||
| 	rsync -arvzc -f "merge $@/.rsync-filter" $< $@ | ||||
|  | ||||
| .PHONY: ml/backend/ggml/ggml apply-patches | ||||
| ml/backend/ggml/ggml: llama/vendor/ggml/ apply-patches | ||||
| 	rsync -arvzc -f "merge $@/.rsync-filter" $< $@ | ||||
|  | ||||
| PATCHES=$(wildcard llama/patches/*.patch) | ||||
|  | ||||
| .PHONY: apply-patches | ||||
| .NOTPARALLEL: | ||||
| apply-patches: $(addsuffix ed, $(PATCHES)) | ||||
|  | ||||
| %.patched: %.patch | ||||
| 	@if git -c user.name=nobody -c 'user.email=<>' -C $(WORKDIR) am -3 $(realpath $<); then touch $@; else git -C $(WORKDIR) am --abort; exit 1; fi | ||||
|  | ||||
| .PHONY: checkout | ||||
| checkout: $(WORKDIR) | ||||
| 	git -C $(WORKDIR) fetch | ||||
| 	git -C $(WORKDIR) checkout -f $(FETCH_HEAD) | ||||
|  | ||||
| $(WORKDIR): | ||||
| 	git clone $(UPSTREAM) $(WORKDIR) | ||||
|  | ||||
| .PHONE: format-patches | ||||
| format-patches: llama/patches | ||||
| 	git -C $(WORKDIR) format-patch \ | ||||
| 		--no-signature \ | ||||
| 		--no-numbered \ | ||||
| 		--zero-commit \ | ||||
| 		-o $(realpath $<) \ | ||||
| 		$(FETCH_HEAD) | ||||
|  | ||||
| .PHONE: clean | ||||
| clean: checkout | ||||
| 	$(RM) $(addsuffix ed, $(PATCHES)) | ||||
							
								
								
									
										112
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										112
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| <div align="center"> | ||||
|   <a href="https://ollama.com" /> | ||||
|   <a href="https://ollama.com"> | ||||
|     <img alt="ollama" height="200px" src="https://github.com/ollama/ollama/assets/3325447/0d0b44e2-8f4a-4e99-9b52-a5c1c741c8f7"> | ||||
|   </a> | ||||
| </div> | ||||
| @@ -18,7 +18,7 @@ Get up and running with large language models. | ||||
|  | ||||
| ### Linux | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl -fsSL https://ollama.com/install.sh | sh | ||||
| ``` | ||||
|  | ||||
| @@ -42,7 +42,7 @@ The official [Ollama Docker image](https://hub.docker.com/r/ollama/ollama) `olla | ||||
|  | ||||
| To run and chat with [Llama 3.2](https://ollama.com/library/llama3.2): | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama run llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -54,6 +54,13 @@ Here are some example models that can be downloaded: | ||||
|  | ||||
| | Model              | Parameters | Size  | Download                         | | ||||
| | ------------------ | ---------- | ----- | -------------------------------- | | ||||
| | Gemma 3            | 1B         | 815MB | `ollama run gemma3:1b`           | | ||||
| | Gemma 3            | 4B         | 3.3GB | `ollama run gemma3`              | | ||||
| | Gemma 3            | 12B        | 8.1GB | `ollama run gemma3:12b`          | | ||||
| | Gemma 3            | 27B        | 17GB  | `ollama run gemma3:27b`          | | ||||
| | QwQ                | 32B        | 20GB  | `ollama run qwq`                 | | ||||
| | DeepSeek-R1        | 7B         | 4.7GB | `ollama run deepseek-r1`         | | ||||
| | DeepSeek-R1        | 671B       | 404GB | `ollama run deepseek-r1:671b`    | | ||||
| | Llama 3.3          | 70B        | 43GB  | `ollama run llama3.3`            | | ||||
| | Llama 3.2          | 3B         | 2.0GB | `ollama run llama3.2`            | | ||||
| | Llama 3.2          | 1B         | 1.3GB | `ollama run llama3.2:1b`         | | ||||
| @@ -62,10 +69,7 @@ Here are some example models that can be downloaded: | ||||
| | Llama 3.1          | 8B         | 4.7GB | `ollama run llama3.1`            | | ||||
| | Llama 3.1          | 405B       | 231GB | `ollama run llama3.1:405b`       | | ||||
| | Phi 4              | 14B        | 9.1GB | `ollama run phi4`                | | ||||
| | Phi 3 Mini         | 3.8B       | 2.3GB | `ollama run phi3`                | | ||||
| | Gemma 2            | 2B         | 1.6GB | `ollama run gemma2:2b`           | | ||||
| | Gemma 2            | 9B         | 5.5GB | `ollama run gemma2`              | | ||||
| | Gemma 2            | 27B        | 16GB  | `ollama run gemma2:27b`          | | ||||
| | Phi 4 Mini         | 3.8B       | 2.5GB | `ollama run phi4-mini`           | | ||||
| | Mistral            | 7B         | 4.1GB | `ollama run mistral`             | | ||||
| | Moondream 2        | 1.4B       | 829MB | `ollama run moondream`           | | ||||
| | Neural Chat        | 7B         | 4.1GB | `ollama run neural-chat`         | | ||||
| @@ -73,7 +77,7 @@ Here are some example models that can be downloaded: | ||||
| | Code Llama         | 7B         | 3.8GB | `ollama run codellama`           | | ||||
| | Llama 2 Uncensored | 7B         | 3.8GB | `ollama run llama2-uncensored`   | | ||||
| | LLaVA              | 7B         | 4.5GB | `ollama run llava`               | | ||||
| | Solar              | 10.7B      | 6.1GB | `ollama run solar`               | | ||||
| | Granite-3.2         | 8B         | 4.9GB | `ollama run granite3.2`          | | ||||
|  | ||||
| > [!NOTE] | ||||
| > You should have at least 8 GB of RAM available to run the 7B models, 16 GB to run the 13B models, and 32 GB to run the 33B models. | ||||
| @@ -92,13 +96,13 @@ Ollama supports importing GGUF models in the Modelfile: | ||||
|  | ||||
| 2. Create the model in Ollama | ||||
|  | ||||
|    ``` | ||||
|    ```shell | ||||
|    ollama create example -f Modelfile | ||||
|    ``` | ||||
|  | ||||
| 3. Run the model | ||||
|  | ||||
|    ``` | ||||
|    ```shell | ||||
|    ollama run example | ||||
|    ``` | ||||
|  | ||||
| @@ -110,7 +114,7 @@ See the [guide](docs/import.md) on importing models for more information. | ||||
|  | ||||
| Models from the Ollama library can be customized with a prompt. For example, to customize the `llama3.2` model: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama pull llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -145,13 +149,13 @@ For more information on working with a Modelfile, see the [Modelfile](docs/model | ||||
|  | ||||
| `ollama create` is used to create a model from a Modelfile. | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama create mymodel -f ./Modelfile | ||||
| ``` | ||||
|  | ||||
| ### Pull a model | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama pull llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -159,13 +163,13 @@ ollama pull llama3.2 | ||||
|  | ||||
| ### Remove a model | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama rm llama3.2 | ||||
| ``` | ||||
|  | ||||
| ### Copy a model | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama cp llama3.2 my-model | ||||
| ``` | ||||
|  | ||||
| @@ -184,37 +188,39 @@ I'm a basic program that prints the famous "Hello, world!" message to the consol | ||||
|  | ||||
| ``` | ||||
| ollama run llava "What's in this image? /Users/jmorgan/Desktop/smile.png" | ||||
| The image features a yellow smiley face, which is likely the central focus of the picture. | ||||
| ``` | ||||
|  | ||||
| > **Output**: The image features a yellow smiley face, which is likely the central focus of the picture. | ||||
|  | ||||
| ### Pass the prompt as an argument | ||||
|  | ||||
| ```shell | ||||
| ollama run llama3.2 "Summarize this file: $(cat README.md)" | ||||
| ``` | ||||
| $ ollama run llama3.2 "Summarize this file: $(cat README.md)" | ||||
|  Ollama is a lightweight, extensible framework for building and running language models on the local machine. It provides a simple API for creating, running, and managing models, as well as a library of pre-built models that can be easily used in a variety of applications. | ||||
| ``` | ||||
|  | ||||
| > **Output**: Ollama is a lightweight, extensible framework for building and running language models on the local machine. It provides a simple API for creating, running, and managing models, as well as a library of pre-built models that can be easily used in a variety of applications. | ||||
|  | ||||
| ### Show model information | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama show llama3.2 | ||||
| ``` | ||||
|  | ||||
| ### List models on your computer | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama list | ||||
| ``` | ||||
|  | ||||
| ### List which models are currently loaded | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama ps | ||||
| ``` | ||||
|  | ||||
| ### Stop a model which is currently running | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama stop llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -230,13 +236,13 @@ See the [developer guide](https://github.com/ollama/ollama/blob/main/docs/develo | ||||
|  | ||||
| Next, start the server: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ./ollama serve | ||||
| ``` | ||||
|  | ||||
| Finally, in a separate shell, run a model: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ./ollama run llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -246,7 +252,7 @@ Ollama has a REST API for running and managing models. | ||||
|  | ||||
| ### Generate a response | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl http://localhost:11434/api/generate -d '{ | ||||
|   "model": "llama3.2", | ||||
|   "prompt":"Why is the sky blue?" | ||||
| @@ -255,7 +261,7 @@ curl http://localhost:11434/api/generate -d '{ | ||||
|  | ||||
| ### Chat with a model | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl http://localhost:11434/api/chat -d '{ | ||||
|   "model": "llama3.2", | ||||
|   "messages": [ | ||||
| @@ -271,6 +277,7 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| ### Web & Desktop | ||||
|  | ||||
| - [Open WebUI](https://github.com/open-webui/open-webui) | ||||
| - [SwiftChat (macOS with ReactNative)](https://github.com/aws-samples/swift-chat) | ||||
| - [Enchanted (macOS native)](https://github.com/AugustDev/enchanted) | ||||
| - [Hollama](https://github.com/fmaclen/hollama) | ||||
| - [Lollms-Webui](https://github.com/ParisNeo/lollms-webui) | ||||
| @@ -278,12 +285,13 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [Bionic GPT](https://github.com/bionic-gpt/bionic-gpt) | ||||
| - [HTML UI](https://github.com/rtcfirefly/ollama-ui) | ||||
| - [Saddle](https://github.com/jikkuatwork/saddle) | ||||
| - [TagSpaces](https://www.tagspaces.org) (A platform for file based apps, [utilizing Ollama](https://docs.tagspaces.org/ai/) for the generation of tags and descriptions) | ||||
| - [Chatbot UI](https://github.com/ivanfioravanti/chatbot-ollama) | ||||
| - [Chatbot UI v2](https://github.com/mckaywrigley/chatbot-ui) | ||||
| - [Typescript UI](https://github.com/ollama-interface/Ollama-Gui?tab=readme-ov-file) | ||||
| - [Minimalistic React UI for Ollama Models](https://github.com/richawo/minimal-llm-ui) | ||||
| - [Ollamac](https://github.com/kevinhermawan/Ollamac) | ||||
| - [big-AGI](https://github.com/enricoros/big-AGI/blob/main/docs/config-local-ollama.md) | ||||
| - [big-AGI](https://github.com/enricoros/big-AGI)  | ||||
| - [Cheshire Cat assistant framework](https://github.com/cheshire-cat-ai/core) | ||||
| - [Amica](https://github.com/semperai/amica) | ||||
| - [chatd](https://github.com/BruceMacD/chatd) | ||||
| @@ -317,6 +325,7 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [RWKV-Runner](https://github.com/josStorer/RWKV-Runner) (RWKV offline LLM deployment tool, also usable as a client for ChatGPT and Ollama) | ||||
| - [Ollama Grid Search](https://github.com/dezoito/ollama-grid-search) (app to evaluate and compare models) | ||||
| - [Olpaka](https://github.com/Otacon/olpaka) (User-friendly Flutter Web App for Ollama) | ||||
| - [Casibase](https://casibase.org) (An open source AI knowledge base and dialogue system combining the latest RAG, SSO, ollama support and multiple large language models.) | ||||
| - [OllamaSpring](https://github.com/CrazyNeil/OllamaSpring) (Ollama Client for macOS) | ||||
| - [LLocal.in](https://github.com/kartikm7/llocal) (Easy to use Electron Desktop Client for Ollama) | ||||
| - [Shinkai Desktop](https://github.com/dcSpark/shinkai-apps) (Two click install Local AI using Ollama + Files + RAG) | ||||
| @@ -339,7 +348,7 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [PartCAD](https://github.com/openvmp/partcad/) (CAD model generation with OpenSCAD and CadQuery) | ||||
| - [Ollama4j Web UI](https://github.com/ollama4j/ollama4j-web-ui) - Java-based Web UI for Ollama built with Vaadin, Spring Boot and Ollama4j | ||||
| - [PyOllaMx](https://github.com/kspviswa/pyOllaMx) - macOS application capable of chatting with both Ollama and Apple MLX models. | ||||
| - [Claude Dev](https://github.com/saoudrizwan/claude-dev) - VSCode extension for multi-file/whole-repo coding | ||||
| - [Cline](https://github.com/cline/cline) - Formerly known as Claude Dev is a VSCode extension for multi-file/whole-repo coding | ||||
| - [Cherry Studio](https://github.com/kangfenmao/cherry-studio) (Desktop client with Ollama support) | ||||
| - [ConfiChat](https://github.com/1runeberg/confichat) (Lightweight, standalone, multi-platform, and privacy focused LLM chat interface with optional encryption) | ||||
| - [Archyve](https://github.com/nickthecook/archyve) (RAG-enabling document library) | ||||
| @@ -353,6 +362,7 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [Web management](https://github.com/lemonit-eric-mao/ollama-web-management) (Web management page) | ||||
| - [Promptery](https://github.com/promptery/promptery) (desktop client for Ollama.) | ||||
| - [Ollama App](https://github.com/JHubi1/ollama-app) (Modern and easy-to-use multi-platform client for Ollama) | ||||
| - [chat-ollama](https://github.com/annilq/chat-ollama) (a React Native client for Ollama) | ||||
| - [SpaceLlama](https://github.com/tcsenpai/spacellama) (Firefox and Chrome extension to quickly summarize web pages with ollama in a sidebar) | ||||
| - [YouLama](https://github.com/tcsenpai/youlama) (Webapp to quickly summarize any YouTube video, supporting Invidious as well) | ||||
| - [DualMind](https://github.com/tcsenpai/dualmind) (Experimental app allowing two models to talk to each other in the terminal or in a web interface) | ||||
| @@ -369,6 +379,25 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [Minima](https://github.com/dmayboroda/minima) (RAG with on-premises or fully local workflow) | ||||
| - [aidful-ollama-model-delete](https://github.com/AidfulAI/aidful-ollama-model-delete) (User interface for simplified model cleanup) | ||||
| - [Perplexica](https://github.com/ItzCrazyKns/Perplexica) (An AI-powered search engine & an open-source alternative to Perplexity AI) | ||||
| - [Ollama Chat WebUI for Docker ](https://github.com/oslook/ollama-webui) (Support for local docker deployment, lightweight ollama webui) | ||||
| - [AI Toolkit for Visual Studio Code](https://aka.ms/ai-tooklit/ollama-docs) (Microsoft-official VSCode extension to chat, test, evaluate models with Ollama support, and use them in your AI applications.) | ||||
| - [MinimalNextOllamaChat](https://github.com/anilkay/MinimalNextOllamaChat) (Minimal Web UI for Chat and Model Control) | ||||
| - [Chipper](https://github.com/TilmanGriesel/chipper) AI interface for tinkerers (Ollama, Haystack RAG, Python) | ||||
| - [ChibiChat](https://github.com/CosmicEventHorizon/ChibiChat) (Kotlin-based Android app to chat with Ollama and Koboldcpp API endpoints) | ||||
| - [LocalLLM](https://github.com/qusaismael/localllm) (Minimal Web-App to run ollama models on it with a GUI) | ||||
| - [Ollamazing](https://github.com/buiducnhat/ollamazing) (Web extension to run Ollama models) | ||||
| - [OpenDeepResearcher-via-searxng](https://github.com/benhaotang/OpenDeepResearcher-via-searxng) (A Deep Research equivent endpoint with Ollama support for running locally) | ||||
| - [AntSK](https://github.com/AIDotNet/AntSK) (Out-of-the-box & Adaptable RAG Chatbot) | ||||
| - [MaxKB](https://github.com/1Panel-dev/MaxKB/) (Ready-to-use & flexible RAG Chatbot) | ||||
| - [yla](https://github.com/danielekp/yla) (Web interface to freely interact with your customized models) | ||||
| - [LangBot](https://github.com/RockChinQ/LangBot) (LLM-based instant messaging bots platform, with Agents, RAG features, supports multiple platforms) | ||||
| - [1Panel](https://github.com/1Panel-dev/1Panel/) (Web-based Linux Server Management Tool) | ||||
| - [AstrBot](https://github.com/Soulter/AstrBot/) (User-friendly LLM-based multi-platform chatbot with a WebUI, supporting RAG, LLM agents, and plugins integration) | ||||
| - [Reins](https://github.com/ibrahimcetin/reins) (Easily tweak parameters, customize system prompts per chat, and enhance your AI experiments with reasoning model support.) | ||||
| - [Ellama](https://github.com/zeozeozeo/ellama) (Friendly native app to chat with an Ollama instance) | ||||
| - [screenpipe](https://github.com/mediar-ai/screenpipe) Build agents powered by your screen history | ||||
| - [Ollamb](https://github.com/hengkysteen/ollamb) (Simple yet rich in features, cross-platform built with Flutter and designed for Ollama. Try the [web demo](https://hengkysteen.github.io/demo/ollamb/).) | ||||
| - [Writeopia](https://github.com/Writeopia/Writeopia) (Text editor with integration with Ollama) | ||||
|  | ||||
| ### Cloud | ||||
|  | ||||
| @@ -408,10 +437,14 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [SwollamaCLI](https://github.com/marcusziade/Swollama) bundled with the Swollama Swift package. [Demo](https://github.com/marcusziade/Swollama?tab=readme-ov-file#cli-usage) | ||||
| - [aichat](https://github.com/sigoden/aichat) All-in-one LLM CLI tool featuring Shell Assistant, Chat-REPL, RAG, AI tools & agents, with access to OpenAI, Claude, Gemini, Ollama, Groq, and more. | ||||
| - [PowershAI](https://github.com/rrg92/powershai) PowerShell module that brings AI to terminal on Windows, including support for Ollama | ||||
| - [DeepShell](https://github.com/Abyss-c0re/deepshell) Your self-hosted AI assistant. Interactive Shell, Files and Folders analysis. | ||||
| - [orbiton](https://github.com/xyproto/orbiton) Configuration-free text editor and IDE with support for tab completion with Ollama. | ||||
| - [orca-cli](https://github.com/molbal/orca-cli) Ollama Registry CLI Application - Browse, pull and download models from Ollama Registry in your terminal. | ||||
| - [GGUF-to-Ollama](https://github.com/jonathanhecl/gguf-to-ollama) - Importing GGUF to Ollama made easy (multiplatform) | ||||
|  | ||||
| ### Apple Vision Pro | ||||
|  | ||||
| - [SwiftChat](https://github.com/aws-samples/swift-chat) (Cross-platform AI chat app supporting Apple Vision Pro via "Designed for iPad") | ||||
| - [Enchanted](https://github.com/AugustDev/enchanted) | ||||
|  | ||||
| ### Database | ||||
| @@ -426,9 +459,10 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
|  | ||||
| - [Pacman](https://archlinux.org/packages/extra/x86_64/ollama/) | ||||
| - [Gentoo](https://github.com/gentoo/guru/tree/master/app-misc/ollama) | ||||
| - [Homebrew](https://formulae.brew.sh/formula/ollama) | ||||
| - [Helm Chart](https://artifacthub.io/packages/helm/ollama-helm/ollama) | ||||
| - [Guix channel](https://codeberg.org/tusharhero/ollama-guix) | ||||
| - [Nix package](https://search.nixos.org/packages?channel=24.05&show=ollama&from=0&size=50&sort=relevance&type=packages&query=ollama) | ||||
| - [Nix package](https://search.nixos.org/packages?show=ollama&from=0&size=50&sort=relevance&type=packages&query=ollama) | ||||
| - [Flox](https://flox.dev/blog/ollama-part-one) | ||||
|  | ||||
| ### Libraries | ||||
| @@ -481,13 +515,21 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [GoLamify](https://github.com/prasad89/golamify) | ||||
| - [Ollama for Haskell](https://github.com/tusharad/ollama-haskell) | ||||
| - [multi-llm-ts](https://github.com/nbonamy/multi-llm-ts) (A Typescript/JavaScript library allowing access to different LLM in unified API) | ||||
| - [LlmTornado](https://github.com/lofcz/llmtornado) (C# library providing a unified interface for major FOSS & Commercial inference APIs) | ||||
| - [Ollama for Zig](https://github.com/dravenk/ollama-zig) | ||||
| - [Abso](https://github.com/lunary-ai/abso) (OpenAI-compatible TypeScript SDK for any LLM provider) | ||||
| - [Nichey](https://github.com/goodreasonai/nichey) is a Python package for generating custom wikis for your research topic | ||||
| - [Ollama for D](https://github.com/kassane/ollama-d) | ||||
|  | ||||
| ### Mobile | ||||
|  | ||||
| - [SwiftChat](https://github.com/aws-samples/swift-chat) (Lightning-fast Cross-platform AI chat app with native UI for Android, iOS and iPad) | ||||
| - [Enchanted](https://github.com/AugustDev/enchanted) | ||||
| - [Maid](https://github.com/Mobile-Artificial-Intelligence/maid) | ||||
| - [Ollama App](https://github.com/JHubi1/ollama-app) (Modern and easy-to-use multi-platform client for Ollama) | ||||
| - [ConfiChat](https://github.com/1runeberg/confichat) (Lightweight, standalone, multi-platform, and privacy focused LLM chat interface with optional encryption) | ||||
| - [Ollama Android Chat](https://github.com/sunshine0523/OllamaServer) (No need for Termux, start the Ollama service with one click on an Android device) | ||||
| - [Reins](https://github.com/ibrahimcetin/reins) (Easily tweak parameters, customize system prompts per chat, and enhance your AI experiments with reasoning model support.) | ||||
|  | ||||
| ### Extensions & Plugins | ||||
|  | ||||
| @@ -531,12 +573,18 @@ See the [API documentation](./docs/api.md) for all endpoints. | ||||
| - [TextCraft](https://github.com/suncloudsmoon/TextCraft) (Copilot in Word alternative using Ollama) | ||||
| - [Alfred Ollama](https://github.com/zeitlings/alfred-ollama) (Alfred Workflow) | ||||
| - [TextLLaMA](https://github.com/adarshM84/TextLLaMA) A Chrome Extension that helps you write emails, correct grammar, and translate into any language | ||||
| - [Simple-Discord-AI](https://github.com/zyphixor/simple-discord-ai) | ||||
| - [LLM Telegram Bot](https://github.com/innightwolfsleep/llm_telegram_bot) (telegram bot, primary for RP. Oobabooga-like buttons, [A1111](https://github.com/AUTOMATIC1111/stable-diffusion-webui) API integration e.t.c) | ||||
| - [mcp-llm](https://github.com/sammcj/mcp-llm) (MCP Server to allow LLMs to call other LLMs) | ||||
|  | ||||
| ### Supported backends | ||||
|  | ||||
| - [llama.cpp](https://github.com/ggerganov/llama.cpp) project founded by Georgi Gerganov. | ||||
|  | ||||
| ### Observability | ||||
|  | ||||
| - [Opik](https://www.comet.com/docs/opik/cookbook/ollama) is an open-source platform to debug, evaluate, and monitor your LLM applications, RAG systems, and agentic workflows with comprehensive tracing, automated evaluations, and production-ready dashboards. Opik supports native intergration to Ollama. | ||||
| - [Lunary](https://lunary.ai/docs/integrations/ollama) is the leading open-source LLM observability platform. It provides a variety of enterprise-grade features such as real-time analytics, prompt templates management, PII masking, and comprehensive agent tracing. | ||||
| - [OpenLIT](https://github.com/openlit/openlit) is an OpenTelemetry-native tool for monitoring Ollama Applications & GPUs using traces and metrics. | ||||
| - [HoneyHive](https://docs.honeyhive.ai/integrations/ollama) is an AI observability and evaluation platform for AI agents. Use HoneyHive to evaluate agent performance, interrogate failures, and monitor quality in production.  | ||||
| - [HoneyHive](https://docs.honeyhive.ai/integrations/ollama) is an AI observability and evaluation platform for AI agents. Use HoneyHive to evaluate agent performance, interrogate failures, and monitor quality in production. | ||||
| - [Langfuse](https://langfuse.com/docs/integrations/ollama) is an open source LLM observability platform that enables teams to collaboratively monitor, evaluate and debug AI applications. | ||||
| - [MLflow Tracing](https://mlflow.org/docs/latest/llms/tracing/index.html#automatic-tracing) is an open source LLM observability tool with a convenient API to log and visualize traces, making it easy to debug and evaluate GenAI applications. | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
| // repository]. | ||||
| // | ||||
| // [the API documentation]: https://github.com/ollama/ollama/blob/main/docs/api.md | ||||
| // [in the GitHub repository]: https://github.com/ollama/ollama/tree/main/examples | ||||
| // [in the GitHub repository]: https://github.com/ollama/ollama/tree/main/api/examples | ||||
| package api | ||||
|  | ||||
| import ( | ||||
| @@ -132,7 +132,7 @@ func (c *Client) do(ctx context.Context, method, path string, reqData, respData | ||||
| const maxBufferSize = 512 * format.KiloByte | ||||
|  | ||||
| func (c *Client) stream(ctx context.Context, method, path string, data any, fn func([]byte) error) error { | ||||
| 	var buf *bytes.Buffer | ||||
| 	var buf io.Reader | ||||
| 	if data != nil { | ||||
| 		bts, err := json.Marshal(data) | ||||
| 		if err != nil { | ||||
|   | ||||
| @@ -1,6 +1,13 @@ | ||||
| package api | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"net/http/httptest" | ||||
| 	"net/url" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| @@ -43,3 +50,206 @@ func TestClientFromEnvironment(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // testError represents an internal error type with status code and message | ||||
| // this is used since the error response from the server is not a standard error struct | ||||
| type testError struct { | ||||
| 	message    string | ||||
| 	statusCode int | ||||
| } | ||||
|  | ||||
| func (e testError) Error() string { | ||||
| 	return e.message | ||||
| } | ||||
|  | ||||
| func TestClientStream(t *testing.T) { | ||||
| 	testCases := []struct { | ||||
| 		name      string | ||||
| 		responses []any | ||||
| 		wantErr   string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "immediate error response", | ||||
| 			responses: []any{ | ||||
| 				testError{ | ||||
| 					message:    "test error message", | ||||
| 					statusCode: http.StatusBadRequest, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wantErr: "test error message", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "error after successful chunks, ok response", | ||||
| 			responses: []any{ | ||||
| 				ChatResponse{Message: Message{Content: "partial response 1"}}, | ||||
| 				ChatResponse{Message: Message{Content: "partial response 2"}}, | ||||
| 				testError{ | ||||
| 					message:    "mid-stream error", | ||||
| 					statusCode: http.StatusOK, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wantErr: "mid-stream error", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "successful stream completion", | ||||
| 			responses: []any{ | ||||
| 				ChatResponse{Message: Message{Content: "chunk 1"}}, | ||||
| 				ChatResponse{Message: Message{Content: "chunk 2"}}, | ||||
| 				ChatResponse{ | ||||
| 					Message:    Message{Content: "final chunk"}, | ||||
| 					Done:       true, | ||||
| 					DoneReason: "stop", | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range testCases { | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 				flusher, ok := w.(http.Flusher) | ||||
| 				if !ok { | ||||
| 					t.Fatal("expected http.Flusher") | ||||
| 				} | ||||
|  | ||||
| 				w.Header().Set("Content-Type", "application/x-ndjson") | ||||
|  | ||||
| 				for _, resp := range tc.responses { | ||||
| 					if errResp, ok := resp.(testError); ok { | ||||
| 						w.WriteHeader(errResp.statusCode) | ||||
| 						err := json.NewEncoder(w).Encode(map[string]string{ | ||||
| 							"error": errResp.message, | ||||
| 						}) | ||||
| 						if err != nil { | ||||
| 							t.Fatal("failed to encode error response:", err) | ||||
| 						} | ||||
| 						return | ||||
| 					} | ||||
|  | ||||
| 					if err := json.NewEncoder(w).Encode(resp); err != nil { | ||||
| 						t.Fatalf("failed to encode response: %v", err) | ||||
| 					} | ||||
| 					flusher.Flush() | ||||
| 				} | ||||
| 			})) | ||||
| 			defer ts.Close() | ||||
|  | ||||
| 			client := NewClient(&url.URL{Scheme: "http", Host: ts.Listener.Addr().String()}, http.DefaultClient) | ||||
|  | ||||
| 			var receivedChunks []ChatResponse | ||||
| 			err := client.stream(context.Background(), http.MethodPost, "/v1/chat", nil, func(chunk []byte) error { | ||||
| 				var resp ChatResponse | ||||
| 				if err := json.Unmarshal(chunk, &resp); err != nil { | ||||
| 					return fmt.Errorf("failed to unmarshal chunk: %w", err) | ||||
| 				} | ||||
| 				receivedChunks = append(receivedChunks, resp) | ||||
| 				return nil | ||||
| 			}) | ||||
|  | ||||
| 			if tc.wantErr != "" { | ||||
| 				if err == nil { | ||||
| 					t.Fatal("expected error but got nil") | ||||
| 				} | ||||
| 				if !strings.Contains(err.Error(), tc.wantErr) { | ||||
| 					t.Errorf("expected error containing %q, got %v", tc.wantErr, err) | ||||
| 				} | ||||
| 				return | ||||
| 			} | ||||
| 			if err != nil { | ||||
| 				t.Errorf("unexpected error: %v", err) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientDo(t *testing.T) { | ||||
| 	testCases := []struct { | ||||
| 		name     string | ||||
| 		response any | ||||
| 		wantErr  string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "immediate error response", | ||||
| 			response: testError{ | ||||
| 				message:    "test error message", | ||||
| 				statusCode: http.StatusBadRequest, | ||||
| 			}, | ||||
| 			wantErr: "test error message", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "server error response", | ||||
| 			response: testError{ | ||||
| 				message:    "internal error", | ||||
| 				statusCode: http.StatusInternalServerError, | ||||
| 			}, | ||||
| 			wantErr: "internal error", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "successful response", | ||||
| 			response: struct { | ||||
| 				ID      string `json:"id"` | ||||
| 				Success bool   `json:"success"` | ||||
| 			}{ | ||||
| 				ID:      "msg_123", | ||||
| 				Success: true, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range testCases { | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 				if errResp, ok := tc.response.(testError); ok { | ||||
| 					w.WriteHeader(errResp.statusCode) | ||||
| 					err := json.NewEncoder(w).Encode(map[string]string{ | ||||
| 						"error": errResp.message, | ||||
| 					}) | ||||
| 					if err != nil { | ||||
| 						t.Fatal("failed to encode error response:", err) | ||||
| 					} | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				w.Header().Set("Content-Type", "application/json") | ||||
| 				if err := json.NewEncoder(w).Encode(tc.response); err != nil { | ||||
| 					t.Fatalf("failed to encode response: %v", err) | ||||
| 				} | ||||
| 			})) | ||||
| 			defer ts.Close() | ||||
|  | ||||
| 			client := NewClient(&url.URL{Scheme: "http", Host: ts.Listener.Addr().String()}, http.DefaultClient) | ||||
|  | ||||
| 			var resp struct { | ||||
| 				ID      string `json:"id"` | ||||
| 				Success bool   `json:"success"` | ||||
| 			} | ||||
| 			err := client.do(context.Background(), http.MethodPost, "/v1/messages", nil, &resp) | ||||
|  | ||||
| 			if tc.wantErr != "" { | ||||
| 				if err == nil { | ||||
| 					t.Fatalf("got nil, want error %q", tc.wantErr) | ||||
| 				} | ||||
| 				if err.Error() != tc.wantErr { | ||||
| 					t.Errorf("error message mismatch: got %q, want %q", err.Error(), tc.wantErr) | ||||
| 				} | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			if err != nil { | ||||
| 				t.Fatalf("got error %q, want nil", err) | ||||
| 			} | ||||
|  | ||||
| 			if expectedResp, ok := tc.response.(struct { | ||||
| 				ID      string `json:"id"` | ||||
| 				Success bool   `json:"success"` | ||||
| 			}); ok { | ||||
| 				if resp.ID != expectedResp.ID { | ||||
| 					t.Errorf("response ID mismatch: got %q, want %q", resp.ID, expectedResp.ID) | ||||
| 				} | ||||
| 				if resp.Success != expectedResp.Success { | ||||
| 					t.Errorf("response Success mismatch: got %v, want %v", resp.Success, expectedResp.Success) | ||||
| 				} | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -2,9 +2,10 @@ | ||||
|  | ||||
| Run the examples in this directory with: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| go run example_name/main.go | ||||
| ``` | ||||
|  | ||||
| ## Chat - Chat with a model | ||||
| - [chat/main.go](chat/main.go) | ||||
|  | ||||
|   | ||||
							
								
								
									
										110
									
								
								api/types.go
									
									
									
									
									
								
							
							
						
						
									
										110
									
								
								api/types.go
									
									
									
									
									
								
							| @@ -10,6 +10,9 @@ import ( | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/ollama/ollama/envconfig" | ||||
| 	"github.com/ollama/ollama/types/model" | ||||
| ) | ||||
|  | ||||
| // StatusError is an error with an HTTP status code and message. | ||||
| @@ -79,7 +82,7 @@ type GenerateRequest struct { | ||||
|  | ||||
| 	// Options lists model-specific options. For example, temperature can be | ||||
| 	// set through this field, if the model supports it. | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| 	Options map[string]any `json:"options"` | ||||
| } | ||||
|  | ||||
| // ChatRequest describes a request sent by [Client.Chat]. | ||||
| @@ -104,7 +107,7 @@ type ChatRequest struct { | ||||
| 	Tools `json:"tools,omitempty"` | ||||
|  | ||||
| 	// Options lists model-specific options. | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| 	Options map[string]any `json:"options"` | ||||
| } | ||||
|  | ||||
| type Tools []Tool | ||||
| @@ -163,6 +166,48 @@ type Tool struct { | ||||
| 	Function ToolFunction `json:"function"` | ||||
| } | ||||
|  | ||||
| // PropertyType can be either a string or an array of strings | ||||
| type PropertyType []string | ||||
|  | ||||
| // UnmarshalJSON implements the json.Unmarshaler interface | ||||
| func (pt *PropertyType) UnmarshalJSON(data []byte) error { | ||||
| 	// Try to unmarshal as a string first | ||||
| 	var s string | ||||
| 	if err := json.Unmarshal(data, &s); err == nil { | ||||
| 		*pt = []string{s} | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	// If that fails, try to unmarshal as an array of strings | ||||
| 	var a []string | ||||
| 	if err := json.Unmarshal(data, &a); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	*pt = a | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // MarshalJSON implements the json.Marshaler interface | ||||
| func (pt PropertyType) MarshalJSON() ([]byte, error) { | ||||
| 	if len(pt) == 1 { | ||||
| 		// If there's only one type, marshal as a string | ||||
| 		return json.Marshal(pt[0]) | ||||
| 	} | ||||
| 	// Otherwise marshal as an array | ||||
| 	return json.Marshal([]string(pt)) | ||||
| } | ||||
|  | ||||
| // String returns a string representation of the PropertyType | ||||
| func (pt PropertyType) String() string { | ||||
| 	if len(pt) == 0 { | ||||
| 		return "" | ||||
| 	} | ||||
| 	if len(pt) == 1 { | ||||
| 		return pt[0] | ||||
| 	} | ||||
| 	return fmt.Sprintf("%v", []string(pt)) | ||||
| } | ||||
|  | ||||
| type ToolFunction struct { | ||||
| 	Name        string `json:"name"` | ||||
| 	Description string `json:"description"` | ||||
| @@ -170,9 +215,9 @@ type ToolFunction struct { | ||||
| 		Type       string   `json:"type"` | ||||
| 		Required   []string `json:"required"` | ||||
| 		Properties map[string]struct { | ||||
| 			Type        string   `json:"type"` | ||||
| 			Description string   `json:"description"` | ||||
| 			Enum        []string `json:"enum,omitempty"` | ||||
| 			Type        PropertyType `json:"type"` | ||||
| 			Description string       `json:"description"` | ||||
| 			Enum        []any        `json:"enum,omitempty"` | ||||
| 		} `json:"properties"` | ||||
| 	} `json:"parameters"` | ||||
| } | ||||
| @@ -258,7 +303,7 @@ type EmbedRequest struct { | ||||
| 	Truncate *bool `json:"truncate,omitempty"` | ||||
|  | ||||
| 	// Options lists model-specific options. | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| 	Options map[string]any `json:"options"` | ||||
| } | ||||
|  | ||||
| // EmbedResponse is the response from [Client.Embed]. | ||||
| @@ -284,7 +329,7 @@ type EmbeddingRequest struct { | ||||
| 	KeepAlive *Duration `json:"keep_alive,omitempty"` | ||||
|  | ||||
| 	// Options lists model-specific options. | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| 	Options map[string]any `json:"options"` | ||||
| } | ||||
|  | ||||
| // EmbeddingResponse is the response from [Client.Embeddings]. | ||||
| @@ -330,7 +375,7 @@ type ShowRequest struct { | ||||
| 	Template string `json:"template"` | ||||
| 	Verbose  bool   `json:"verbose"` | ||||
|  | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| 	Options map[string]any `json:"options"` | ||||
|  | ||||
| 	// Deprecated: set the model name with Model instead | ||||
| 	Name string `json:"name"` | ||||
| @@ -338,16 +383,18 @@ type ShowRequest struct { | ||||
|  | ||||
| // ShowResponse is the response returned from [Client.Show]. | ||||
| type ShowResponse struct { | ||||
| 	License       string         `json:"license,omitempty"` | ||||
| 	Modelfile     string         `json:"modelfile,omitempty"` | ||||
| 	Parameters    string         `json:"parameters,omitempty"` | ||||
| 	Template      string         `json:"template,omitempty"` | ||||
| 	System        string         `json:"system,omitempty"` | ||||
| 	Details       ModelDetails   `json:"details,omitempty"` | ||||
| 	Messages      []Message      `json:"messages,omitempty"` | ||||
| 	ModelInfo     map[string]any `json:"model_info,omitempty"` | ||||
| 	ProjectorInfo map[string]any `json:"projector_info,omitempty"` | ||||
| 	ModifiedAt    time.Time      `json:"modified_at,omitempty"` | ||||
| 	License       string             `json:"license,omitempty"` | ||||
| 	Modelfile     string             `json:"modelfile,omitempty"` | ||||
| 	Parameters    string             `json:"parameters,omitempty"` | ||||
| 	Template      string             `json:"template,omitempty"` | ||||
| 	System        string             `json:"system,omitempty"` | ||||
| 	Details       ModelDetails       `json:"details,omitempty"` | ||||
| 	Messages      []Message          `json:"messages,omitempty"` | ||||
| 	ModelInfo     map[string]any     `json:"model_info,omitempty"` | ||||
| 	ProjectorInfo map[string]any     `json:"projector_info,omitempty"` | ||||
| 	Tensors       []Tensor           `json:"tensors,omitempty"` | ||||
| 	Capabilities  []model.Capability `json:"capabilities,omitempty"` | ||||
| 	ModifiedAt    time.Time          `json:"modified_at,omitempty"` | ||||
| } | ||||
|  | ||||
| // CopyRequest is the request passed to [Client.Copy]. | ||||
| @@ -359,9 +406,9 @@ type CopyRequest struct { | ||||
| // PullRequest is the request passed to [Client.Pull]. | ||||
| type PullRequest struct { | ||||
| 	Model    string `json:"model"` | ||||
| 	Insecure bool   `json:"insecure,omitempty"` | ||||
| 	Username string `json:"username"` | ||||
| 	Password string `json:"password"` | ||||
| 	Insecure bool   `json:"insecure,omitempty"` // Deprecated: ignored | ||||
| 	Username string `json:"username"`           // Deprecated: ignored | ||||
| 	Password string `json:"password"`           // Deprecated: ignored | ||||
| 	Stream   *bool  `json:"stream,omitempty"` | ||||
|  | ||||
| 	// Deprecated: set the model name with Model instead | ||||
| @@ -465,6 +512,13 @@ type ModelDetails struct { | ||||
| 	QuantizationLevel string   `json:"quantization_level"` | ||||
| } | ||||
|  | ||||
| // Tensor describes the metadata for a given tensor. | ||||
| type Tensor struct { | ||||
| 	Name  string   `json:"name"` | ||||
| 	Type  string   `json:"type"` | ||||
| 	Shape []uint64 `json:"shape"` | ||||
| } | ||||
|  | ||||
| func (m *Metrics) Summary() { | ||||
| 	if m.TotalDuration > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "total duration:       %v\n", m.TotalDuration) | ||||
| @@ -493,7 +547,7 @@ func (m *Metrics) Summary() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (opts *Options) FromMap(m map[string]interface{}) error { | ||||
| func (opts *Options) FromMap(m map[string]any) error { | ||||
| 	valueOpts := reflect.ValueOf(opts).Elem() // names of the fields in the options struct | ||||
| 	typeOpts := reflect.TypeOf(opts).Elem()   // types of the fields in the options struct | ||||
|  | ||||
| @@ -550,12 +604,12 @@ func (opts *Options) FromMap(m map[string]interface{}) error { | ||||
| 				} | ||||
| 				field.SetString(val) | ||||
| 			case reflect.Slice: | ||||
| 				// JSON unmarshals to []interface{}, not []string | ||||
| 				val, ok := val.([]interface{}) | ||||
| 				// JSON unmarshals to []any, not []string | ||||
| 				val, ok := val.([]any) | ||||
| 				if !ok { | ||||
| 					return fmt.Errorf("option %q must be of type array", key) | ||||
| 				} | ||||
| 				// convert []interface{} to []string | ||||
| 				// convert []any to []string | ||||
| 				slice := make([]string, len(val)) | ||||
| 				for i, item := range val { | ||||
| 					str, ok := item.(string) | ||||
| @@ -609,7 +663,7 @@ func DefaultOptions() Options { | ||||
|  | ||||
| 		Runner: Runner{ | ||||
| 			// options set when the model is loaded | ||||
| 			NumCtx:    2048, | ||||
| 			NumCtx:    int(envconfig.ContextLength()), | ||||
| 			NumBatch:  512, | ||||
| 			NumGPU:    -1, // -1 here indicates that NumGPU should be set dynamically | ||||
| 			NumThread: 0,  // let the runtime decide | ||||
| @@ -662,7 +716,7 @@ func (d *Duration) UnmarshalJSON(b []byte) (err error) { | ||||
| } | ||||
|  | ||||
| // FormatParams converts specified parameter options to their correct types | ||||
| func FormatParams(params map[string][]string) (map[string]interface{}, error) { | ||||
| func FormatParams(params map[string][]string) (map[string]any, error) { | ||||
| 	opts := Options{} | ||||
| 	valueOpts := reflect.ValueOf(&opts).Elem() // names of the fields in the options struct | ||||
| 	typeOpts := reflect.TypeOf(opts)           // types of the fields in the options struct | ||||
| @@ -676,7 +730,7 @@ func FormatParams(params map[string][]string) (map[string]interface{}, error) { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	out := make(map[string]interface{}) | ||||
| 	out := make(map[string]any) | ||||
| 	// iterate params and set values based on json struct tags | ||||
| 	for key, vals := range params { | ||||
| 		if opt, ok := jsonOpts[key]; !ok { | ||||
|   | ||||
| @@ -134,7 +134,7 @@ func TestUseMmapParsingFromJSON(t *testing.T) { | ||||
|  | ||||
| 	for _, test := range tests { | ||||
| 		t.Run(test.name, func(t *testing.T) { | ||||
| 			var oMap map[string]interface{} | ||||
| 			var oMap map[string]any | ||||
| 			err := json.Unmarshal([]byte(test.req), &oMap) | ||||
| 			require.NoError(t, err) | ||||
| 			opts := DefaultOptions() | ||||
| @@ -231,3 +231,144 @@ func TestMessage_UnmarshalJSON(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestToolFunction_UnmarshalJSON(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name    string | ||||
| 		input   string | ||||
| 		wantErr string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "valid enum with same types", | ||||
| 			input: `{ | ||||
| 				"name": "test", | ||||
| 				"description": "test function", | ||||
| 				"parameters": { | ||||
| 					"type": "object", | ||||
| 					"required": ["test"], | ||||
| 					"properties": { | ||||
| 						"test": { | ||||
| 							"type": "string", | ||||
| 							"description": "test prop", | ||||
| 							"enum": ["a", "b", "c"] | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}`, | ||||
| 			wantErr: "", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "empty enum array", | ||||
| 			input: `{ | ||||
| 				"name": "test", | ||||
| 				"description": "test function", | ||||
| 				"parameters": { | ||||
| 					"type": "object", | ||||
| 					"required": ["test"], | ||||
| 					"properties": { | ||||
| 						"test": { | ||||
| 							"type": "string", | ||||
| 							"description": "test prop", | ||||
| 							"enum": [] | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}`, | ||||
| 			wantErr: "", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			var tf ToolFunction | ||||
| 			err := json.Unmarshal([]byte(tt.input), &tf) | ||||
|  | ||||
| 			if tt.wantErr != "" { | ||||
| 				require.Error(t, err) | ||||
| 				assert.Contains(t, err.Error(), tt.wantErr) | ||||
| 			} else { | ||||
| 				require.NoError(t, err) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPropertyType_UnmarshalJSON(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name     string | ||||
| 		input    string | ||||
| 		expected PropertyType | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name:     "string type", | ||||
| 			input:    `"string"`, | ||||
| 			expected: PropertyType{"string"}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:     "array of types", | ||||
| 			input:    `["string", "number"]`, | ||||
| 			expected: PropertyType{"string", "number"}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:     "array with single type", | ||||
| 			input:    `["string"]`, | ||||
| 			expected: PropertyType{"string"}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, test := range tests { | ||||
| 		t.Run(test.name, func(t *testing.T) { | ||||
| 			var pt PropertyType | ||||
| 			if err := json.Unmarshal([]byte(test.input), &pt); err != nil { | ||||
| 				t.Errorf("Unexpected error: %v", err) | ||||
| 			} | ||||
|  | ||||
| 			if len(pt) != len(test.expected) { | ||||
| 				t.Errorf("Length mismatch: got %v, expected %v", len(pt), len(test.expected)) | ||||
| 			} | ||||
|  | ||||
| 			for i, v := range pt { | ||||
| 				if v != test.expected[i] { | ||||
| 					t.Errorf("Value mismatch at index %d: got %v, expected %v", i, v, test.expected[i]) | ||||
| 				} | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPropertyType_MarshalJSON(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name     string | ||||
| 		input    PropertyType | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name:     "single type", | ||||
| 			input:    PropertyType{"string"}, | ||||
| 			expected: `"string"`, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:     "multiple types", | ||||
| 			input:    PropertyType{"string", "number"}, | ||||
| 			expected: `["string","number"]`, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:     "empty type", | ||||
| 			input:    PropertyType{}, | ||||
| 			expected: `[]`, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, test := range tests { | ||||
| 		t.Run(test.name, func(t *testing.T) { | ||||
| 			data, err := json.Marshal(test.input) | ||||
| 			if err != nil { | ||||
| 				t.Errorf("Unexpected error: %v", err) | ||||
| 			} | ||||
|  | ||||
| 			if string(data) != test.expected { | ||||
| 				t.Errorf("Marshaled data mismatch: got %v, expected %v", string(data), test.expected) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -17,6 +17,6 @@ If you want to build the installer, youll need to install | ||||
| In the top directory of this repo, run the following powershell script | ||||
| to build the ollama CLI, ollama app, and ollama installer. | ||||
|  | ||||
| ``` | ||||
| ```powershell | ||||
| powershell -ExecutionPolicy Bypass -File .\scripts\build_windows.ps1 | ||||
| ``` | ||||
|   | ||||
							
								
								
									
										178
									
								
								benchmark/server_benchmark_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								benchmark/server_benchmark_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | ||||
| package benchmark | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"flag" | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/ollama/ollama/api" | ||||
| ) | ||||
|  | ||||
| // Command line flags | ||||
| var modelFlag string | ||||
|  | ||||
| func init() { | ||||
| 	flag.StringVar(&modelFlag, "m", "", "Name of the model to benchmark") | ||||
| 	flag.Lookup("m").DefValue = "model" | ||||
| } | ||||
|  | ||||
| // modelName returns the model name from flags, failing the test if not set | ||||
| func modelName(b *testing.B) string { | ||||
| 	if modelFlag == "" { | ||||
| 		b.Fatal("Error: -m flag is required for benchmark tests") | ||||
| 	} | ||||
| 	return modelFlag | ||||
| } | ||||
|  | ||||
| type TestCase struct { | ||||
| 	name      string | ||||
| 	prompt    string | ||||
| 	maxTokens int | ||||
| } | ||||
|  | ||||
| // runGenerateBenchmark contains the common generate and metrics logic | ||||
| func runGenerateBenchmark(b *testing.B, ctx context.Context, client *api.Client, req *api.GenerateRequest) { | ||||
| 	start := time.Now() | ||||
| 	var ttft time.Duration | ||||
| 	var metrics api.Metrics | ||||
|  | ||||
| 	err := client.Generate(ctx, req, func(resp api.GenerateResponse) error { | ||||
| 		if ttft == 0 && resp.Response != "" { | ||||
| 			ttft = time.Since(start) | ||||
| 		} | ||||
| 		if resp.Done { | ||||
| 			metrics = resp.Metrics | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
|  | ||||
| 	// Report custom metrics as part of the benchmark results | ||||
| 	b.ReportMetric(float64(ttft.Milliseconds()), "ttft_ms") | ||||
| 	b.ReportMetric(float64(metrics.LoadDuration.Milliseconds()), "load_ms") | ||||
|  | ||||
| 	// Token throughput metrics | ||||
| 	promptThroughput := float64(metrics.PromptEvalCount) / metrics.PromptEvalDuration.Seconds() | ||||
| 	genThroughput := float64(metrics.EvalCount) / metrics.EvalDuration.Seconds() | ||||
| 	b.ReportMetric(promptThroughput, "prompt_tok/s") | ||||
| 	b.ReportMetric(genThroughput, "gen_tok/s") | ||||
|  | ||||
| 	// Token counts | ||||
| 	b.ReportMetric(float64(metrics.PromptEvalCount), "prompt_tokens") | ||||
| 	b.ReportMetric(float64(metrics.EvalCount), "gen_tokens") | ||||
| 	if err != nil { | ||||
| 		b.Fatal(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // BenchmarkColdStart runs benchmarks with model loading from cold state | ||||
| func BenchmarkColdStart(b *testing.B) { | ||||
| 	client := setup(b) | ||||
| 	tests := []TestCase{ | ||||
| 		{"short_prompt", "Write a long story", 100}, | ||||
| 		{"medium_prompt", "Write a detailed economic analysis", 500}, | ||||
| 		{"long_prompt", "Write a comprehensive AI research paper", 1000}, | ||||
| 	} | ||||
| 	m := modelName(b) | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		b.Run(fmt.Sprintf("%s/cold/%s", m, tt.name), func(b *testing.B) { | ||||
| 			ctx := context.Background() | ||||
|  | ||||
| 			// Set number of tokens as our throughput metric | ||||
| 			b.SetBytes(int64(tt.maxTokens)) | ||||
|  | ||||
| 			for b.Loop() { | ||||
| 				b.StopTimer() | ||||
| 				// Ensure model is unloaded before each iteration | ||||
| 				unload(client, m, b) | ||||
| 				b.StartTimer() | ||||
|  | ||||
| 				req := &api.GenerateRequest{ | ||||
| 					Model:   m, | ||||
| 					Prompt:  tt.prompt, | ||||
| 					Options: map[string]any{"num_predict": tt.maxTokens, "temperature": 0.1}, | ||||
| 				} | ||||
|  | ||||
| 				runGenerateBenchmark(b, ctx, client, req) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // BenchmarkWarmStart runs benchmarks with pre-loaded model | ||||
| func BenchmarkWarmStart(b *testing.B) { | ||||
| 	client := setup(b) | ||||
| 	tests := []TestCase{ | ||||
| 		{"short_prompt", "Write a long story", 100}, | ||||
| 		{"medium_prompt", "Write a detailed economic analysis", 500}, | ||||
| 		{"long_prompt", "Write a comprehensive AI research paper", 1000}, | ||||
| 	} | ||||
| 	m := modelName(b) | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		b.Run(fmt.Sprintf("%s/warm/%s", m, tt.name), func(b *testing.B) { | ||||
| 			ctx := context.Background() | ||||
|  | ||||
| 			// Pre-warm the model | ||||
| 			warmup(client, m, tt.prompt, b) | ||||
|  | ||||
| 			// Set number of tokens as our throughput metric | ||||
| 			b.SetBytes(int64(tt.maxTokens)) | ||||
|  | ||||
| 			for b.Loop() { | ||||
| 				req := &api.GenerateRequest{ | ||||
| 					Model:   m, | ||||
| 					Prompt:  tt.prompt, | ||||
| 					Options: map[string]any{"num_predict": tt.maxTokens, "temperature": 0.1}, | ||||
| 				} | ||||
|  | ||||
| 				runGenerateBenchmark(b, ctx, client, req) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // setup verifies server and model availability | ||||
| func setup(b *testing.B) *api.Client { | ||||
| 	client, err := api.ClientFromEnvironment() | ||||
| 	if err != nil { | ||||
| 		b.Fatal(err) | ||||
| 	} | ||||
| 	if _, err := client.Show(context.Background(), &api.ShowRequest{Model: modelName(b)}); err != nil { | ||||
| 		b.Fatalf("Model unavailable: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	return client | ||||
| } | ||||
|  | ||||
| // warmup ensures the model is loaded and warmed up | ||||
| func warmup(client *api.Client, model string, prompt string, b *testing.B) { | ||||
| 	for range 3 { | ||||
| 		err := client.Generate( | ||||
| 			context.Background(), | ||||
| 			&api.GenerateRequest{ | ||||
| 				Model:   model, | ||||
| 				Prompt:  prompt, | ||||
| 				Options: map[string]any{"num_predict": 50, "temperature": 0.1}, | ||||
| 			}, | ||||
| 			func(api.GenerateResponse) error { return nil }, | ||||
| 		) | ||||
| 		if err != nil { | ||||
| 			b.Logf("Error during model warm-up: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // unload forces model unloading using KeepAlive: 0 parameter | ||||
| func unload(client *api.Client, model string, b *testing.B) { | ||||
| 	req := &api.GenerateRequest{ | ||||
| 		Model:     model, | ||||
| 		KeepAlive: &api.Duration{Duration: 0}, | ||||
| 	} | ||||
| 	if err := client.Generate(context.Background(), req, func(api.GenerateResponse) error { return nil }); err != nil { | ||||
| 		b.Logf("Unload error: %v", err) | ||||
| 	} | ||||
| 	time.Sleep(1 * time.Second) | ||||
| } | ||||
							
								
								
									
										90
									
								
								cmd/cmd.go
									
									
									
									
									
								
							
							
						
						
									
										90
									
								
								cmd/cmd.go
									
									
									
									
									
								
							| @@ -18,6 +18,8 @@ import ( | ||||
| 	"os/signal" | ||||
| 	"path/filepath" | ||||
| 	"runtime" | ||||
| 	"slices" | ||||
| 	"sort" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sync/atomic" | ||||
| @@ -34,10 +36,9 @@ import ( | ||||
| 	"github.com/ollama/ollama/api" | ||||
| 	"github.com/ollama/ollama/envconfig" | ||||
| 	"github.com/ollama/ollama/format" | ||||
| 	"github.com/ollama/ollama/llama" | ||||
| 	"github.com/ollama/ollama/llama/runner" | ||||
| 	"github.com/ollama/ollama/parser" | ||||
| 	"github.com/ollama/ollama/progress" | ||||
| 	"github.com/ollama/ollama/runner" | ||||
| 	"github.com/ollama/ollama/server" | ||||
| 	"github.com/ollama/ollama/types/model" | ||||
| 	"github.com/ollama/ollama/version" | ||||
| @@ -59,7 +60,7 @@ func getModelfileName(cmd *cobra.Command) (string, error) { | ||||
|  | ||||
| 	_, err = os.Stat(absName) | ||||
| 	if err != nil { | ||||
| 		return filename, err | ||||
| 		return "", err | ||||
| 	} | ||||
|  | ||||
| 	return absName, nil | ||||
| @@ -256,6 +257,7 @@ func StopHandler(cmd *cobra.Command, args []string) error { | ||||
| 		if strings.Contains(err.Error(), "not found") { | ||||
| 			return fmt.Errorf("couldn't find model \"%s\" to stop", args[0]) | ||||
| 		} | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @@ -266,7 +268,7 @@ func RunHandler(cmd *cobra.Command, args []string) error { | ||||
| 	opts := runOptions{ | ||||
| 		Model:    args[0], | ||||
| 		WordWrap: os.Getenv("TERM") == "xterm-256color", | ||||
| 		Options:  map[string]interface{}{}, | ||||
| 		Options:  map[string]any{}, | ||||
| 	} | ||||
|  | ||||
| 	format, err := cmd.Flags().GetString("format") | ||||
| @@ -338,7 +340,21 @@ func RunHandler(cmd *cobra.Command, args []string) error { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	opts.MultiModal = len(info.ProjectorInfo) != 0 | ||||
| 	opts.MultiModal = slices.Contains(info.Capabilities, model.CapabilityVision) | ||||
|  | ||||
| 	// TODO: remove the projector info and vision info checks below, | ||||
| 	// these are left in for backwards compatibility with older servers | ||||
| 	// that don't have the capabilities field in the model info | ||||
| 	if len(info.ProjectorInfo) != 0 { | ||||
| 		opts.MultiModal = true | ||||
| 	} | ||||
| 	for k := range info.ModelInfo { | ||||
| 		if strings.Contains(k, ".vision.") { | ||||
| 			opts.MultiModal = true | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	opts.ParentModel = info.Details.ParentModel | ||||
|  | ||||
| 	if interactive { | ||||
| @@ -559,8 +575,9 @@ func ShowHandler(cmd *cobra.Command, args []string) error { | ||||
| 	parameters, errParams := cmd.Flags().GetBool("parameters") | ||||
| 	system, errSystem := cmd.Flags().GetBool("system") | ||||
| 	template, errTemplate := cmd.Flags().GetBool("template") | ||||
| 	verbose, errVerbose := cmd.Flags().GetBool("verbose") | ||||
|  | ||||
| 	for _, boolErr := range []error{errLicense, errModelfile, errParams, errSystem, errTemplate} { | ||||
| 	for _, boolErr := range []error{errLicense, errModelfile, errParams, errSystem, errTemplate, errVerbose} { | ||||
| 		if boolErr != nil { | ||||
| 			return errors.New("error retrieving flags") | ||||
| 		} | ||||
| @@ -598,7 +615,7 @@ func ShowHandler(cmd *cobra.Command, args []string) error { | ||||
| 		return errors.New("only one of '--license', '--modelfile', '--parameters', '--system', or '--template' can be specified") | ||||
| 	} | ||||
|  | ||||
| 	req := api.ShowRequest{Name: args[0]} | ||||
| 	req := api.ShowRequest{Name: args[0], Verbose: verbose} | ||||
| 	resp, err := client.Show(cmd.Context(), &req) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| @@ -621,10 +638,10 @@ func ShowHandler(cmd *cobra.Command, args []string) error { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	return showInfo(resp, os.Stdout) | ||||
| 	return showInfo(resp, verbose, os.Stdout) | ||||
| } | ||||
|  | ||||
| func showInfo(resp *api.ShowResponse, w io.Writer) error { | ||||
| func showInfo(resp *api.ShowResponse, verbose bool, w io.Writer) error { | ||||
| 	tableRender := func(header string, rows func() [][]string) { | ||||
| 		fmt.Fprintln(w, " ", header) | ||||
| 		table := tablewriter.NewWriter(w) | ||||
| @@ -658,6 +675,15 @@ func showInfo(resp *api.ShowResponse, w io.Writer) error { | ||||
| 		return | ||||
| 	}) | ||||
|  | ||||
| 	if len(resp.Capabilities) > 0 { | ||||
| 		tableRender("Capabilities", func() (rows [][]string) { | ||||
| 			for _, capability := range resp.Capabilities { | ||||
| 				rows = append(rows, []string{"", capability.String()}) | ||||
| 			} | ||||
| 			return | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	if resp.ProjectorInfo != nil { | ||||
| 		tableRender("Projector", func() (rows [][]string) { | ||||
| 			arch := resp.ProjectorInfo["general.architecture"].(string) | ||||
| @@ -681,6 +707,47 @@ func showInfo(resp *api.ShowResponse, w io.Writer) error { | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	if resp.ModelInfo != nil && verbose { | ||||
| 		tableRender("Metadata", func() (rows [][]string) { | ||||
| 			keys := make([]string, 0, len(resp.ModelInfo)) | ||||
| 			for k := range resp.ModelInfo { | ||||
| 				keys = append(keys, k) | ||||
| 			} | ||||
| 			sort.Strings(keys) | ||||
|  | ||||
| 			for _, k := range keys { | ||||
| 				var v string | ||||
| 				switch vData := resp.ModelInfo[k].(type) { | ||||
| 				case bool: | ||||
| 					v = fmt.Sprintf("%t", vData) | ||||
| 				case string: | ||||
| 					v = vData | ||||
| 				case float64: | ||||
| 					v = fmt.Sprintf("%g", vData) | ||||
| 				case []any: | ||||
| 					n := 3 | ||||
| 					if len(vData) < n { | ||||
| 						n = len(vData) | ||||
| 					} | ||||
| 					v = fmt.Sprintf("%v", vData[:n]) | ||||
| 				default: | ||||
| 					v = fmt.Sprintf("%T", vData) | ||||
| 				} | ||||
| 				rows = append(rows, []string{"", k, v}) | ||||
| 			} | ||||
| 			return | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	if len(resp.Tensors) > 0 && verbose { | ||||
| 		tableRender("Tensors", func() (rows [][]string) { | ||||
| 			for _, t := range resp.Tensors { | ||||
| 				rows = append(rows, []string{"", t.Name, t.Type, fmt.Sprint(t.Shape)}) | ||||
| 			} | ||||
| 			return | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	head := func(s string, n int) (rows [][]string) { | ||||
| 		scanner := bufio.NewScanner(strings.NewReader(s)) | ||||
| 		for scanner.Scan() && (len(rows) < n || n < 0) { | ||||
| @@ -785,7 +852,7 @@ type runOptions struct { | ||||
| 	Format      string | ||||
| 	System      string | ||||
| 	Images      []api.ImageData | ||||
| 	Options     map[string]interface{} | ||||
| 	Options     map[string]any | ||||
| 	MultiModal  bool | ||||
| 	KeepAlive   *api.Duration | ||||
| } | ||||
| @@ -1187,6 +1254,7 @@ func NewCLI() *cobra.Command { | ||||
| 	showCmd.Flags().Bool("parameters", false, "Show parameters of a model") | ||||
| 	showCmd.Flags().Bool("template", false, "Show template of a model") | ||||
| 	showCmd.Flags().Bool("system", false, "Show system message of a model") | ||||
| 	showCmd.Flags().BoolP("verbose", "v", false, "Show detailed model information") | ||||
|  | ||||
| 	runCmd := &cobra.Command{ | ||||
| 		Use:     "run MODEL [PROMPT]", | ||||
| @@ -1271,7 +1339,6 @@ func NewCLI() *cobra.Command { | ||||
|  | ||||
| 	runnerCmd := &cobra.Command{ | ||||
| 		Use:    "runner", | ||||
| 		Short:  llama.PrintSystemInfo(), | ||||
| 		Hidden: true, | ||||
| 		RunE: func(cmd *cobra.Command, args []string) error { | ||||
| 			return runner.Execute(os.Args[1:]) | ||||
| @@ -1314,7 +1381,6 @@ func NewCLI() *cobra.Command { | ||||
| 				envVars["OLLAMA_NOPRUNE"], | ||||
| 				envVars["OLLAMA_ORIGINS"], | ||||
| 				envVars["OLLAMA_SCHED_SPREAD"], | ||||
| 				envVars["OLLAMA_TMPDIR"], | ||||
| 				envVars["OLLAMA_FLASH_ATTENTION"], | ||||
| 				envVars["OLLAMA_KV_CACHE_TYPE"], | ||||
| 				envVars["OLLAMA_LLM_LIBRARY"], | ||||
|   | ||||
							
								
								
									
										319
									
								
								cmd/cmd_test.go
									
									
									
									
									
								
							
							
						
						
									
										319
									
								
								cmd/cmd_test.go
									
									
									
									
									
								
							| @@ -10,11 +10,13 @@ import ( | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	"github.com/spf13/cobra" | ||||
|  | ||||
| 	"github.com/ollama/ollama/api" | ||||
| 	"github.com/ollama/ollama/types/model" | ||||
| ) | ||||
|  | ||||
| func TestShowInfo(t *testing.T) { | ||||
| @@ -26,7 +28,7 @@ func TestShowInfo(t *testing.T) { | ||||
| 				ParameterSize:     "7B", | ||||
| 				QuantizationLevel: "FP16", | ||||
| 			}, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -56,7 +58,7 @@ func TestShowInfo(t *testing.T) { | ||||
| 				ParameterSize:     "7B", | ||||
| 				QuantizationLevel: "FP16", | ||||
| 			}, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -67,6 +69,60 @@ func TestShowInfo(t *testing.T) { | ||||
|     embedding length    0        | ||||
|     quantization        FP16     | ||||
|  | ||||
| ` | ||||
| 		if diff := cmp.Diff(expect, b.String()); diff != "" { | ||||
| 			t.Errorf("unexpected output (-want +got):\n%s", diff) | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("verbose model", func(t *testing.T) { | ||||
| 		var b bytes.Buffer | ||||
| 		if err := showInfo(&api.ShowResponse{ | ||||
| 			Details: api.ModelDetails{ | ||||
| 				Family:            "test", | ||||
| 				ParameterSize:     "8B", | ||||
| 				QuantizationLevel: "FP16", | ||||
| 			}, | ||||
| 			Parameters: ` | ||||
| 			stop up`, | ||||
| 			ModelInfo: map[string]any{ | ||||
| 				"general.architecture":    "test", | ||||
| 				"general.parameter_count": float64(8_000_000_000), | ||||
| 				"some.true_bool":          true, | ||||
| 				"some.false_bool":         false, | ||||
| 				"test.context_length":     float64(1000), | ||||
| 				"test.embedding_length":   float64(11434), | ||||
| 			}, | ||||
| 			Tensors: []api.Tensor{ | ||||
| 				{Name: "blk.0.attn_k.weight", Type: "BF16", Shape: []uint64{42, 3117}}, | ||||
| 				{Name: "blk.0.attn_q.weight", Type: "FP16", Shape: []uint64{3117, 42}}, | ||||
| 			}, | ||||
| 		}, true, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		expect := `  Model | ||||
|     architecture        test      | ||||
|     parameters          8B        | ||||
|     context length      1000      | ||||
|     embedding length    11434     | ||||
|     quantization        FP16      | ||||
|  | ||||
|   Parameters | ||||
|     stop    up     | ||||
|  | ||||
|   Metadata | ||||
|     general.architecture       test      | ||||
|     general.parameter_count    8e+09     | ||||
|     some.false_bool            false     | ||||
|     some.true_bool             true      | ||||
|     test.context_length        1000      | ||||
|     test.embedding_length      11434     | ||||
|  | ||||
|   Tensors | ||||
|     blk.0.attn_k.weight    BF16    [42 3117]     | ||||
|     blk.0.attn_q.weight    FP16    [3117 42]     | ||||
|  | ||||
| ` | ||||
| 		if diff := cmp.Diff(expect, b.String()); diff != "" { | ||||
| 			t.Errorf("unexpected output (-want +got):\n%s", diff) | ||||
| @@ -88,7 +144,7 @@ func TestShowInfo(t *testing.T) { | ||||
| 			stop you | ||||
| 			stop up | ||||
| 			temperature 99`, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -125,7 +181,7 @@ func TestShowInfo(t *testing.T) { | ||||
| 				"clip.vision.embedding_length": float64(0), | ||||
| 				"clip.vision.projection_dim":   float64(0), | ||||
| 			}, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -158,7 +214,7 @@ func TestShowInfo(t *testing.T) { | ||||
| Ahoy, matey! | ||||
| Weigh anchor! | ||||
| 			`, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -187,7 +243,7 @@ Weigh anchor! | ||||
| 				QuantizationLevel: "FP16", | ||||
| 			}, | ||||
| 			License: license, | ||||
| 		}, &b); err != nil { | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| @@ -205,6 +261,34 @@ Weigh anchor! | ||||
| 			t.Errorf("unexpected output (-want +got):\n%s", diff) | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("capabilities", func(t *testing.T) { | ||||
| 		var b bytes.Buffer | ||||
| 		if err := showInfo(&api.ShowResponse{ | ||||
| 			Details: api.ModelDetails{ | ||||
| 				Family:            "test", | ||||
| 				ParameterSize:     "7B", | ||||
| 				QuantizationLevel: "FP16", | ||||
| 			}, | ||||
| 			Capabilities: []model.Capability{model.CapabilityVision, model.CapabilityTools}, | ||||
| 		}, false, &b); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		expect := "  Model\n" + | ||||
| 			"    architecture    test    \n" + | ||||
| 			"    parameters      7B      \n" + | ||||
| 			"    quantization    FP16    \n" + | ||||
| 			"\n" + | ||||
| 			"  Capabilities\n" + | ||||
| 			"    vision    \n" + | ||||
| 			"    tools     \n" + | ||||
| 			"\n" | ||||
|  | ||||
| 		if diff := cmp.Diff(expect, b.String()); diff != "" { | ||||
| 			t.Errorf("unexpected output (-want +got):\n%s", diff) | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestDeleteHandler(t *testing.T) { | ||||
| @@ -279,7 +363,7 @@ func TestGetModelfileName(t *testing.T) { | ||||
| 			name:          "no modelfile specified, no modelfile exists", | ||||
| 			modelfileName: "", | ||||
| 			fileExists:    false, | ||||
| 			expectedName:  "Modelfile", | ||||
| 			expectedName:  "", | ||||
| 			expectedErr:   os.ErrNotExist, | ||||
| 		}, | ||||
| 		{ | ||||
| @@ -293,7 +377,7 @@ func TestGetModelfileName(t *testing.T) { | ||||
| 			name:          "modelfile specified, no modelfile exists", | ||||
| 			modelfileName: "crazyfile", | ||||
| 			fileExists:    false, | ||||
| 			expectedName:  "crazyfile", | ||||
| 			expectedName:  "", | ||||
| 			expectedErr:   os.ErrNotExist, | ||||
| 		}, | ||||
| 		{ | ||||
| @@ -490,6 +574,96 @@ func TestPushHandler(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestListHandler(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name           string | ||||
| 		args           []string | ||||
| 		serverResponse []api.ListModelResponse | ||||
| 		expectedError  string | ||||
| 		expectedOutput string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "list all models", | ||||
| 			args: []string{}, | ||||
| 			serverResponse: []api.ListModelResponse{ | ||||
| 				{Name: "model1", Digest: "sha256:abc123", Size: 1024, ModifiedAt: time.Now().Add(-24 * time.Hour)}, | ||||
| 				{Name: "model2", Digest: "sha256:def456", Size: 2048, ModifiedAt: time.Now().Add(-48 * time.Hour)}, | ||||
| 			}, | ||||
| 			expectedOutput: "NAME      ID              SIZE      MODIFIED     \n" + | ||||
| 				"model1    sha256:abc12    1.0 KB    24 hours ago    \n" + | ||||
| 				"model2    sha256:def45    2.0 KB    2 days ago      \n", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "filter models by prefix", | ||||
| 			args: []string{"model1"}, | ||||
| 			serverResponse: []api.ListModelResponse{ | ||||
| 				{Name: "model1", Digest: "sha256:abc123", Size: 1024, ModifiedAt: time.Now().Add(-24 * time.Hour)}, | ||||
| 				{Name: "model2", Digest: "sha256:def456", Size: 2048, ModifiedAt: time.Now().Add(-24 * time.Hour)}, | ||||
| 			}, | ||||
| 			expectedOutput: "NAME      ID              SIZE      MODIFIED     \n" + | ||||
| 				"model1    sha256:abc12    1.0 KB    24 hours ago    \n", | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:          "server error", | ||||
| 			args:          []string{}, | ||||
| 			expectedError: "server error", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			mockServer := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 				if r.URL.Path != "/api/tags" || r.Method != http.MethodGet { | ||||
| 					t.Errorf("unexpected request to %s %s", r.Method, r.URL.Path) | ||||
| 					http.Error(w, "not found", http.StatusNotFound) | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				if tt.expectedError != "" { | ||||
| 					http.Error(w, tt.expectedError, http.StatusInternalServerError) | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				response := api.ListResponse{Models: tt.serverResponse} | ||||
| 				if err := json.NewEncoder(w).Encode(response); err != nil { | ||||
| 					t.Fatal(err) | ||||
| 				} | ||||
| 			})) | ||||
| 			defer mockServer.Close() | ||||
|  | ||||
| 			t.Setenv("OLLAMA_HOST", mockServer.URL) | ||||
|  | ||||
| 			cmd := &cobra.Command{} | ||||
| 			cmd.SetContext(context.TODO()) | ||||
|  | ||||
| 			// Capture stdout | ||||
| 			oldStdout := os.Stdout | ||||
| 			r, w, _ := os.Pipe() | ||||
| 			os.Stdout = w | ||||
|  | ||||
| 			err := ListHandler(cmd, tt.args) | ||||
|  | ||||
| 			// Restore stdout and get output | ||||
| 			w.Close() | ||||
| 			os.Stdout = oldStdout | ||||
| 			output, _ := io.ReadAll(r) | ||||
|  | ||||
| 			if tt.expectedError == "" { | ||||
| 				if err != nil { | ||||
| 					t.Errorf("expected no error, got %v", err) | ||||
| 				} | ||||
| 				if got := string(output); got != tt.expectedOutput { | ||||
| 					t.Errorf("expected output:\n%s\ngot:\n%s", tt.expectedOutput, got) | ||||
| 				} | ||||
| 			} else { | ||||
| 				if err == nil || !strings.Contains(err.Error(), tt.expectedError) { | ||||
| 					t.Errorf("expected error containing %q, got %v", tt.expectedError, err) | ||||
| 				} | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCreateHandler(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name           string | ||||
| @@ -616,3 +790,132 @@ func TestCreateHandler(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewCreateRequest(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name     string | ||||
| 		from     string | ||||
| 		opts     runOptions | ||||
| 		expected *api.CreateRequest | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"basic test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "", | ||||
| 				Prompt:      "You are a fun AI agent", | ||||
| 				Messages:    []api.Message{}, | ||||
| 				WordWrap:    true, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:  "mymodel", | ||||
| 				Model: "newmodel", | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"parent model test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "parentmodel", | ||||
| 				Messages:    []api.Message{}, | ||||
| 				WordWrap:    true, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:  "parentmodel", | ||||
| 				Model: "newmodel", | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"parent model as filepath test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "/some/file/like/etc/passwd", | ||||
| 				Messages:    []api.Message{}, | ||||
| 				WordWrap:    true, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:  "mymodel", | ||||
| 				Model: "newmodel", | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"parent model as windows filepath test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "D:\\some\\file\\like\\etc\\passwd", | ||||
| 				Messages:    []api.Message{}, | ||||
| 				WordWrap:    true, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:  "mymodel", | ||||
| 				Model: "newmodel", | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"options test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "parentmodel", | ||||
| 				Options: map[string]any{ | ||||
| 					"temperature": 1.0, | ||||
| 				}, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:  "parentmodel", | ||||
| 				Model: "newmodel", | ||||
| 				Parameters: map[string]any{ | ||||
| 					"temperature": 1.0, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"messages test", | ||||
| 			"newmodel", | ||||
| 			runOptions{ | ||||
| 				Model:       "mymodel", | ||||
| 				ParentModel: "parentmodel", | ||||
| 				System:      "You are a fun AI agent", | ||||
| 				Messages: []api.Message{ | ||||
| 					{ | ||||
| 						Role:    "user", | ||||
| 						Content: "hello there!", | ||||
| 					}, | ||||
| 					{ | ||||
| 						Role:    "assistant", | ||||
| 						Content: "hello to you!", | ||||
| 					}, | ||||
| 				}, | ||||
| 				WordWrap: true, | ||||
| 			}, | ||||
| 			&api.CreateRequest{ | ||||
| 				From:   "parentmodel", | ||||
| 				Model:  "newmodel", | ||||
| 				System: "You are a fun AI agent", | ||||
| 				Messages: []api.Message{ | ||||
| 					{ | ||||
| 						Role:    "user", | ||||
| 						Content: "hello there!", | ||||
| 					}, | ||||
| 					{ | ||||
| 						Role:    "assistant", | ||||
| 						Content: "hello to you!", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			actual := NewCreateRequest(tt.from, tt.opts) | ||||
| 			if !cmp.Equal(actual, tt.expected) { | ||||
| 				t.Errorf("expected output %#v, got %#v", tt.expected, actual) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -18,6 +18,7 @@ import ( | ||||
| 	"github.com/ollama/ollama/envconfig" | ||||
| 	"github.com/ollama/ollama/readline" | ||||
| 	"github.com/ollama/ollama/types/errtypes" | ||||
| 	"github.com/ollama/ollama/types/model" | ||||
| ) | ||||
|  | ||||
| type MultilineState int | ||||
| @@ -195,6 +196,10 @@ func generateInteractive(cmd *cobra.Command, opts runOptions) error { | ||||
| 			opts.Messages = []api.Message{} | ||||
| 			fmt.Printf("Loading model '%s'\n", opts.Model) | ||||
| 			if err := loadOrUnloadModel(cmd, &opts); err != nil { | ||||
| 				if strings.Contains(err.Error(), "not found") { | ||||
| 					fmt.Printf("error: %v\n", err) | ||||
| 					continue | ||||
| 				} | ||||
| 				return err | ||||
| 			} | ||||
| 			continue | ||||
| @@ -343,7 +348,7 @@ func generateInteractive(cmd *cobra.Command, opts runOptions) error { | ||||
|  | ||||
| 				switch args[1] { | ||||
| 				case "info": | ||||
| 					_ = showInfo(resp, os.Stderr) | ||||
| 					_ = showInfo(resp, false, os.Stderr) | ||||
| 				case "license": | ||||
| 					if resp.License == "" { | ||||
| 						fmt.Println("No license was specified for this model.") | ||||
| @@ -455,9 +460,16 @@ func generateInteractive(cmd *cobra.Command, opts runOptions) error { | ||||
| } | ||||
|  | ||||
| func NewCreateRequest(name string, opts runOptions) *api.CreateRequest { | ||||
| 	parentModel := opts.ParentModel | ||||
|  | ||||
| 	modelName := model.ParseName(parentModel) | ||||
| 	if !modelName.IsValid() { | ||||
| 		parentModel = "" | ||||
| 	} | ||||
|  | ||||
| 	req := &api.CreateRequest{ | ||||
| 		Name: name, | ||||
| 		From: cmp.Or(opts.ParentModel, opts.Model), | ||||
| 		Model: name, | ||||
| 		From:  cmp.Or(parentModel, opts.Model), | ||||
| 	} | ||||
|  | ||||
| 	if opts.System != "" { | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llama/runner" | ||||
| 	"github.com/ollama/ollama/runner" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
|   | ||||
| @@ -9,12 +9,17 @@ import ( | ||||
| 	"log/slog" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type ModelParameters struct { | ||||
| 	Architectures []string `json:"architectures"` | ||||
| 	VocabSize     uint32   `json:"vocab_size"` | ||||
| 	Architectures []string       `json:"architectures"` | ||||
| 	VocabSize     uint32         `json:"vocab_size"` | ||||
| 	TextModel     TextParameters `json:"text_config"` | ||||
| } | ||||
|  | ||||
| type TextParameters struct { | ||||
| 	VocabSize uint32 `json:"vocab_size"` | ||||
| } | ||||
|  | ||||
| type AdapterParameters struct { | ||||
| @@ -27,8 +32,8 @@ type AdapterParameters struct { | ||||
| 	} `json:"lora_parameters"` | ||||
| } | ||||
|  | ||||
| func (ModelParameters) KV(t *Tokenizer) llm.KV { | ||||
| 	kv := llm.KV{ | ||||
| func (ModelParameters) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := ggml.KV{ | ||||
| 		"general.file_type":            uint32(1), | ||||
| 		"general.quantization_version": uint32(2), | ||||
| 		"tokenizer.ggml.pre":           t.Pre, | ||||
| @@ -54,7 +59,7 @@ func (ModelParameters) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p AdapterParameters) KV() llm.KV { | ||||
| func (p AdapterParameters) KV() ggml.KV { | ||||
| 	var alpha float32 | ||||
| 	if p.LoraParameters.Alpha == 0 { | ||||
| 		alpha = float32(p.Alpha) | ||||
| @@ -62,7 +67,7 @@ func (p AdapterParameters) KV() llm.KV { | ||||
| 		alpha = p.LoraParameters.Alpha | ||||
| 	} | ||||
|  | ||||
| 	kv := llm.KV{ | ||||
| 	kv := ggml.KV{ | ||||
| 		"adapter.lora.alpha": alpha, | ||||
| 		"adapter.type":       "lora", | ||||
| 		"general.file_type":  uint32(1), | ||||
| @@ -79,19 +84,19 @@ func (ModelParameters) specialTokenTypes() []string { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (ModelParameters) writeFile(ws io.WriteSeeker, kv llm.KV, ts []llm.Tensor) error { | ||||
| 	return llm.WriteGGUF(ws, kv, ts) | ||||
| func (ModelParameters) writeFile(ws io.WriteSeeker, kv ggml.KV, ts []ggml.Tensor) error { | ||||
| 	return ggml.WriteGGUF(ws, kv, ts) | ||||
| } | ||||
|  | ||||
| func (AdapterParameters) writeFile(ws io.WriteSeeker, kv llm.KV, ts []llm.Tensor) error { | ||||
| 	return llm.WriteGGUF(ws, kv, ts) | ||||
| func (AdapterParameters) writeFile(ws io.WriteSeeker, kv ggml.KV, ts []ggml.Tensor) error { | ||||
| 	return ggml.WriteGGUF(ws, kv, ts) | ||||
| } | ||||
|  | ||||
| type ModelConverter interface { | ||||
| 	// KV maps parameters to LLM key-values | ||||
| 	KV(*Tokenizer) llm.KV | ||||
| 	KV(*Tokenizer) ggml.KV | ||||
| 	// Tensors maps input tensors to LLM tensors. Model specific modifications can be done here. | ||||
| 	Tensors([]Tensor) []llm.Tensor | ||||
| 	Tensors([]Tensor) []ggml.Tensor | ||||
| 	// Replacements returns a list of string pairs to replace in tensor names. | ||||
| 	// See [strings.Replacer](https://pkg.go.dev/strings#Replacer) for details | ||||
| 	Replacements() []string | ||||
| @@ -99,7 +104,7 @@ type ModelConverter interface { | ||||
| 	// specialTokenTypes returns any special token types the model uses | ||||
| 	specialTokenTypes() []string | ||||
| 	// writeFile writes the model to the provided io.WriteSeeker | ||||
| 	writeFile(io.WriteSeeker, llm.KV, []llm.Tensor) error | ||||
| 	writeFile(io.WriteSeeker, ggml.KV, []ggml.Tensor) error | ||||
| } | ||||
|  | ||||
| type moreParser interface { | ||||
| @@ -108,17 +113,17 @@ type moreParser interface { | ||||
|  | ||||
| type AdapterConverter interface { | ||||
| 	// KV maps parameters to LLM key-values | ||||
| 	KV(llm.KV) llm.KV | ||||
| 	KV(ggml.KV) ggml.KV | ||||
| 	// Tensors maps input tensors to LLM tensors. Adapter specific modifications can be done here. | ||||
| 	Tensors([]Tensor) []llm.Tensor | ||||
| 	Tensors([]Tensor) []ggml.Tensor | ||||
| 	// Replacements returns a list of string pairs to replace in tensor names. | ||||
| 	// See [strings.Replacer](https://pkg.go.dev/strings#Replacer) for details | ||||
| 	Replacements() []string | ||||
|  | ||||
| 	writeFile(io.WriteSeeker, llm.KV, []llm.Tensor) error | ||||
| 	writeFile(io.WriteSeeker, ggml.KV, []ggml.Tensor) error | ||||
| } | ||||
|  | ||||
| func ConvertAdapter(fsys fs.FS, ws io.WriteSeeker, baseKV llm.KV) error { | ||||
| func ConvertAdapter(fsys fs.FS, ws io.WriteSeeker, baseKV ggml.KV) error { | ||||
| 	bts, err := fs.ReadFile(fsys, "adapter_config.json") | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| @@ -177,22 +182,28 @@ func ConvertModel(fsys fs.FS, ws io.WriteSeeker) error { | ||||
|  | ||||
| 	var conv ModelConverter | ||||
| 	switch p.Architectures[0] { | ||||
| 	case "LlamaForCausalLM", "MistralForCausalLM": | ||||
| 	case "LlamaForCausalLM": | ||||
| 		conv = &llamaModel{} | ||||
| 	case "Mistral3ForConditionalGeneration": | ||||
| 		conv = &mistral3Model{} | ||||
| 	case "MixtralForCausalLM": | ||||
| 		conv = &mixtralModel{} | ||||
| 	case "GemmaForCausalLM": | ||||
| 		conv = &gemmaModel{} | ||||
| 	case "Gemma2ForCausalLM": | ||||
| 		conv = &gemma2Model{} | ||||
| 	case "Gemma3ForCausalLM", "Gemma3ForConditionalGeneration": | ||||
| 		conv = &gemma3Model{Architecture: p.Architectures[0]} | ||||
| 	case "Phi3ForCausalLM": | ||||
| 		conv = &phi3Model{} | ||||
| 	case "Qwen2ForCausalLM": | ||||
| 		conv = &qwen2Model{} | ||||
| 	case "BertModel": | ||||
| 		conv = &bertModel{} | ||||
| 	case "CohereForCausalLM": | ||||
| 		conv = &commandrModel{} | ||||
| 	default: | ||||
| 		return errors.New("unsupported architecture") | ||||
| 		return fmt.Errorf("unsupported architecture %q", p.Architectures[0]) | ||||
| 	} | ||||
|  | ||||
| 	if err := json.Unmarshal(bts, conv); err != nil { | ||||
| @@ -211,7 +222,14 @@ func ConvertModel(fsys fs.FS, ws io.WriteSeeker) error { | ||||
| 	} | ||||
|  | ||||
| 	vocabSize := int(p.VocabSize) | ||||
| 	if vocabSize == 0 { | ||||
| 		tVocabSize := int(p.TextModel.VocabSize) | ||||
| 		vocabSize = tVocabSize | ||||
| 	} | ||||
|  | ||||
| 	switch { | ||||
| 	case vocabSize == 0: | ||||
| 		slog.Warn("vocabulary size was not explicitly set by the model", "default size", len(t.Vocabulary.Tokens)) | ||||
| 	case vocabSize > len(t.Vocabulary.Tokens): | ||||
| 		slog.Warn("vocabulary is smaller than expected, padding with dummy tokens", "expect", vocabSize, "actual", len(t.Vocabulary.Tokens)) | ||||
| 		for i := range vocabSize - len(t.Vocabulary.Tokens) { | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import ( | ||||
| 	"slices" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type bertModel struct { | ||||
| @@ -85,7 +85,7 @@ func (p *bertModel) parseMore(fsys fs.FS) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (p *bertModel) KV(t *Tokenizer) llm.KV { | ||||
| func (p *bertModel) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "bert" | ||||
| 	kv["bert.attention.causal"] = false | ||||
| @@ -132,8 +132,8 @@ func (p *bertModel) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *bertModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (p *bertModel) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		if slices.Contains([]string{ | ||||
| 			"embeddings.position_ids", | ||||
| @@ -143,7 +143,7 @@ func (p *bertModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
							
								
								
									
										76
									
								
								convert/convert_commandr.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								convert/convert_commandr.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| package convert | ||||
|  | ||||
| import ( | ||||
| 	"cmp" | ||||
|  | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type commandrModel struct { | ||||
| 	ModelParameters | ||||
| 	MaxPositionEmbeddings uint32  `json:"max_position_embeddings"` | ||||
| 	HiddenSize            uint32  `json:"hidden_size"` | ||||
| 	HiddenLayers          uint32  `json:"num_hidden_layers"` | ||||
| 	IntermediateSize      uint32  `json:"intermediate_size"` | ||||
| 	NumAttentionHeads     uint32  `json:"num_attention_heads"` | ||||
| 	NumKeyValueHeads      uint32  `json:"num_key_value_heads"` | ||||
| 	LayerNormEPS          float32 `json:"layer_norm_eps"` | ||||
| 	RopeTheta             float32 `json:"rope_theta"` | ||||
| 	UseQKNorm             bool    `json:"use_qk_norm"` | ||||
| 	MaxLength             uint32  `json:"model_max_length"` | ||||
| 	LogitScale            float32 `json:"logit_scale"` | ||||
| 	NCtx                  uint32  `json:"n_ctx"` | ||||
| } | ||||
|  | ||||
| var _ ModelConverter = (*commandrModel)(nil) | ||||
|  | ||||
| func (p *commandrModel) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "command-r" | ||||
| 	kv["general.name"] = "command-r" | ||||
| 	kv["command-r.context_length"] = cmp.Or(p.MaxLength, p.MaxPositionEmbeddings, p.NCtx) | ||||
| 	kv["command-r.embedding_length"] = p.HiddenSize | ||||
| 	kv["command-r.block_count"] = p.HiddenLayers | ||||
| 	kv["command-r.feed_forward_length"] = p.IntermediateSize | ||||
| 	kv["command-r.attention.head_count"] = p.NumAttentionHeads | ||||
| 	kv["command-r.attention.head_count_kv"] = p.NumKeyValueHeads | ||||
| 	kv["command-r.attention.layer_norm_epsilon"] = p.LayerNormEPS | ||||
| 	kv["command-r.rope.freq_base"] = p.RopeTheta | ||||
| 	kv["command-r.max_position_embeddings"] = cmp.Or(p.MaxLength, p.MaxPositionEmbeddings) | ||||
| 	kv["command-r.logit_scale"] = p.LogitScale | ||||
| 	kv["command-r.rope.scaling.type"] = "none" | ||||
|  | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *commandrModel) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
| 			WriterTo: t, | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	return out | ||||
| } | ||||
|  | ||||
| func (p *commandrModel) Replacements() []string { | ||||
| 	return []string{ | ||||
| 		"self_attn.q_norm", "attn_q_norm", | ||||
| 		"self_attn.k_norm", "attn_k_norm", | ||||
| 		"model.layers", "blk", | ||||
| 		"input_layernorm", "attn_norm", | ||||
| 		"mlp.down_proj", "ffn_down", | ||||
| 		"mlp.gate_proj", "ffn_gate", | ||||
| 		"mlp.up_proj", "ffn_up", | ||||
| 		"self_attn.k_proj", "attn_k", | ||||
| 		"self_attn.o_proj", "attn_output", | ||||
| 		"self_attn.q_proj", "attn_q", | ||||
| 		"self_attn.v_proj", "attn_v", | ||||
| 		"model.norm", "output_norm", | ||||
| 		"model.embed_tokens", "token_embd", | ||||
| 	} | ||||
| } | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"github.com/pdevine/tensor" | ||||
| 	"github.com/pdevine/tensor/native" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type gemmaModel struct { | ||||
| @@ -23,7 +23,7 @@ type gemmaModel struct { | ||||
|  | ||||
| var _ ModelConverter = (*gemmaModel)(nil) | ||||
|  | ||||
| func (p *gemmaModel) KV(t *Tokenizer) llm.KV { | ||||
| func (p *gemmaModel) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "gemma" | ||||
| 	kv["gemma.context_length"] = p.MaxPositionEmbeddings | ||||
| @@ -42,14 +42,14 @@ func (p *gemmaModel) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *gemmaModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (p *gemmaModel) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		if strings.HasSuffix(t.Name(), "_norm.weight") { | ||||
| 		if !strings.HasPrefix(t.Name(), "v.") && strings.HasSuffix(t.Name(), "_norm.weight") { | ||||
| 			t.SetRepacker(p.addOne) | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| package convert | ||||
|  | ||||
| import ( | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| ) | ||||
| import "github.com/ollama/ollama/fs/ggml" | ||||
|  | ||||
| type gemma2Model struct { | ||||
| 	gemmaModel | ||||
| @@ -11,7 +9,7 @@ type gemma2Model struct { | ||||
| 	FinalLogitSoftcap     float32 `json:"final_logit_softcapping"` | ||||
| } | ||||
|  | ||||
| func (p *gemma2Model) KV(t *Tokenizer) llm.KV { | ||||
| func (p *gemma2Model) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "gemma2" | ||||
| 	kv["gemma2.context_length"] = p.MaxPositionEmbeddings | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"github.com/pdevine/tensor" | ||||
| 	"github.com/pdevine/tensor/native" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type gemma2Adapter struct { | ||||
| @@ -15,14 +15,14 @@ type gemma2Adapter struct { | ||||
|  | ||||
| var _ AdapterConverter = (*gemma2Adapter)(nil) | ||||
|  | ||||
| func (p *gemma2Adapter) KV(baseKV llm.KV) llm.KV { | ||||
| func (p *gemma2Adapter) KV(baseKV ggml.KV) ggml.KV { | ||||
| 	kv := p.AdapterParameters.KV() | ||||
| 	kv["general.architecture"] = "gemma2" | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *gemma2Adapter) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (p *gemma2Adapter) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		shape := t.Shape() | ||||
| 		if (strings.HasSuffix(t.Name(), "weight.lora_a") && shape[0] > shape[1]) || | ||||
| @@ -31,7 +31,7 @@ func (p *gemma2Adapter) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 			t.SetRepacker(p.repack) | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
							
								
								
									
										142
									
								
								convert/convert_gemma3.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								convert/convert_gemma3.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | ||||
| package convert | ||||
|  | ||||
| import ( | ||||
| 	"cmp" | ||||
|  | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type gemma3Model struct { | ||||
| 	gemmaModel | ||||
| 	Architecture string | ||||
| 	TextModel    struct { | ||||
| 		HeadDim          uint32 `json:"head_dim"` | ||||
| 		HiddenSize       uint32 `json:"hidden_size"` | ||||
| 		HiddenLayers     uint32 `json:"num_hidden_layers"` | ||||
| 		IntermediateSize uint32 `json:"intermediate_size"` | ||||
| 		SlidingWindow    uint32 `json:"sliding_window"` | ||||
| 	} `json:"text_config"` | ||||
| 	VisionModel struct { | ||||
| 		NumAttentionHeads uint32  `json:"num_attention_heads"` // attention.head_count 16 | ||||
| 		LayerNormEpsilon  float32 `json:"layer_norm_eps"`      // attention.layer_norm_epsilon 1e-05 | ||||
| 		NumHiddenLayers   uint32  `json:"num_hidden_layers"`   // block_count 32 | ||||
| 		HiddenSize        uint32  `json:"hidden_size"`         // embedding_length 1280 | ||||
| 		IntermediateSize  uint32  `json:"intermediate_size"`   // feed_forward_length 5120 | ||||
| 		ImageSize         uint32  `json:"image_size"`          // image_size 560 | ||||
| 		NumChannels       uint32  `json:"num_channels"`        // num_channels 3 | ||||
| 		PatchSize         uint32  `json:"patch_size"`          // patch_size 14 | ||||
| 	} `json:"vision_config"` | ||||
| 	MaxPositionEmbeddings    uint32  `json:"max_position_embeddings"` | ||||
| 	NumAttentionHeads        uint32  `json:"num_attention_heads"` | ||||
| 	NumKeyValueHeads         uint32  `json:"num_key_value_heads"` | ||||
| 	RMSNormEPS               float32 `json:"rms_norm_eps"` | ||||
| 	HeadDim                  uint32  `json:"head_dim"` | ||||
| 	FinalLogitSoftcap        float32 `json:"final_logit_softcapping"` | ||||
| 	RopeLocalTheta           float32 `json:"rope_local_base_freq"` | ||||
| 	RopeGlobalTheta          float32 `json:"rope_global_base_freq"` | ||||
| 	SlidingWindow            uint32  `json:"sliding_window"` | ||||
| 	MultiModalTokensPerImage uint32  `json:"mm_tokens_per_image"` | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	gemma4BLayerCount  = 34 | ||||
| 	gemma12BLayerCount = 48 | ||||
| 	gemma27BLayerCount = 62 | ||||
| ) | ||||
|  | ||||
| func (p *gemma3Model) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "gemma3" | ||||
|  | ||||
| 	numBlocks := cmp.Or(p.HiddenLayers, p.TextModel.HiddenLayers) | ||||
| 	kv["gemma3.block_count"] = numBlocks | ||||
|  | ||||
| 	var ( | ||||
| 		numHeads   uint32 | ||||
| 		numKVHeads uint32 | ||||
| 	) | ||||
|  | ||||
| 	switch numBlocks { | ||||
| 	case gemma4BLayerCount: | ||||
| 		numHeads = 8 | ||||
| 		numKVHeads = 4 | ||||
| 	case gemma12BLayerCount: | ||||
| 		numHeads = 16 | ||||
| 		numKVHeads = 8 | ||||
| 	case gemma27BLayerCount: | ||||
| 		numHeads = 32 | ||||
| 		numKVHeads = 16 | ||||
| 	default: | ||||
| 		numHeads = p.NumAttentionHeads | ||||
| 		numKVHeads = p.NumKeyValueHeads | ||||
| 	} | ||||
|  | ||||
| 	kv["gemma3.attention.head_count"] = numHeads | ||||
| 	kv["gemma3.attention.head_count_kv"] = numKVHeads | ||||
|  | ||||
| 	switch p.Architecture { | ||||
| 	case "Gemma3ForCausalLM": | ||||
| 		kv["gemma3.context_length"] = p.MaxPositionEmbeddings | ||||
| 		kv["gemma3.attention.layer_norm_rms_epsilon"] = p.RMSNormEPS | ||||
| 		kv["gemma3.attention.key_length"] = p.HeadDim | ||||
| 		kv["gemma3.attention.value_length"] = p.HeadDim | ||||
| 		kv["gemma3.attention.sliding_window"] = p.SlidingWindow | ||||
| 		kv["gemma3.final_logit_softcapping"] = cmp.Or(p.FinalLogitSoftcap, 30) | ||||
| 		kv["gemma3.rope.local.freq_base"] = cmp.Or(p.RopeLocalTheta, 10000.0) | ||||
| 		kv["gemma3.rope.global.freq_base"] = cmp.Or(p.RopeGlobalTheta, 1000000.0) | ||||
| 		kv["gemma3.embedding_length"] = p.HiddenSize | ||||
| 		kv["gemma3.feed_forward_length"] = p.IntermediateSize | ||||
| 	default: | ||||
| 		kv["gemma3.context_length"] = cmp.Or(p.MaxPositionEmbeddings, 131072) | ||||
| 		kv["gemma3.embedding_length"] = p.TextModel.HiddenSize | ||||
| 		kv["gemma3.feed_forward_length"] = p.TextModel.IntermediateSize | ||||
| 		kv["gemma3.attention.sliding_window"] = p.TextModel.SlidingWindow | ||||
| 		kv["gemma3.vision.block_count"] = p.VisionModel.NumHiddenLayers | ||||
| 		kv["gemma3.vision.embedding_length"] = p.VisionModel.HiddenSize | ||||
| 		kv["gemma3.vision.feed_forward_length"] = p.VisionModel.IntermediateSize | ||||
| 		kv["gemma3.vision.image_size"] = p.VisionModel.ImageSize | ||||
| 		kv["gemma3.vision.patch_size"] = p.VisionModel.PatchSize | ||||
| 		kv["gemma3.vision.num_channels"] = cmp.Or(p.VisionModel.NumChannels, 3) | ||||
| 		kv["gemma3.vision.attention.head_count"] = p.VisionModel.NumAttentionHeads | ||||
| 		kv["gemma3.vision.attention.layer_norm_epsilon"] = cmp.Or(p.VisionModel.LayerNormEpsilon, 1e-6) | ||||
| 		kv["gemma3.attention.key_length"] = cmp.Or(p.TextModel.HeadDim, 256) | ||||
| 		kv["gemma3.attention.value_length"] = cmp.Or(p.TextModel.HeadDim, 256) | ||||
| 	} | ||||
|  | ||||
| 	if p.MultiModalTokensPerImage > 0 { | ||||
| 		kv["gemma3.mm.tokens_per_image"] = p.MultiModalTokensPerImage | ||||
| 	} | ||||
|  | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *gemma3Model) Replacements() []string { | ||||
| 	return []string{ | ||||
| 		"lm_head", "output", | ||||
| 		"model.embed_tokens", "token_embd", | ||||
| 		"model.norm", "output_norm", | ||||
| 		"vision_tower.vision_model.embeddings", "v", | ||||
| 		"vision_tower.vision_model", "v", | ||||
| 		"vision_model.vision_model.embeddings", "v", | ||||
| 		"vision_model.vision_model", "v", | ||||
| 		"language_model.", "", | ||||
| 		"model.layers", "blk", | ||||
| 		"encoder.layers", "blk", | ||||
| 		"input_layernorm", "attn_norm", | ||||
| 		"self_attn.q_proj", "attn_q", | ||||
| 		"self_attn.q_norm", "attn_q_norm", | ||||
| 		"self_attn.k_proj", "attn_k", | ||||
| 		"self_attn.k_norm", "attn_k_norm", | ||||
| 		"self_attn.v_proj", "attn_v", | ||||
| 		"self_attn.o_proj", "attn_output", | ||||
| 		"self_attn.out_proj", "attn_output", | ||||
| 		"mlp.gate_proj", "ffn_gate", | ||||
| 		"mlp.down_proj", "ffn_down", | ||||
| 		"mlp.up_proj", "ffn_up", | ||||
| 		"post_attention_layernorm", "post_attention_norm", | ||||
| 		"pre_feedforward_layernorm", "ffn_norm", | ||||
| 		"post_feedforward_layernorm", "post_ffw_norm", | ||||
| 		"input_projection_weight", "input_projection.weight", | ||||
| 		"multi_modal_projector", "mm", | ||||
| 	} | ||||
| } | ||||
| @@ -9,7 +9,7 @@ import ( | ||||
| 	"github.com/pdevine/tensor" | ||||
| 	"github.com/pdevine/tensor/native" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type llamaModel struct { | ||||
| @@ -46,7 +46,7 @@ type llamaModel struct { | ||||
|  | ||||
| var _ ModelConverter = (*llamaModel)(nil) | ||||
|  | ||||
| func (p *llamaModel) KV(t *Tokenizer) llm.KV { | ||||
| func (p *llamaModel) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "llama" | ||||
| 	kv["llama.vocab_size"] = p.VocabSize | ||||
| @@ -120,11 +120,11 @@ func (p *llamaModel) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *llamaModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (p *llamaModel) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
|  | ||||
| 	if p.RopeScaling.factors != nil { | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     "rope_freqs.weight", | ||||
| 			Kind:     0, | ||||
| 			Shape:    []uint64{uint64(len(p.RopeScaling.factors))}, | ||||
| @@ -138,7 +138,7 @@ func (p *llamaModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 			t.SetRepacker(p.repack) | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import ( | ||||
| 	"github.com/pdevine/tensor" | ||||
| 	"github.com/pdevine/tensor/native" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type llamaAdapter struct { | ||||
| @@ -18,7 +18,7 @@ type llamaAdapter struct { | ||||
|  | ||||
| var _ AdapterConverter = (*llamaAdapter)(nil) | ||||
|  | ||||
| func (p *llamaAdapter) KV(baseKV llm.KV) llm.KV { | ||||
| func (p *llamaAdapter) KV(baseKV ggml.KV) ggml.KV { | ||||
| 	kv := p.AdapterParameters.KV() | ||||
| 	kv["general.architecture"] = "llama" | ||||
| 	kv["llama.attention.head_count"] = baseKV["llama.attention.head_count"] | ||||
| @@ -29,8 +29,8 @@ func (p *llamaAdapter) KV(baseKV llm.KV) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *llamaAdapter) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (p *llamaAdapter) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		shape := t.Shape() | ||||
| 		if (strings.HasSuffix(t.Name(), "weight.lora_a") && shape[0] > shape[1]) || | ||||
| @@ -41,7 +41,7 @@ func (p *llamaAdapter) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 			t.SetRepacker(p.repack) | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    shape, | ||||
|   | ||||
							
								
								
									
										190
									
								
								convert/convert_mistral.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								convert/convert_mistral.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,190 @@ | ||||
| package convert | ||||
|  | ||||
| import ( | ||||
| 	"cmp" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/pdevine/tensor" | ||||
| 	"github.com/pdevine/tensor/native" | ||||
|  | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type mistral3Model struct { | ||||
| 	ModelParameters | ||||
| 	ImageTokenIndex    uint32 `json:"image_token_index"` | ||||
| 	SpatialMergeSize   uint32 `json:"spatial_merge_size"` | ||||
| 	VisionFeatureLayer int32  `json:"vision_feature_layer"` | ||||
| 	TextModel          struct { | ||||
| 		NumHiddenLayers       uint32  `json:"num_hidden_layers"` | ||||
| 		MaxPositionEmbeddings uint32  `json:"max_position_embeddings"` | ||||
| 		HiddenSize            uint32  `json:"hidden_size"` | ||||
| 		IntermediateSize      uint32  `json:"intermediate_size"` | ||||
| 		NumAttentionHeads     uint32  `json:"num_attention_heads"` | ||||
| 		NumKeyValueHeads      uint32  `json:"num_key_value_heads"` | ||||
| 		RopeTheta             float32 `json:"rope_theta"` | ||||
| 		RMSNormEPS            float32 `json:"rms_norm_eps"` | ||||
| 		HeadDim               uint32  `json:"head_dim"` | ||||
| 		SlidingWindow         *uint32 `json:"sliding_window"` | ||||
| 		HiddenAct             string  `json:"hidden_act"` | ||||
| 		VocabSize             uint32  `json:"vocab_size"` | ||||
| 	} `json:"text_config"` | ||||
| 	VisionModel struct { | ||||
| 		NumAttentionHeads uint32  `json:"num_attention_heads"` | ||||
| 		NumHiddenLayers   uint32  `json:"num_hidden_layers"` | ||||
| 		HiddenSize        uint32  `json:"hidden_size"` | ||||
| 		IntermediateSize  uint32  `json:"intermediate_size"` | ||||
| 		ImageSize         uint32  `json:"image_size"` | ||||
| 		NumChannels       uint32  `json:"num_channels"` | ||||
| 		PatchSize         uint32  `json:"patch_size"` | ||||
| 		HeadDim           uint32  `json:"head_dim"` | ||||
| 		HiddenAct         string  `json:"hidden_act"` | ||||
| 		RopeTheta         float32 `json:"rope_theta"` | ||||
| 	} `json:"vision_config"` | ||||
| 	MultiModalProjectorBias bool   `json:"multimodal_projector_bias"` | ||||
| 	ProjectorHiddenAct      string `json:"projector_hidden_act"` | ||||
| } | ||||
|  | ||||
| func (p *mistral3Model) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "mistral3" | ||||
| 	kv["mistral3.vocab_size"] = p.TextModel.VocabSize | ||||
|  | ||||
| 	// Text configuration | ||||
| 	kv["mistral3.block_count"] = p.TextModel.NumHiddenLayers | ||||
| 	kv["mistral3.context_length"] = p.TextModel.MaxPositionEmbeddings | ||||
| 	kv["mistral3.embedding_length"] = p.TextModel.HiddenSize | ||||
| 	kv["mistral3.feed_forward_length"] = p.TextModel.IntermediateSize | ||||
| 	kv["mistral3.attention.head_count"] = p.TextModel.NumAttentionHeads | ||||
| 	kv["mistral3.attention.head_count_kv"] = p.TextModel.NumKeyValueHeads | ||||
| 	kv["mistral3.attention.layer_norm_rms_epsilon"] = p.TextModel.RMSNormEPS | ||||
| 	kv["mistral3.attention.key_length"] = p.TextModel.HeadDim | ||||
| 	kv["mistral3.attention.value_length"] = p.TextModel.HeadDim | ||||
| 	kv["mistral3.rope.dimension_count"] = p.TextModel.HiddenSize / p.TextModel.NumHiddenLayers | ||||
| 	kv["mistral3.rope.freq_base"] = p.TextModel.RopeTheta | ||||
|  | ||||
| 	// Vision configuration | ||||
| 	kv["mistral3.vision.block_count"] = p.VisionModel.NumHiddenLayers | ||||
| 	kv["mistral3.vision.embedding_length"] = p.VisionModel.HiddenSize | ||||
| 	kv["mistral3.vision.feed_forward_length"] = p.VisionModel.IntermediateSize | ||||
| 	kv["mistral3.vision.attention.head_count"] = p.VisionModel.NumAttentionHeads | ||||
| 	kv["mistral3.vision.attention.key_length"] = p.VisionModel.HeadDim | ||||
| 	kv["mistral3.vision.image_size"] = p.VisionModel.ImageSize | ||||
| 	kv["mistral3.vision.patch_size"] = p.VisionModel.PatchSize | ||||
| 	kv["mistral3.vision.num_channels"] = p.VisionModel.NumChannels | ||||
| 	// kv["mistral3.vision.attention.layer_norm_epsilon"] = 1e-05 // Default value | ||||
| 	kv["mistral3.vision.rope.freq_base"] = p.VisionModel.RopeTheta | ||||
|  | ||||
| 	// Multimodal configuration | ||||
| 	kv["mistral3.image_token_index"] = p.ImageTokenIndex | ||||
| 	kv["mistral3.spatial_merge_size"] = p.SpatialMergeSize | ||||
|  | ||||
| 	kv["mistral3.mm.projector_bias"] = p.MultiModalProjectorBias | ||||
|  | ||||
| 	if p.ProjectorHiddenAct != "" { | ||||
| 		kv["mistral3.mm.projector_hidden_act"] = p.ProjectorHiddenAct | ||||
| 	} | ||||
|  | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *mistral3Model) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
|  | ||||
| 	for _, t := range ts { | ||||
| 		if !strings.HasPrefix(t.Name(), "v.") { | ||||
| 			if strings.HasSuffix(t.Name(), ".attn_q.weight") || | ||||
| 				strings.HasSuffix(t.Name(), ".attn_k.weight") { | ||||
| 				t.SetRepacker(p.repack) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
| 			WriterTo: t, | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	return out | ||||
| } | ||||
|  | ||||
| func (p *mistral3Model) Replacements() []string { | ||||
| 	return []string{ | ||||
| 		"language_model.model.norm", "output_norm", | ||||
| 		"language_model.model.", "", | ||||
| 		"language_model.", "", | ||||
| 		"layers", "blk", | ||||
| 		"transformer.layers", "blk", | ||||
| 		"vision_tower", "v", | ||||
| 		"ln_pre", "encoder_norm", | ||||
| 		"input_layernorm", "attn_norm", | ||||
| 		"post_attention_layernorm", "ffn_norm", | ||||
| 		"embed_tokens", "token_embd", | ||||
| 		"self_attn.q_proj", "attn_q", | ||||
| 		"self_attn.k_proj", "attn_k", | ||||
| 		"self_attn.v_proj", "attn_v", | ||||
| 		"self_attn.o_proj", "attn_output", | ||||
| 		"mlp.down_proj", "ffn_down", | ||||
| 		"mlp.gate_proj", "ffn_gate", | ||||
| 		"mlp.up_proj", "ffn_up", | ||||
| 		"attention.q_proj", "attn_q", | ||||
| 		"attention.k_proj", "attn_k", | ||||
| 		"attention.v_proj", "attn_v", | ||||
| 		"attention.o_proj", "attn_output", | ||||
| 		"attention_norm", "attn_norm", | ||||
| 		"feed_forward.gate_proj", "ffn_gate", | ||||
| 		"feed_forward.down_proj", "ffn_down", | ||||
| 		"feed_forward.up_proj", "ffn_up", | ||||
| 		"multi_modal_projector", "mm", | ||||
| 		"ffn_norm", "ffn_norm", | ||||
| 		"lm_head", "output", | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *mistral3Model) repack(name string, data []float32, shape []uint64) ([]float32, error) { | ||||
| 	var dims []int | ||||
| 	for _, dim := range shape { | ||||
| 		dims = append(dims, int(dim)) | ||||
| 	} | ||||
|  | ||||
| 	var heads uint32 | ||||
| 	if strings.HasSuffix(name, ".attn_q.weight") { | ||||
| 		heads = p.TextModel.NumAttentionHeads | ||||
| 	} else if strings.HasSuffix(name, ".attn_k.weight") { | ||||
| 		heads = cmp.Or(p.TextModel.NumKeyValueHeads, p.TextModel.NumAttentionHeads) | ||||
| 	} else { | ||||
| 		return nil, fmt.Errorf("unknown tensor for repack: %s", name) | ||||
| 	} | ||||
|  | ||||
| 	n := tensor.New(tensor.WithShape(dims...), tensor.WithBacking(data)) | ||||
| 	if err := n.Reshape(append([]int{int(heads), 2, dims[0] / int(heads) / 2}, dims[1:]...)...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	if err := n.T(0, 2, 1, 3); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	if err := n.Reshape(dims...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	if err := n.Transpose(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	ts, err := native.SelectF32(n, 1) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var f32s []float32 | ||||
| 	for _, t := range ts { | ||||
| 		f32s = append(f32s, t...) | ||||
| 	} | ||||
|  | ||||
| 	return f32s, nil | ||||
| } | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"slices" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type mixtralModel struct { | ||||
| @@ -15,7 +15,7 @@ type mixtralModel struct { | ||||
| 	NumExpertsPerToken uint32 `json:"num_experts_per_tok"` | ||||
| } | ||||
|  | ||||
| func (p *mixtralModel) KV(t *Tokenizer) llm.KV { | ||||
| func (p *mixtralModel) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.llamaModel.KV(t) | ||||
|  | ||||
| 	if p.NumLocalExperts > 0 { | ||||
| @@ -29,7 +29,7 @@ func (p *mixtralModel) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *mixtralModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| func (p *mixtralModel) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	oldnew := []string{ | ||||
| 		"model.layers", "blk", | ||||
| 		"w1", "ffn_gate_exps", | ||||
| @@ -56,10 +56,10 @@ func (p *mixtralModel) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 		return true | ||||
| 	}) | ||||
|  | ||||
| 	var out []llm.Tensor | ||||
| 	var out []ggml.Tensor | ||||
| 	for n, e := range experts { | ||||
| 		// TODO(mxyng): sanity check experts | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     n, | ||||
| 			Kind:     e[0].Kind(), | ||||
| 			Shape:    append([]uint64{uint64(len(e))}, e[0].Shape()...), | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import ( | ||||
| 	"strings" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type phi3Model struct { | ||||
| @@ -37,7 +37,7 @@ type phi3Model struct { | ||||
|  | ||||
| var _ ModelConverter = (*phi3Model)(nil) | ||||
|  | ||||
| func (p *phi3Model) KV(t *Tokenizer) llm.KV { | ||||
| func (p *phi3Model) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := p.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "phi3" | ||||
| 	kv["phi3.context_length"] = p.MaxPositionEmbeddings | ||||
| @@ -68,19 +68,19 @@ func (p *phi3Model) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (p *phi3Model) Tensors(ts []Tensor) []llm.Tensor { | ||||
| func (p *phi3Model) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var addRopeFactors sync.Once | ||||
|  | ||||
| 	out := make([]llm.Tensor, 0, len(ts)+2) | ||||
| 	out := make([]ggml.Tensor, 0, len(ts)+2) | ||||
| 	for _, t := range ts { | ||||
| 		if strings.HasPrefix(t.Name(), "blk.0.") { | ||||
| 			addRopeFactors.Do(func() { | ||||
| 				out = append(out, llm.Tensor{ | ||||
| 				out = append(out, ggml.Tensor{ | ||||
| 					Name:     "rope_factors_long.weight", | ||||
| 					Kind:     0, | ||||
| 					Shape:    []uint64{uint64(len(p.RopeScaling.LongFactor))}, | ||||
| 					WriterTo: p.RopeScaling.LongFactor, | ||||
| 				}, llm.Tensor{ | ||||
| 				}, ggml.Tensor{ | ||||
| 					Name:     "rope_factors_short.weight", | ||||
| 					Kind:     0, | ||||
| 					Shape:    []uint64{uint64(len(p.RopeScaling.ShortFactor))}, | ||||
| @@ -89,7 +89,7 @@ func (p *phi3Model) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 			}) | ||||
| 		} | ||||
|  | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| package convert | ||||
|  | ||||
| import "github.com/ollama/ollama/llm" | ||||
| import "github.com/ollama/ollama/fs/ggml" | ||||
|  | ||||
| type qwen2Model struct { | ||||
| 	ModelParameters | ||||
| @@ -21,7 +21,7 @@ type qwen2Model struct { | ||||
|  | ||||
| var _ ModelConverter = (*qwen2Model)(nil) | ||||
|  | ||||
| func (q *qwen2Model) KV(t *Tokenizer) llm.KV { | ||||
| func (q *qwen2Model) KV(t *Tokenizer) ggml.KV { | ||||
| 	kv := q.ModelParameters.KV(t) | ||||
| 	kv["general.architecture"] = "qwen2" | ||||
| 	kv["qwen2.block_count"] = q.HiddenLayers | ||||
| @@ -45,10 +45,10 @@ func (q *qwen2Model) KV(t *Tokenizer) llm.KV { | ||||
| 	return kv | ||||
| } | ||||
|  | ||||
| func (q *qwen2Model) Tensors(ts []Tensor) []llm.Tensor { | ||||
| 	var out []llm.Tensor | ||||
| func (q *qwen2Model) Tensors(ts []Tensor) []ggml.Tensor { | ||||
| 	var out []ggml.Tensor | ||||
| 	for _, t := range ts { | ||||
| 		out = append(out, llm.Tensor{ | ||||
| 		out = append(out, ggml.Tensor{ | ||||
| 			Name:     t.Name(), | ||||
| 			Kind:     t.Kind(), | ||||
| 			Shape:    t.Shape(), | ||||
|   | ||||
| @@ -20,7 +20,7 @@ import ( | ||||
|  | ||||
| 	"golang.org/x/exp/maps" | ||||
|  | ||||
| 	"github.com/ollama/ollama/llm" | ||||
| 	"github.com/ollama/ollama/fs/ggml" | ||||
| ) | ||||
|  | ||||
| type tensorData struct { | ||||
| @@ -29,7 +29,7 @@ type tensorData struct { | ||||
| 	Shape   []int  `json:"shape"` | ||||
| } | ||||
|  | ||||
| func convertFull(t *testing.T, fsys fs.FS) (*os.File, llm.KV, *llm.Tensors) { | ||||
| func convertFull(t *testing.T, fsys fs.FS) (*os.File, ggml.KV, ggml.Tensors) { | ||||
| 	t.Helper() | ||||
|  | ||||
| 	f, err := os.CreateTemp(t.TempDir(), "f16") | ||||
| @@ -48,7 +48,7 @@ func convertFull(t *testing.T, fsys fs.FS) (*os.File, llm.KV, *llm.Tensors) { | ||||
| 	} | ||||
| 	t.Cleanup(func() { r.Close() }) | ||||
|  | ||||
| 	m, _, err := llm.DecodeGGML(r, math.MaxInt) | ||||
| 	m, _, err := ggml.Decode(r, math.MaxInt) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| @@ -60,7 +60,7 @@ func convertFull(t *testing.T, fsys fs.FS) (*os.File, llm.KV, *llm.Tensors) { | ||||
| 	return r, m.KV(), m.Tensors() | ||||
| } | ||||
|  | ||||
| func generateResultsJSON(t *testing.T, f *os.File, kv llm.KV, tensors *llm.Tensors) map[string]string { | ||||
| func generateResultsJSON(t *testing.T, f *os.File, kv ggml.KV, tensors ggml.Tensors) map[string]string { | ||||
| 	actual := make(map[string]string) | ||||
| 	for k, v := range kv { | ||||
| 		if s, ok := v.(json.Marshaler); !ok { | ||||
| @@ -75,7 +75,7 @@ func generateResultsJSON(t *testing.T, f *os.File, kv llm.KV, tensors *llm.Tenso | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, tensor := range tensors.Items { | ||||
| 	for _, tensor := range tensors.Items() { | ||||
| 		sha256sum := sha256.New() | ||||
| 		sr := io.NewSectionReader(f, int64(tensors.Offset+tensor.Offset), int64(tensor.Size())) | ||||
| 		if _, err := io.Copy(sha256sum, sr); err != nil { | ||||
| @@ -109,6 +109,7 @@ func TestConvertModel(t *testing.T) { | ||||
| 		"all-MiniLM-L6-v2", | ||||
| 		"gemma-2-9b-it", | ||||
| 		"Qwen2.5-0.5B-Instruct", | ||||
| 		"c4ai-command-r-v01", | ||||
| 	} | ||||
|  | ||||
| 	for i := range cases { | ||||
| @@ -331,7 +332,7 @@ func TestConvertAdapter(t *testing.T) { | ||||
| 			} | ||||
| 			defer r.Close() | ||||
|  | ||||
| 			m, _, err := llm.DecodeGGML(r, math.MaxInt) | ||||
| 			m, _, err := ggml.Decode(r, math.MaxInt) | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
|   | ||||
| @@ -62,10 +62,7 @@ func parseTensors(fsys fs.FS, replacer *strings.Replacer) ([]Tensor, error) { | ||||
| 		Pattern string | ||||
| 		Func    func(fs.FS, *strings.Replacer, ...string) ([]Tensor, error) | ||||
| 	}{ | ||||
| 		{"model-*-of-*.safetensors", parseSafetensors}, | ||||
| 		{"model.safetensors", parseSafetensors}, | ||||
| 		{"adapters.safetensors", parseSafetensors}, | ||||
| 		{"adapter_model.safetensors", parseSafetensors}, | ||||
| 		{"*.safetensors", parseSafetensors}, | ||||
| 		{"pytorch_model-*-of-*.bin", parseTorch}, | ||||
| 		{"pytorch_model.bin", parseTorch}, | ||||
| 		{"consolidated.*.pth", parseTorch}, | ||||
|   | ||||
| @@ -1360,7 +1360,7 @@ func file_sentencepiece_model_proto_rawDescGZIP() []byte { | ||||
|  | ||||
| var file_sentencepiece_model_proto_enumTypes = make([]protoimpl.EnumInfo, 2) | ||||
| var file_sentencepiece_model_proto_msgTypes = make([]protoimpl.MessageInfo, 6) | ||||
| var file_sentencepiece_model_proto_goTypes = []interface{}{ | ||||
| var file_sentencepiece_model_proto_goTypes = []any{ | ||||
| 	(TrainerSpec_ModelType)(0),         // 0: sentencepiece.TrainerSpec.ModelType | ||||
| 	(ModelProto_SentencePiece_Type)(0), // 1: sentencepiece.ModelProto.SentencePiece.Type | ||||
| 	(*TrainerSpec)(nil),                // 2: sentencepiece.TrainerSpec | ||||
| @@ -1392,7 +1392,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 		return | ||||
| 	} | ||||
| 	if !protoimpl.UnsafeEnabled { | ||||
| 		file_sentencepiece_model_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[0].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*TrainerSpec); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -1406,7 +1406,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_sentencepiece_model_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[1].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*NormalizerSpec); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -1420,7 +1420,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_sentencepiece_model_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[2].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*SelfTestData); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -1434,7 +1434,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_sentencepiece_model_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[3].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*ModelProto); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -1448,7 +1448,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_sentencepiece_model_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[4].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*SelfTestData_Sample); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -1460,7 +1460,7 @@ func file_sentencepiece_model_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_sentencepiece_model_proto_msgTypes[5].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_sentencepiece_model_proto_msgTypes[5].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*ModelProto_SentencePiece); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
|   | ||||
							
								
								
									
										344
									
								
								convert/testdata/c4ai-command-r-v01.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										344
									
								
								convert/testdata/c4ai-command-r-v01.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,344 @@ | ||||
| { | ||||
|     "general.architecture": "command-r", | ||||
|     "general.name": "command-r", | ||||
|     "command-r.attention.head_count": "64", | ||||
|     "command-r.attention.head_count_kv": "64", | ||||
|     "command-r.attention.layer_norm_epsilon": "1e-05", | ||||
|     "command-r.block_count": "40", | ||||
|     "command-r.context_length": "131072", | ||||
|     "command-r.embedding_length": "8192", | ||||
|     "command-r.feed_forward_length": "22528", | ||||
|     "command-r.logit_scale": "0.0625", | ||||
|     "command-r.rope.freq_base": "8e+06", | ||||
|     "command-r.rope.scaling.type": "none", | ||||
|     "tokenizer.ggml.add_bos_token": "true", | ||||
|     "tokenizer.ggml.add_eos_token": "false", | ||||
|     "tokenizer.ggml.bos_token_id": "5", | ||||
|     "tokenizer.ggml.eos_token_id": "255001", | ||||
|     "tokenizer.ggml.merges": "902a060cac8884a5793d2a857dd2e53a259de46c8d08c4deb243c239671e1350", | ||||
|     "tokenizer.ggml.model": "gpt2", | ||||
|     "tokenizer.ggml.padding_token_id": "0", | ||||
|     "tokenizer.ggml.token_type": "b7a352ccd1c99d4413bcf452c2db707b0526d0e1216616b865560fab80296462", | ||||
|     "tokenizer.ggml.tokens": "815ac90ff23565081522d7258f46648c8a0619eb847a9c7c31b238a9b984e4ae", | ||||
|     "blk.0.attn_k.weight": "6fcfdb466f9ceb1229404ce4ec4e480751b8d00da12707a11783dad7256cb864", | ||||
|     "blk.0.attn_norm.weight": "6063317f731371864049c7704a70772f1eb632194201ebdc2ed0f8e483507c72", | ||||
|     "blk.0.attn_output.weight": "920f49716a1e2fc73b6794ec777947f1c122701e63ed302422ac89e90f06e9da", | ||||
|     "blk.0.attn_q.weight": "ddbcd7cde197e632564ac58e4f25d9e3a8ca52917329eeb6081eb41a797932ab", | ||||
|     "blk.0.attn_v.weight": "318fc02a189d87420f0cbf57f47f11e00c21ec1ed472ce0a2a895b44f7fa0fca", | ||||
|     "blk.0.ffn_down.weight": "aa71975b6eb1f4c77b03d2ac4a194cf8d95718efac741bb12f0f3ff79a27f9bc", | ||||
|     "blk.0.ffn_gate.weight": "42967702fa0bc738b88dc50007ace26dbe74a5a9e0978124dd093f818241a9e1", | ||||
|     "blk.0.ffn_up.weight": "5282c8788b086bd30f46525e7995a17464882a72703fd27165491afdd8bfd4af", | ||||
|     "blk.1.attn_k.weight": "cd248882e64fd2c3402c44790ebe12440133dc671b6893fdad0564c461973adc", | ||||
|     "blk.1.attn_norm.weight": "ba84e1c8fd30af6ec94208db4078befac8c921aad3acb887812887f3282ea2be", | ||||
|     "blk.1.attn_output.weight": "2efa3ef7c5666ccceb05e339b83ad680cc0d2c3ec78203f5da5959f23a80e14f", | ||||
|     "blk.1.attn_q.weight": "5106f2e255358a1303c22e8b5f0ec044852bb30a866c52cabefd30017a7a6b7d", | ||||
|     "blk.1.attn_v.weight": "a211a634a1a5df1d5f973645438be0461dd922210f9747c6b04e386c7f1ebe95", | ||||
|     "blk.1.ffn_down.weight": "37093afe48d32c578ec956c9ed85242cd000d6aa979e60526aafa10c822dbb10", | ||||
|     "blk.1.ffn_gate.weight": "469860819e9159caefb1aad0bc66db790f3393f05fd87b08e52256a7ed256543", | ||||
|     "blk.1.ffn_up.weight": "736742c97d35d1a011f9cafd3c0ce947ad559bb2fba6da73c816f6bfd0fa9aeb", | ||||
|     "blk.2.attn_k.weight": "92c219d92804d832ab404bd6dc7339c90877bb7cf405dd030c121f8b27757739", | ||||
|     "blk.2.attn_norm.weight": "61e4466069474b76b6d1e702566420eb669faf3556b00ff7b824784aca13a2d6", | ||||
|     "blk.2.attn_output.weight": "d2fb38a2b2171fd91caf037faa585a62225819aa232d86fd4f7f9d2c3c8a45e9", | ||||
|     "blk.2.attn_q.weight": "f6faf5cc6844e3daa4f9f68d90f5458c64879de68a7728860e38374e30c3429d", | ||||
|     "blk.2.attn_v.weight": "f340ef8f7341d987a6f37c0e9afe0aef5be67be00c0ce5f57612daf73319cce1", | ||||
|     "blk.2.ffn_down.weight": "c7be61a701d779860b621b143fb6365b607bf99ec7c0f153b07908ac8120885a", | ||||
|     "blk.2.ffn_gate.weight": "b64f0878187bd3392abfa4c3e8ad2f8b4c133903e54246747ff8f3b4639ad83e", | ||||
|     "blk.2.ffn_up.weight": "50b11c712652e90ee7428dbb45cffebb80662ac982bc72bd9eafff361b5eb5a8", | ||||
|     "blk.3.attn_k.weight": "2b7bcbe9ee5c9c630c8c8d7483887e78b73581016f4cbb6933db2a147a25f431", | ||||
|     "blk.3.attn_norm.weight": "0181dac7f4eee7252980323e8032cf339bef2046ce0a16c0fd72af7c98a8a37b", | ||||
|     "blk.3.attn_output.weight": "aef8843b636ce231da9e7c9acbee197883cc15df0e2887709324c6a50f16da7b", | ||||
|     "blk.3.attn_q.weight": "55404130fa10e81322d33eb378aa0de31a92990ce7730f1338c0ace0406bb1b1", | ||||
|     "blk.3.attn_v.weight": "76f7fb8040d82b957d689ce34fea2302a6640ad5bbaa0052ad2b7ebce270c33d", | ||||
|     "blk.3.ffn_down.weight": "648628933eff3b357c3729c33c5b1ae51c28e59b9c19acd1601a2ff7c5d5d9a5", | ||||
|     "blk.3.ffn_gate.weight": "6a588885d16e98d5f50ebed05af089154f680085ca9c97691e5b489088630a4a", | ||||
|     "blk.3.ffn_up.weight": "e12455a1d702f4986e1a663493e3d5102b367af74d45557522002a35d63ecac2", | ||||
|     "blk.4.attn_k.weight": "40d943380a8a85e4eab147934bf6e16f23cc8ab753f6636526382c074d182288", | ||||
|     "blk.4.attn_norm.weight": "4ab2c098983d4599fe540eef624c4df954adb7473faebda7471ef0ba4134814c", | ||||
|     "blk.4.attn_output.weight": "d14b91e40f58bf4a3c8c2eca0b12bb541de406574af39027d56f6c588a147082", | ||||
|     "blk.4.attn_q.weight": "e1224960a3562107488589f883fa32414bae41712fa8dbd47c5f3e3a7801452f", | ||||
|     "blk.4.attn_v.weight": "063f297bc4aa6e709fc32c4c32e35af7d07d80e83cb939b76adbba858006c03d", | ||||
|     "blk.4.ffn_down.weight": "f88a18020c5e1caaa29596895eb348e76ee5bfad27ed57651a86cd8cd1f9b5aa", | ||||
|     "blk.4.ffn_gate.weight": "48e7e1eed3fb52e92e61d3557dd0ec002418327090e034ce4322fd68542266f8", | ||||
|     "blk.4.ffn_up.weight": "1ca8a7aa17355b6ce0d9ad5539fdad3899fa47fd359c285fbfb31f19f47bf073", | ||||
|     "blk.5.attn_k.weight": "2bdf15f8e73d068d972380f25d207004cf0bf3b5bfa46946803ba6fba07d9175", | ||||
|     "blk.5.attn_norm.weight": "60448d7cde6e1b6467aa31bdea012e39cdb08c88081cee7d102dca4f93f766ef", | ||||
|     "blk.5.attn_output.weight": "f9f687d7c457537f9fca8a4087a59f1c3bebfaf5537b94e42c831a13224f7799", | ||||
|     "blk.5.attn_q.weight": "987db7a2ad68657a92625e1980effbb1f79697c2183f2b9f3b3a0570c51b0ab9", | ||||
|     "blk.5.attn_v.weight": "cf696891148f3e4783ad1d20f93462ae091eb8651c656bba9b662253b6263e02", | ||||
|     "blk.5.ffn_down.weight": "c0662b0bd0929136005fb9d691fdd9b2c33867d9ce9622339a6a456b720b059a", | ||||
|     "blk.5.ffn_gate.weight": "200bbdfab615d7a3a84719b6ced7751e3ce52757ef212d96f87798bc1de5e987", | ||||
|     "blk.5.ffn_up.weight": "df5d23e7e035fb1b9d163da7ddfdfe38da6a37e86e96534dc02ad20f011b55b3", | ||||
|     "blk.6.attn_k.weight": "c0dae2d272a7c5a2fa004bbb8475dbab362fc1f6d008e73d5a4434a9382ac6ba", | ||||
|     "blk.6.attn_norm.weight": "51c57ac8b55e04354d5dca6bb9c0cf4177639d3b038e80209e33036209688f64", | ||||
|     "blk.6.attn_output.weight": "229d97892c62f85bcdf431675250e01c976ad69ffa450b01fb543bf88f14a2fb", | ||||
|     "blk.6.attn_q.weight": "c20e49621821bd46ed156e6823864a5bda4f317750e71ab8dc54e44eb48cf7c2", | ||||
|     "blk.6.attn_v.weight": "53ceb1a2ee43fce3c7b5b33c58a9fc5ee7f44dc1c6f29bc9dbefc37582102dc9", | ||||
|     "blk.6.ffn_down.weight": "7923c943b7629d560a032d1efa210d1d75c6692140f1be94464ee7ed24f44ed0", | ||||
|     "blk.6.ffn_gate.weight": "57593d350361af753a6a39f53b066282634c0fb44f396f6f2966a574b01d8f8c", | ||||
|     "blk.6.ffn_up.weight": "327b6a7a387098b8899d3ded04a4d4e7c658ca61b80d4e7b17594be232721602", | ||||
|     "blk.7.attn_k.weight": "9ca48b87a10116fd8868e62b76f211d4bb91f166096be9061439ee2e1c3a5c20", | ||||
|     "blk.7.attn_norm.weight": "cd56cfcc4e2ad6b96e23ea7b0d32b4caf236107d99a0b22c56760b62e63c8cfd", | ||||
|     "blk.7.attn_output.weight": "7352b509a03cae2491ffc060e577d189341a0f861233f18c96f9d275dc4234bf", | ||||
|     "blk.7.attn_q.weight": "2b3791c8c008c33ddbe12bedba8191322ceea2dcce5cf0eb7a93d40ad254e672", | ||||
|     "blk.7.attn_v.weight": "3ae721d52466487a3d48150581e57f6d64ea1e83ab929f23b28c3d777422eeb6", | ||||
|     "blk.7.ffn_down.weight": "3b6fa8ececdb3c34af3a5363863d6f94289c1c95bf47fce3a3ddcf184c5f0848", | ||||
|     "blk.7.ffn_gate.weight": "dbd7df6c5ae5eb4adb859f0d36453813a4e289a359a1ba8f72d67fcbf21c3e22", | ||||
|     "blk.7.ffn_up.weight": "de68380a334b4c5cfd4c318b0e9854aec59bd79aa0f0c30af3f56414f83482b0", | ||||
|     "blk.8.attn_k.weight": "7303c4e4480abc72a7ee271811311199245fb5c2ea27a2bd3b8cad3a53a03c27", | ||||
|     "blk.8.attn_norm.weight": "2e3d1921898d1b943ce1a1b6818546c8b471d6d542da24f51a8b514b8c3dd4ef", | ||||
|     "blk.8.attn_output.weight": "30421520887b66bf97a18dbcdc283bc8d0b60590b612fd638a319a6eae923227", | ||||
|     "blk.8.attn_q.weight": "73e064d5433c9b500068a1c31744dbd53f4ade298fb450a0e8c97f62cf1f8a8d", | ||||
|     "blk.8.attn_v.weight": "27e21f8b9a9a8533e8178ca34a72aa1d786393d57302b7806dcdf3e51de511a8", | ||||
|     "blk.8.ffn_down.weight": "bf694bd8e00047982108000e7b3dee7b225db8b19abc595e5697b6bbefd92e7c", | ||||
|     "blk.8.ffn_gate.weight": "d55fdbf8606d9141b774b0500c58944fd1253b9e69d1f765eaa9a680b9f2ca40", | ||||
|     "blk.8.ffn_up.weight": "1ae3f580655e7c8e8dd6c34fa4ac574fdfc5e3f1a8536da0c5442d3a2976f0e7", | ||||
|     "blk.9.attn_k.weight": "b18080626012d8aabcf78542d6c7bf31c712bf55a70172fbfe173fcf34481036", | ||||
|     "blk.9.attn_norm.weight": "2e3620620dc09998c6d3063a7d5de5433fbbae8c11e5b00d13f145d39140e162", | ||||
|     "blk.9.attn_output.weight": "69c3c0e27ef1c0fc933eeb7b612b70909f18cde238873c0d576a2ba9714ef174", | ||||
|     "blk.9.attn_q.weight": "68330e5aa28a28873c9a6e67f032186ef651df2df5844e0f27094ba349fbe4ab", | ||||
|     "blk.9.attn_v.weight": "3df8d45a102be082d0793a51cb82aa62a43cd0e9d047ba4115ca0f2414b39325", | ||||
|     "blk.9.ffn_down.weight": "1d6cc162b73745b135b4f040a0aac3c06d5135a3dc5b2421e7ee2af48662fd7f", | ||||
|     "blk.9.ffn_gate.weight": "034a9d40fb1e32b534b45f4bccd65cbe43c4a6a3f5d01132bd245ca0005de5fc", | ||||
|     "blk.9.ffn_up.weight": "c838c38d0e1a0ac0da17eb2a66023ed31929f07d8fcfe1cc546df26096c91f0c", | ||||
|     "blk.10.attn_k.weight": "a78507cb72f744b86ceaa032596e74e5571c822d0226d334881169addb32cbd5", | ||||
|     "blk.10.attn_norm.weight": "35f48d0b28ee0e6b4cad4e983925737562d64824be5b168b3e26df3d6b260cf1", | ||||
|     "blk.10.attn_output.weight": "53712db06796de39b131323e7abf9a58551b6d52da6db66a471580386d396252", | ||||
|     "blk.10.attn_q.weight": "efe08429ba196026b81cd1c471e1c7418afd9e966659feb3936b674aa0803b58", | ||||
|     "blk.10.attn_v.weight": "7ec6055e134f89da0cbe79ec9f13ef2e442ac584b1f03c3e13e7d0cdad0078bd", | ||||
|     "blk.10.ffn_down.weight": "37e66af4bcd1f3079e841e892255b8255070655901864ea3a8c602a7f681a640", | ||||
|     "blk.10.ffn_gate.weight": "1825282bc34830d371c6edcc3c1e73e6ecc1e10f4aea0122dbb7acc1d6f7b1bc", | ||||
|     "blk.10.ffn_up.weight": "819b3b276a4d4c14a35ed6682d5ef18a5e8ed468e5ce3f12e8c75ec18ac20ec4", | ||||
|     "blk.11.attn_k.weight": "5327e6a2af82dfff0619a14971f5864a15553c36fead84e1af42c7630f2729c6", | ||||
|     "blk.11.attn_norm.weight": "fec363b3c4a43036d2c635fb8aa9e122dd87ee79811839f2f6cd955be3373e7b", | ||||
|     "blk.11.attn_output.weight": "ccf7b38f18ee8798b8a6a35018e2df3eb3e007de62876befb68025dd66c79763", | ||||
|     "blk.11.attn_q.weight": "da8c4a1c824ffe174e39f126cd72f7ef83c56aff1259d452a1212de80f98f5e9", | ||||
|     "blk.11.attn_v.weight": "d17ae6bb77f03982b55d341eb67acb5969e9ad3da5994b96eafc09793dcfe3a0", | ||||
|     "blk.11.ffn_down.weight": "a6bac521e2791345f22c57205fa1c2f2f687794dfd24d0e98d50ae0d0eb6088a", | ||||
|     "blk.11.ffn_gate.weight": "5ed902c488cb51ba5635f3df08258c5f84f31a679a00211ea5f9d8b824ef6d9d", | ||||
|     "blk.11.ffn_up.weight": "ee9f1437eb890d2cf9df2574afa1cecf20aafdd847cd75b152d7eb74419afd34", | ||||
|     "blk.12.attn_k.weight": "5a069c06e1019b0f889088e67458f7a11ec77fa190ada6069e46211f62219947", | ||||
|     "blk.12.attn_norm.weight": "194d7e5fcc8c49aea62daf1940532419cf3c505afdce6be377286b677db5db8f", | ||||
|     "blk.12.attn_output.weight": "6534995fd4d6fecb55e317add4b1723aba4d825e1e9471d0b08813dfdc247176", | ||||
|     "blk.12.attn_q.weight": "4ab51ca519b5995581fa34f846276feca3b907ef2b51f192f6cc0b3263c3f5a2", | ||||
|     "blk.12.attn_v.weight": "5652ca3fa81ef9a1ac1543d71fc6813f8517f8ec54b25c701f6f98061614830f", | ||||
|     "blk.12.ffn_down.weight": "4b2c263f54c88516b8eb273bb8d9615b01c5c8b484dc70358adb91b50b300edd", | ||||
|     "blk.12.ffn_gate.weight": "8f50c3c3e3e8568991d6c1b0e74b500cf4f208e7700bbb8e87c3f6a6d359b6b5", | ||||
|     "blk.12.ffn_up.weight": "1c1a581fec1fbe959e1427fa513f400100b5e1ee9d83932630be9905fb49c231", | ||||
|     "blk.13.attn_k.weight": "efd7a38c46f08d8376d82974f33c644e3a02220e142d63b1704718699a8a884c", | ||||
|     "blk.13.attn_norm.weight": "d28fa4f1bd75abbd063b0e622e08f579c89cd0c0c5ce63c1952ec9f944f8ee13", | ||||
|     "blk.13.attn_output.weight": "71e0068a639288718bdb70a6cfdefd50bc8b3ec3993347a65129e70001ca5827", | ||||
|     "blk.13.attn_q.weight": "b97077adc92cff07a2e07d80ee38f214ad8713571c69cd5c70ebd43dc501ac87", | ||||
|     "blk.13.attn_v.weight": "79b3e2749ab4b459c81e96e322b215f1e8af645eb346e176c326bd00cf6ed2fd", | ||||
|     "blk.13.ffn_down.weight": "9f8687d11effa1db7cfecf7bec5631734bcf2962aad74a9f519144491e08ec85", | ||||
|     "blk.13.ffn_gate.weight": "7d14dfa0543852e7777fe8fff29ca533744cbcf1ebcf10067e5adfc4eb345e65", | ||||
|     "blk.13.ffn_up.weight": "852b9527b97fdab211ff3f832a660ee1d93ccb56906144c50f01319a6e8ee615", | ||||
|     "blk.14.attn_k.weight": "79e926b20f36f66d58226cb358881f2f68ae7b468787d33cafae5110287a14a0", | ||||
|     "blk.14.attn_norm.weight": "97d481b63deb0df6142c2c6cd23043720c62eb609e390f47a7113751c79974ec", | ||||
|     "blk.14.attn_output.weight": "aa6e94d7176d5c79fbb89b96e5f13ce75702ce3dd23ee52986446da436a6c3d6", | ||||
|     "blk.14.attn_q.weight": "214becb6d1bb460da9fb8ace0f99b9a5afa9edf7aa7acc19606c7401b11d6305", | ||||
|     "blk.14.attn_v.weight": "488b0e6d7f1a7a2ed0972aaa6d10ef9c775ee5373460324efcf5b3e3da9311df", | ||||
|     "blk.14.ffn_down.weight": "29c7ad16cf9542e30996a1a01ab95b844533b28051f04cc7949c371afb796471", | ||||
|     "blk.14.ffn_gate.weight": "b7ef208f2b054803665b377f5a5980c122c026841809cf855c6ba06d1c3a885a", | ||||
|     "blk.14.ffn_up.weight": "76a5cc28100748d79c4398ce7b9176aab4d661548b6293a82f99144812e5b70e", | ||||
|     "blk.15.attn_k.weight": "a6b8f9e98ab878fa7ebc5d080978ebf2d050acc2ab2fa8ea9188eb10e27702c8", | ||||
|     "blk.15.attn_norm.weight": "a26d07a9752d6dccb68e3a8a2a49fd0752cdd0a415e05547819bc37d9ba63d5e", | ||||
|     "blk.15.attn_output.weight": "c63616c69048ccbee801e05be4f56d21fda21aa0cc470f41d57c31b4d9283a4d", | ||||
|     "blk.15.attn_q.weight": "fd595a67bf96c6ba16eb148a9d02fa52fa3c1d33ed10be28a08f851409fd6e64", | ||||
|     "blk.15.attn_v.weight": "1c5c9d33fa07c05d5f4ed0032c6c4aa83d863f0d31c94a66109d239dcd03cea3", | ||||
|     "blk.15.ffn_down.weight": "585ea62ab8aff7d7d212ea5c1a03226fda6b68370c890b776834af70c948dcbc", | ||||
|     "blk.15.ffn_gate.weight": "a13c63f86f879b03a573d5dd2a25cfd1f4dc73e8132e6454ecc23e538b4cdf6f", | ||||
|     "blk.15.ffn_up.weight": "f7112450f57c12fcd511f049e0dc0b541625a107a7901c3261ed9e984299f65c", | ||||
|     "blk.16.attn_k.weight": "2d2c8b11dd71fba6d1c106aa1673c113a5448653cca7eab897c8739212ed5003", | ||||
|     "blk.16.attn_norm.weight": "95c2ec7be9469690e18a9a1779684acb3e9da44b13e263a0da840305646fbf8a", | ||||
|     "blk.16.attn_output.weight": "31a65046e677f54dae654ded4e733479fcc0f7283d83076b7dc7cbcae8528230", | ||||
|     "blk.16.attn_q.weight": "bfc6292b9c6d49b7118d08060242a138182eb182d136ba5dfaf469437c16081d", | ||||
|     "blk.16.attn_v.weight": "68f81d037340217d87c7853ff4d6edfbc46d9e827ee6d5bff7c3f6238e3a95ad", | ||||
|     "blk.16.ffn_down.weight": "bbd6629691950cef4d5113e1c6670e91b216a9b872cb92cee02dfda4d6c4f7b8", | ||||
|     "blk.16.ffn_gate.weight": "63cb56f282b7401ed6c76e5bb6fdf1bf68a64f9af0c82c014209b55bcb5191d0", | ||||
|     "blk.16.ffn_up.weight": "b54f39a2541063cbfb6f713aa81c3b69a04100e999aa2ebbeec195dc382eceec", | ||||
|     "blk.17.attn_k.weight": "3d9ba49799cc56664ec30a002bcad61eb651294212a68c3ddb573eb042aef5a4", | ||||
|     "blk.17.attn_norm.weight": "42ee0db4b9d63257bca0012a30b12737ead1caafeb5ed3d93c8f48ffec4b46de", | ||||
|     "blk.17.attn_output.weight": "a38fd100f05c9041c592bc739e287de0b10d08ef2bda41a879225bdca9002f71", | ||||
|     "blk.17.attn_q.weight": "8a3bee285b0180a9eb35662e449ee4cbe16d992bdd48fb3a94bc4a347728cfa2", | ||||
|     "blk.17.attn_v.weight": "d7f8f1b8b863494ed4392a1656775912e9b264ad36016547b12e832a1d6757d6", | ||||
|     "blk.17.ffn_down.weight": "bb7ee58f61da8630972e25b621996fbe8ec06f4dc9ab1e268ab5b120c526ca28", | ||||
|     "blk.17.ffn_gate.weight": "6b652dbf167fee09a45ebfd78d500ff6548fb2756dbe5343ffec3f7e6207179f", | ||||
|     "blk.17.ffn_up.weight": "3b67f727e55e742715de978fab80457781e7a3762bc48f79d13b45dcb8de664c", | ||||
|     "blk.18.attn_k.weight": "ff7fe57c57b90c6fcc0aefc39ec24593c3a7d1ea1c23770480075a015450e0f5", | ||||
|     "blk.18.attn_norm.weight": "1d40faca082d2633ef0ccf19e121870dd6c7c3e2154607c7f3543fa96e99cb2d", | ||||
|     "blk.18.attn_output.weight": "9adfecaaa397a92db4687efd5fcabfa0daef9e6b0493763b7ff5ebc185c43a6c", | ||||
|     "blk.18.attn_q.weight": "ad1803eb9b291948639277afe981e666b07167eb3fcae903ba5b73bf86d8f50b", | ||||
|     "blk.18.attn_v.weight": "308cf23399adccf27401a4ab60d74dac6fb9d4cd4b9c5940d9145118d1881b34", | ||||
|     "blk.18.ffn_down.weight": "7de4ac9a561fb580619b745687dfd7ca8a69ef70471dee978741b80e9ff7bead", | ||||
|     "blk.18.ffn_gate.weight": "0c66970f696b33bd5ee8f1f2fbcb41fd78fa5ccabdc927e11a4d5a4089f19c69", | ||||
|     "blk.18.ffn_up.weight": "66a42e988e8a1f468fabf976c48e9e4bb045eaac6916ef16555ac101cd674abc", | ||||
|     "blk.19.attn_k.weight": "a928ab50390bacbcebe2e4b66922498134ce22d7b93beaa87d6cf4ab52eb7174", | ||||
|     "blk.19.attn_norm.weight": "b4a02c55b46c2a96aec9c64a254087cf48e6c1d4b6f31782c77a46fc4daebad1", | ||||
|     "blk.19.attn_output.weight": "b768319c641dff1eac5d1f8ceb960c9899c795bf2b24c1d6bf70aa24fda45f77", | ||||
|     "blk.19.attn_q.weight": "79ef3f57d187d3954a26362096e1b6c222d76f537dff73e034d6e9999935b8bc", | ||||
|     "blk.19.attn_v.weight": "ce13d6b13e24fcb2d5bc6a2662e5bd295b31b12db10a6d0307f86cf29b8d5001", | ||||
|     "blk.19.ffn_down.weight": "cf90d7e2137482cfd50934a8223ad774621d08554969da80a9712df5e6227eb0", | ||||
|     "blk.19.ffn_gate.weight": "71ce30150f003b6eeb3bf7464e05b6ae615f135110d8e47f0a47fd973e537c0f", | ||||
|     "blk.19.ffn_up.weight": "7f92aca0cc29866633feec701ec01a85a8ee2fd4e2b9630173a6cffb1d9d50ee", | ||||
|     "blk.20.attn_k.weight": "a2df23159d6fb74ef28e14b61028fe8b00a693a2fc9234a980be74f20b958682", | ||||
|     "blk.20.attn_norm.weight": "c6cd5f1b096fc5efa4eb59ca1c8c4bd28730f3dcedd59a63601663eccc6724ed", | ||||
|     "blk.20.attn_output.weight": "896a8a166d0f006d4b09867ae4345426303cbc3fb13a18d3d4e1bde00f16dbdf", | ||||
|     "blk.20.attn_q.weight": "01eb79588fe61baea0da43e99f4dc5939590e1bafd01e12dadb8326f102bfea2", | ||||
|     "blk.20.attn_v.weight": "bd39630fdd5a7c859ac1addaf53e63faf524c3f32f5f4896d86b6e746b1d5c06", | ||||
|     "blk.20.ffn_down.weight": "0304a5d39957a0e3f031c4bcc4549a135d396c8d97c8d276fd1c823ce86560c2", | ||||
|     "blk.20.ffn_gate.weight": "117b79d595b1dca0c8b37586beaecc4d84411507276212dc286cde7fc36c9bef", | ||||
|     "blk.20.ffn_up.weight": "6e799346db145c125f01783539749d3828fcc451cd4f10c5352f047a47e28714", | ||||
|     "blk.21.attn_k.weight": "1c37e4c0664147e775bb006b226b9553e3421140cd96288ea755f81731ab80ba", | ||||
|     "blk.21.attn_norm.weight": "00ae783a29000ccda5e4bdbff03df0752fb82805dc3f9b987500ebd80714476e", | ||||
|     "blk.21.attn_output.weight": "7588b84f9fb19f15095b5265c60b4a4e7ae74bcc47d4607dfa5d0bfab6f136cb", | ||||
|     "blk.21.attn_q.weight": "a65f1c0dd06d45bb97532d3e932689c1eecfe7359089b39174a96a149335cbc1", | ||||
|     "blk.21.attn_v.weight": "4220b77e7d5e8709b4eef33a679b5dad11f297085ef44c9977f9e54ef08f7a2d", | ||||
|     "blk.21.ffn_down.weight": "b8c082a0530d4b5328e67db0df84c5498f2af956de23c639fa0198ffea853950", | ||||
|     "blk.21.ffn_gate.weight": "cd1b656ee72d00e9835ef667c19ef89a88de261eb8eb7c0e936e0f9ddf83ef9f", | ||||
|     "blk.21.ffn_up.weight": "dc445f73e36ec7a3bd86884186b728f8e0187f32848c3b8b69d4d41f8571bf31", | ||||
|     "blk.22.attn_k.weight": "e37cf0b893ec8b9ee8c78dd139b8d9c45cb997a3bc0c3d93a70ca1c3f6af8859", | ||||
|     "blk.22.attn_norm.weight": "248a27838d3c46cc03a5c312facc84e2e0e2c990ef8401e93da25918497f88d1", | ||||
|     "blk.22.attn_output.weight": "fc191a18f6d18332c66761f7ab28008bfe295dd1f5c8741a2488442f9e00d0f5", | ||||
|     "blk.22.attn_q.weight": "4b193a2ab8bc2b085db18f2bf3eeba26e02b537b2cdd738160c8f14b165d0f5a", | ||||
|     "blk.22.attn_v.weight": "7a60ce5ccac7e045e55ba1e1e85bd2a0f93f8c781daee96c5223665e22f0c666", | ||||
|     "blk.22.ffn_down.weight": "e0a34fb4244e2c7168f3dbaa1904c15d339ec39999cdf27128bbaf619ee0a237", | ||||
|     "blk.22.ffn_gate.weight": "8bac872d4b8549c8812f927efa309f1792b524f33601095fff61b826de5a5615", | ||||
|     "blk.22.ffn_up.weight": "b67fa2b94dd901b6ec64c0853ce8ca2d86fe9cb1cc6d2f15fbbbe0e691c0c648", | ||||
|     "blk.23.attn_k.weight": "2c32e66ad01942b819ac09a197c71579fe66f02226a264fdd72ad1e02c67a27e", | ||||
|     "blk.23.attn_norm.weight": "825fdc94deb439cb93c713eeb077c1052b90ed658d6d464fc4ad3d611e911d48", | ||||
|     "blk.23.attn_output.weight": "95ca6707a95b8750b0c7c5d379d368f0f2e7ebef631954e7d4d8ec0f41f13a3a", | ||||
|     "blk.23.attn_q.weight": "6eccc84faca5fac015d1b26e2854501edcfd292a302228fe14cf99f5eb59a34b", | ||||
|     "blk.23.attn_v.weight": "b343ac3d226040f1033ee049668aa1d89b1774bc18431965682e5dbdce78ccdc", | ||||
|     "blk.23.ffn_down.weight": "9fc599befea8d3b1e342d564a110074f66d2542df406c4b90b6bdc5828fbb2b2", | ||||
|     "blk.23.ffn_gate.weight": "488556c1b0c9f0b20b0c99b4bac2e0f4046b81edb601d7b91e7e5b3bab47d667", | ||||
|     "blk.23.ffn_up.weight": "1088e291d7008dd9c7c2dd6830af686a8a84b724d123a016209bd5156d6898f1", | ||||
|     "blk.24.attn_k.weight": "a923fbe35e61e009a53927d7828818e0592bb737d6a1106c4b0b5a1efc367e07", | ||||
|     "blk.24.attn_norm.weight": "9b51aaaa939cefafdd9b13a7e5b74ac7fa2d603427e55a16a909d6f3f353750a", | ||||
|     "blk.24.attn_output.weight": "1beb2baba56f8409466434b037771248c2f620ec5f53e15f44c271d5a2d9ecf4", | ||||
|     "blk.24.attn_q.weight": "4b0194fe5bfae0c6bf6131dcf8cb6e2b994f6ea10b27cb03574f0f4f8cc0c950", | ||||
|     "blk.24.attn_v.weight": "6ac34b1ab0f66226d85bca1194a7c212cd93d384ecbc8b8395de48aec0970a61", | ||||
|     "blk.24.ffn_down.weight": "5508f74cb732a662c2936b32ac5e90742d172b9f961a747b0e5cba0e5906a89d", | ||||
|     "blk.24.ffn_gate.weight": "095e39b8584403835f9bb1ac33e0e81f54175575e4800273d281b845bff381e7", | ||||
|     "blk.24.ffn_up.weight": "2d43ec21637dda12973de367b0113ee9840b0d815bf6fce042f7c3f270b0b530", | ||||
|     "blk.25.attn_k.weight": "9e2aee029f3d2c7f67dfc7926e72c8228fb978382c8e5a4701bbf82c93801419", | ||||
|     "blk.25.attn_norm.weight": "220cd7164fb4cdbe22d26058e4153b26c27c7b5ce2bec8e95bf2c0ea08d23103", | ||||
|     "blk.25.attn_output.weight": "a17f4a5dc6aa51f03dbd75602d98e9491767c205cdc2c3a5f8667fc54bbf7c64", | ||||
|     "blk.25.attn_q.weight": "f60827496835c440c794bf57ce9780704d10a59d8229886bf75ebb18900ba4ef", | ||||
|     "blk.25.attn_v.weight": "9cac217e9e9f4f4c85f14ee51165a77c580165bd4a34b202389169bbe61a1ced", | ||||
|     "blk.25.ffn_down.weight": "a0f36949b663e80849581dfb71e7babcc73580793bbcb0c80ab26d5a6e000359", | ||||
|     "blk.25.ffn_gate.weight": "df4d1be4d50d6afe5ad3ef0d0e0fac76a33e85c963dea769641d612dd53e7d13", | ||||
|     "blk.25.ffn_up.weight": "992da76be762632e25ebc5ef4d03728eece1b43f7c4e31827df19ca724aea694", | ||||
|     "blk.26.attn_k.weight": "34199ff856ac32a500c754539d070258574192a34ecba87a182897cb59fdff52", | ||||
|     "blk.26.attn_norm.weight": "a8e9dfb2dae5d22b5c0aec5f3675991c0e3c3e6a44153db2579136b73f456e00", | ||||
|     "blk.26.attn_output.weight": "1c4f257ffb0d7db0f11cfb275e38b4af736917b43ad82de1badce3f1d227da4d", | ||||
|     "blk.26.attn_q.weight": "33d55786274c2e718cf61e8fbecf3dfa5ee0c208f0b716d42b061f55459acb3c", | ||||
|     "blk.26.attn_v.weight": "684b636939cd4ffcfec5a6238a0790ffa43d853c95783af9b9e8275e74071a7a", | ||||
|     "blk.26.ffn_down.weight": "89d0bf066db154e6d312b5433aed1714f6a28b40f4c52e3e1530ee07703303c8", | ||||
|     "blk.26.ffn_gate.weight": "393d649bebe5e2940e1b043649f6c860b4b8b9f380f30e9da1744a830f358156", | ||||
|     "blk.26.ffn_up.weight": "179edc85ababd9d8440cc6093eecd1004290aa1cb96434b26ecf7585b6cca17b", | ||||
|     "blk.27.attn_k.weight": "334841445a7f1e14731b08f56eb0b1f0938c63823d28bc6d078c4c5f05b36f19", | ||||
|     "blk.27.attn_norm.weight": "57344471bbda2e9deffdfdb2dd05a07aa47f8761e24de53525588639145bf551", | ||||
|     "blk.27.attn_output.weight": "506126af9ee54b535d49f97e36f630e74834f480329f098d6d62e96246d8d65a", | ||||
|     "blk.27.attn_q.weight": "dd984df1acb4783849e25ba7ae378bfd385cd9efc540fb798cd5bdd873f0118f", | ||||
|     "blk.27.attn_v.weight": "b4b3fe9a4455d34c297ff20a2f537b647cef424741d840a747b265f23d320ac0", | ||||
|     "blk.27.ffn_down.weight": "621fdb185ba0d35ba5476dae73d2c81ec1482a0e878d5bfd5c3b29fe837af013", | ||||
|     "blk.27.ffn_gate.weight": "e4fbab45f2ec506fa374103251a0bdb7baa6f576080bdd796f3e9db92098e08f", | ||||
|     "blk.27.ffn_up.weight": "a0c57e463e988002bbd6a6c6792baa21a65e6f89ae303a2c301951b0ae6e4bbe", | ||||
|     "blk.28.attn_k.weight": "bac36cbd52ec5056841663865e1291ddab4b47ef9a2544dd285d4503bfb0e4a0", | ||||
|     "blk.28.attn_norm.weight": "5774a9df2bbb2e86d1f70179c7b92d81e1f401160148b3328fb64db6646a5425", | ||||
|     "blk.28.attn_output.weight": "e8712622d1569557000c75f26c3f55fad267fd300463c2c2cfe3afbfa1c8f908", | ||||
|     "blk.28.attn_q.weight": "11677751fddee52cc739699c02836f7be54d96038be4240be5d4f53d00161608", | ||||
|     "blk.28.attn_v.weight": "e5ee459b8958d65e1445997b9aa1e90e2f5d17761ebcf5357313119a45322507", | ||||
|     "blk.28.ffn_down.weight": "3934518f9f85292da8475fe38a8edcbfc4e24ac56c351b472d6351f98750871e", | ||||
|     "blk.28.ffn_gate.weight": "6ba735d57e98d0847e487f25ffaa25256deaa8abec76f428cb70bd9774279d83", | ||||
|     "blk.28.ffn_up.weight": "977fae6e1e5353114fc645dd98429464749758765cbc6e6457593d596e57850c", | ||||
|     "blk.29.attn_k.weight": "8122a457307d580ad6f1e0acea09a2f593d97f595ba0d6737f5fea16d2433642", | ||||
|     "blk.29.attn_norm.weight": "d626f721e05aa1202439b01027031d4caf1adace61ed37870a277cb6297c77cc", | ||||
|     "blk.29.attn_output.weight": "7fb7122ab1b6b1e6615ca746897da27bc52c92cb70d3147183cdde61795b72b3", | ||||
|     "blk.29.attn_q.weight": "be43e94ff6b6e391024dc824101efa0ddf4005d5b002ac26cb03765c0c73c2fa", | ||||
|     "blk.29.attn_v.weight": "af93c85ebff908f74f9935b81bde0516ca487c84139868a1ce079c3ae20036b1", | ||||
|     "blk.29.ffn_down.weight": "39dae12340ed3120bd19c495fe0872b559613641e41fde69d02d8631900b84c0", | ||||
|     "blk.29.ffn_gate.weight": "36fd482439840ef197c9f3b8905d86acfcea49bcf018544106ca465d4bf8d5c7", | ||||
|     "blk.29.ffn_up.weight": "5243fbdfdc1e2a1dd84b6210a9869d18a014db9088897e345240cdc99990bd5d", | ||||
|     "blk.30.attn_k.weight": "948f263616bd3788b2b968baafd69b9c5bd1b77578665f096c4b7e247b4cea42", | ||||
|     "blk.30.attn_norm.weight": "e168df981e744874ff303faf2eb470e5f6868c2040ba5f383f6c5148669975e7", | ||||
|     "blk.30.attn_output.weight": "4cf0ccca04b792573b756655a24fc89cfb1f272da8305633f0bc66ef14990b93", | ||||
|     "blk.30.attn_q.weight": "21e07d6cba6c50d65350289258209717174a13c42be57e8141d69712cbaf32c1", | ||||
|     "blk.30.attn_v.weight": "65a8ca29c7237b3182ccf03e2fc94e84f9a53d0e160fb679ab401c853170dd9c", | ||||
|     "blk.30.ffn_down.weight": "8b00500a6d00d84058f6658ee1d6f06fb4fcae2f90d4341792259362923b3c13", | ||||
|     "blk.30.ffn_gate.weight": "5bc0e19ab7a31b50ac2118ad1b36e31055271a322cd8ff661d47c3ac0210703c", | ||||
|     "blk.30.ffn_up.weight": "f37a0561955725bd59ee2d064fa9f4e00a12a1b620b624db3bc3add5330bc321", | ||||
|     "blk.31.attn_k.weight": "9a5663edda227f5d87533897146764f8e8a7481b9e71fae197c39204f8463221", | ||||
|     "blk.31.attn_norm.weight": "060a4f438a1ee5e220b5b5278ad2f5c085a428bf38c515766781815597c87529", | ||||
|     "blk.31.attn_output.weight": "6ada5d3cad9dea4780ffbb43302bb6ccc2f24eddd0fc4f5f84c9ce0fc0c6e5dd", | ||||
|     "blk.31.attn_q.weight": "bb5d08c08603907981ad388d5d8b70fcc9b98034ba264b8474c8890cc0297af0", | ||||
|     "blk.31.attn_v.weight": "e01b4252ea9c6a889c32b21144b441a347464d04536ef4f6572425be55759796", | ||||
|     "blk.31.ffn_down.weight": "8ba4d679c36e93ba65ba03180385ef35ea86b3b7cdf2fded9df59369f1c09630", | ||||
|     "blk.31.ffn_gate.weight": "e5b41dc93645f8b5e8eebae3ada3ea43a18f97ce2654228655170b07b463ccb0", | ||||
|     "blk.31.ffn_up.weight": "25b88cdddc8b547af294ed107d3d1312e90b983cae87936fa6062ecd8ea02539", | ||||
|     "blk.32.attn_k.weight": "4bcf86dc0858c8ca2fbdf6aa76674d43eb698f78979fdc1a38f556a7af1facc4", | ||||
|     "blk.32.attn_norm.weight": "cdcc12f3b8b9773c6722736bfb748a2729230b21478cbcc4104859d3148df815", | ||||
|     "blk.32.attn_output.weight": "d43f1196822995ed89a9365c97054753a8b30ce20b6e273c8edcc42673a1e141", | ||||
|     "blk.32.attn_q.weight": "ebf2972bb3865cbc5be4840113a322089752038344beab2a0122c7cb4fb399b6", | ||||
|     "blk.32.attn_v.weight": "714db81704ff34fa137512903c1013acee7877467473e46600728b9240582eb7", | ||||
|     "blk.32.ffn_down.weight": "2cde3da1258bb170a79d5d3cdfe10c86a71eb34b77da46b74c5ed71e7f4fe274", | ||||
|     "blk.32.ffn_gate.weight": "c7e1ed792532613ff9d4e5834b6536e2e0f47df2303bc0fdaa90aac0c1f4e8db", | ||||
|     "blk.32.ffn_up.weight": "d8d6f13fe66a716e28f79101a29817f0c0d6f99969a6f017d51bafd1a16c600c", | ||||
|     "blk.33.attn_k.weight": "a0a28f6cbca88da00cab2ca37094d9b0503bf9defdae77b91895b911c408cbb6", | ||||
|     "blk.33.attn_norm.weight": "0251200c24cc8445607ace6dc8c5aa0566567997262b7cca53a11ac23cc564b2", | ||||
|     "blk.33.attn_output.weight": "b2423205bdf6a1096d43c44d8d12f1a84fcd4e1bb70fcf6dc8542b8b8a71a13c", | ||||
|     "blk.33.attn_q.weight": "00b425c3ef71065ce5e0234e702bf38143b4952da78a85f52ab2c2e3073d97ab", | ||||
|     "blk.33.attn_v.weight": "035edd2335df816c42c765a5e66b9d9b9e15a822a8dc1863508145499c942c14", | ||||
|     "blk.33.ffn_down.weight": "4894a923a3db75bae4496ba3ce5f28796ad31fe33996a066271fb8654964310e", | ||||
|     "blk.33.ffn_gate.weight": "8f6c819b8bbfbe3357fae89e1ac5a3d58be85b3b04be3bacf7b62775869046ff", | ||||
|     "blk.33.ffn_up.weight": "257c3544b5b544fd5d839665bf5caf107a329b59dbc3751efcaa24ae63c56179", | ||||
|     "blk.34.attn_k.weight": "b6cd8bba892e38dac4a2ebc3ba1bce49e71b967fc436fde30c6d76f54a18935f", | ||||
|     "blk.34.attn_norm.weight": "2b3c8e60a064cba9955752bbbbdd92c71ba5c2f1bd721097bdbe88b5abc68787", | ||||
|     "blk.34.attn_output.weight": "8cc272551c9aaca9db5a660c6927bab94a0243d74a30b2bc165f06bd577714ea", | ||||
|     "blk.34.attn_q.weight": "74b561eb4792484e6a94b58fe2583848c3ae28ff2f1bf3d02939a0cfdfa49990", | ||||
|     "blk.34.attn_v.weight": "dba19e24ff05154dc5a1f55c023729303a583d13d68732ce22ea74d4410dc8f0", | ||||
|     "blk.34.ffn_down.weight": "76eca5dfeb274c35774e0bf9f22ee420ed9085c8e99aa2cd5a236e4918b44c61", | ||||
|     "blk.34.ffn_gate.weight": "9af0862d5fcbc24732846488e653db8242a467765c0cdbc00332b3a40256b4a6", | ||||
|     "blk.34.ffn_up.weight": "2a03126bf73587eaba99ece2066103d12e47bcd4ce30ff6c17b2f383b81d40df", | ||||
|     "blk.35.attn_k.weight": "52513fc0cd4e997a842729af7d21dd09399bce0a339558374738be266d0fa2f0", | ||||
|     "blk.35.attn_norm.weight": "e5281fa911964263ccf1630b14762edbd41d0b9472d6ec695fc600fed4892c35", | ||||
|     "blk.35.attn_output.weight": "b391d6705d5dc6f48326b5fd16573f679edf64109d86fb729a498819676590ca", | ||||
|     "blk.35.attn_q.weight": "d16446921966db9b0e0539626ad22a2511ace780e59379d6a4162d8c5441440b", | ||||
|     "blk.35.attn_v.weight": "9d8cdf23ffdb0c5c74106843390b94b24c9f33ef0eb9998d39f78c73390101ea", | ||||
|     "blk.35.ffn_down.weight": "938eb6301f7bbf162d7dd965682a5ed11d0a4a530c6fedd7e5469ce80012fc17", | ||||
|     "blk.35.ffn_gate.weight": "5ad84f5a0c8edcfea1ecf1a3e3d21d85ceda0c4ad9e3c6ca68885eeff8ed3c2f", | ||||
|     "blk.35.ffn_up.weight": "1c4330d9dc71bf4c98812c34356c51f520f47610a534152aa6d29284b758090d", | ||||
|     "blk.36.attn_k.weight": "ef720655e5ca2465f13db2dfc4732fb4ef2c9d53acde52f514fd4f301e974081", | ||||
|     "blk.36.attn_norm.weight": "88f4b9310b3c8c2644e3029160cd35678c79dfa59280430e03f5c29a6fe84a58", | ||||
|     "blk.36.attn_output.weight": "aec6f915fffd7bb72cd783273e871b4f09605950089d45e72059d1316b6c4b01", | ||||
|     "blk.36.attn_q.weight": "72f9408a2405d42f8db6ce5fcf1d26a3660b6f225fc60e77d0277109cfcb82ed", | ||||
|     "blk.36.attn_v.weight": "0f3b3d851dc44b3893ef53f6cca5b4acc9658bacfe1cc2d13c3d704ddd409b67", | ||||
|     "blk.36.ffn_down.weight": "470aec48ce8c5129a6654d9fd26fcae72776f9fc1429a8bb05818072a876475d", | ||||
|     "blk.36.ffn_gate.weight": "7f5f296d09cf55679767b5d15de3eff489c456782119f25204be4b1647f18dcf", | ||||
|     "blk.36.ffn_up.weight": "b7ef74a1f7ffb4982711d93f1787be3a70edc3d2358d5203c41d8900508037d4", | ||||
|     "blk.37.attn_k.weight": "c4ffa5412e4ff2dcfe1aed991c1f54169fd171a4c7638e4b9f21a1ca64c5e1d6", | ||||
|     "blk.37.attn_norm.weight": "4eb6c888d841cccfacf5b963f8611120f6ff24b84af0b5714fd9ab36dcda422f", | ||||
|     "blk.37.attn_output.weight": "db2a7bbf9682f9f6eea672dae8e150738f1bf74dbc80edc7022017a3f040c8ac", | ||||
|     "blk.37.attn_q.weight": "e38c0462aff139afcbab289189823527e453abc9e541154adde5e7af88cacf0b", | ||||
|     "blk.37.attn_v.weight": "952eb2492ed452a72f96bcc12d4b2affad9dfdf46ee39ce4a5d7b57a5dc301e5", | ||||
|     "blk.37.ffn_down.weight": "25f23a8fbc44febf6dc4848fd7fe03a580e2822bd3b3b5a51f4990826bfe3e4e", | ||||
|     "blk.37.ffn_gate.weight": "707da5eb40118b035305d3262444382351f170a20a537386a70e90c5a83a7817", | ||||
|     "blk.37.ffn_up.weight": "d2d2ba5cfc4ef47338dd7384219e22bf030a5a2209e0354d88f5bbaaafd20e87", | ||||
|     "blk.38.attn_k.weight": "abc4bb189dedf7ce661e79028427623a4f91ac091c2cd60e31b58bc62b1cda71", | ||||
|     "blk.38.attn_norm.weight": "9f4803a7d03fd40fcb83d85f84eb1d5682ea4e5bb084f210c02850675d804c3d", | ||||
|     "blk.38.attn_output.weight": "77cb66007f1a41df7135d0e7f900ceb499c2f667dfc3f1a6ac01a3203bbd3ccf", | ||||
|     "blk.38.attn_q.weight": "d94a8b26cd375bf2bcaa76597e314aa8268ee50a479d00931e5e0e021feadb5d", | ||||
|     "blk.38.attn_v.weight": "660c907888bc5016dc69b7d35fe6f55c7ded697c93be0e2d332a2f17aff88758", | ||||
|     "blk.38.ffn_down.weight": "6f06173bae5b00ffaf88ef383619a8b9c6a8d0d5c6494695d17f6c1de1a68a13", | ||||
|     "blk.38.ffn_gate.weight": "89f99be149d03f116527bfcabe073c50001c874de40fb6e817f6619027f3cd05", | ||||
|     "blk.38.ffn_up.weight": "8d57557c8d5e2d2688b73f01dddf1ce8d5194990cda6358153320aea88aac7f8", | ||||
|     "blk.39.attn_k.weight": "21be09c988b46c8393e6c2ec9230f3b5136eb7607dd1953ba92d0811c2f0dd75", | ||||
|     "blk.39.attn_norm.weight": "ba7c1912dd1c4e2d16917201f62396fd0600e4a451137eaddff255548c209abd", | ||||
|     "blk.39.attn_output.weight": "acfaf4abb3fd27fd899b5563c3877f176b597d8f6cdb2f2fd3f3a0bd4da15ed6", | ||||
|     "blk.39.attn_q.weight": "e8adbc140d4c8f0db2a27ca584c5531d5b1e080555fe627e34d80d0814a92bed", | ||||
|     "blk.39.attn_v.weight": "92f96b0e1f724e73a0f90a76c145654418844c04a6d4b14c05eb5af8a62bf8dc", | ||||
|     "blk.39.ffn_down.weight": "4d9ee7c65fc16fe95d10c47b79ac6a525741947600a64b5fcea5d300a82c50de", | ||||
|     "blk.39.ffn_gate.weight": "7e18507989f39b32191133d2657c2ee3b74f42f070579204d727eb72215793d1", | ||||
|     "blk.39.ffn_up.weight": "22cda752269c9757ba918abede1df95bb0f83a5c772dea13c8deea3d5f2723d9", | ||||
|     "output_norm.weight": "2858cf0e39d32caf52b7861378ace076000241e147f10b9eb21d8a5cd149e3cb" | ||||
| } | ||||
| @@ -6,7 +6,9 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io/fs" | ||||
| 	"log/slog" | ||||
| 	"os" | ||||
| 	"reflect" | ||||
| 	"slices" | ||||
|  | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| @@ -15,6 +17,8 @@ import ( | ||||
| ) | ||||
|  | ||||
| func parseSentencePiece(fsys fs.FS) (*Vocabulary, error) { | ||||
| 	slog.Debug("using spm vocabulary") | ||||
|  | ||||
| 	ast, err := parseAdditionalSpecialTokens(fsys) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @@ -43,10 +47,19 @@ func parseSentencePiece(fsys fs.FS) (*Vocabulary, error) { | ||||
| 			v.Types = append(v.Types, int32(t)) | ||||
| 		default: | ||||
| 			tt := int32(sentencepiece.ModelProto_SentencePiece_NORMAL) | ||||
| 			if slices.Contains(ast, piece.GetPiece()) { | ||||
|  | ||||
| 			// temporary fix to handle gemma3 broken configs | ||||
| 			if slices.Contains([]string{"<end_of_turn>", "<start_of_turn>"}, piece.GetPiece()) { | ||||
| 				tt = int32(sentencepiece.ModelProto_SentencePiece_CONTROL) | ||||
| 			} | ||||
|  | ||||
| 			for _, t := range ast { | ||||
| 				if t.Content == piece.GetPiece() { | ||||
| 					tt = int32(sentencepiece.ModelProto_SentencePiece_CONTROL) | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			v.Types = append(v.Types, tt) | ||||
| 		} | ||||
| 	} | ||||
| @@ -78,10 +91,16 @@ func parseSentencePiece(fsys fs.FS) (*Vocabulary, error) { | ||||
| 		return cmp.Compare(i.id, j.id) | ||||
| 	}) | ||||
|  | ||||
| 	n := len(v.Tokens) | ||||
| 	for i, t := range ts { | ||||
| 		if t.id != i+n { | ||||
| 			return nil, fmt.Errorf("invalid token id: %d", t.id) | ||||
| 	for _, t := range ts { | ||||
| 		if t.id < len(v.Tokens) { | ||||
| 			if v.Tokens[t.id] == t.content { | ||||
| 				slog.Warn("tokenizer", "duplicate token", t.content, "id", t.id) | ||||
| 				continue | ||||
| 			} | ||||
| 			return nil, fmt.Errorf("token mismatch: %s != %s at pos [%d]", t.content, v.Tokens[t.id], t.id) | ||||
| 		} | ||||
| 		if t.id != len(v.Tokens) { | ||||
| 			return nil, fmt.Errorf("invalid token id: [%d] as pos [%d]", t.id, len(v.Tokens)) | ||||
| 		} | ||||
|  | ||||
| 		v.Tokens = append(v.Tokens, t.content) | ||||
| @@ -92,7 +111,15 @@ func parseSentencePiece(fsys fs.FS) (*Vocabulary, error) { | ||||
| 	return &v, nil | ||||
| } | ||||
|  | ||||
| func parseAdditionalSpecialTokens(fsys fs.FS) ([]string, error) { | ||||
| type specialToken struct { | ||||
| 	Content    string `json:"content"` | ||||
| 	Lstrip     bool   `json:"lstrip"` | ||||
| 	Normalized bool   `json:"normalized"` | ||||
| 	Rstrip     bool   `json:"rstrip"` | ||||
| 	SingleWord bool   `json:"single_word"` | ||||
| } | ||||
|  | ||||
| func parseAdditionalSpecialTokens(fsys fs.FS) ([]specialToken, error) { | ||||
| 	f, err := fsys.Open("special_tokens_map.json") | ||||
| 	if errors.Is(err, os.ErrNotExist) { | ||||
| 		return nil, nil | ||||
| @@ -102,12 +129,43 @@ func parseAdditionalSpecialTokens(fsys fs.FS) ([]string, error) { | ||||
| 	defer f.Close() | ||||
|  | ||||
| 	var m struct { | ||||
| 		AdditionalSpecialTokens []string `json:"additional_special_tokens"` | ||||
| 		AdditionalSpecialTokens any `json:"additional_special_tokens"` | ||||
| 	} | ||||
|  | ||||
| 	if err := json.NewDecoder(f).Decode(&m); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return m.AdditionalSpecialTokens, nil | ||||
| 	var ast []specialToken | ||||
|  | ||||
| 	switch st := m.AdditionalSpecialTokens.(type) { | ||||
| 	case []string: | ||||
| 		for _, s := range st { | ||||
| 			ast = append(ast, specialToken{Content: s}) | ||||
| 		} | ||||
| 	case []any: | ||||
| 		for _, s := range st { | ||||
| 			// marshal and unmarshal the object to get the special token | ||||
| 			tMap := s.(map[string]any) | ||||
| 			data, err := json.Marshal(tMap) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
|  | ||||
| 			var token specialToken | ||||
| 			err = json.Unmarshal(data, &token) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
|  | ||||
| 			ast = append(ast, token) | ||||
| 		} | ||||
|  | ||||
| 	default: | ||||
| 		slog.Warn("special token", "unknown token", reflect.TypeOf(st)) | ||||
| 	} | ||||
|  | ||||
| 	slog.Debug("spm tokenizer", "additional tokens", ast) | ||||
|  | ||||
| 	return ast, nil | ||||
| } | ||||
|   | ||||
| @@ -9,8 +9,6 @@ import ( | ||||
| 	"path/filepath" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/ollama/ollama/envconfig" | ||||
| ) | ||||
|  | ||||
| // Determine if the given ROCm lib directory is usable by checking for existence of some glob patterns | ||||
| @@ -41,13 +39,10 @@ func commonAMDValidateLibDir() (string, error) { | ||||
| 	// Favor our bundled version | ||||
|  | ||||
| 	// Installer payload location if we're running the installed binary | ||||
| 	exe, err := os.Executable() | ||||
| 	if err == nil { | ||||
| 		rocmTargetDir := filepath.Join(filepath.Dir(exe), envconfig.LibRelativeToExe(), "lib", "ollama") | ||||
| 		if rocmLibUsable(rocmTargetDir) { | ||||
| 			slog.Debug("detected ROCM next to ollama executable " + rocmTargetDir) | ||||
| 			return rocmTargetDir, nil | ||||
| 		} | ||||
| 	rocmTargetDir := filepath.Join(LibOllamaPath, "rocm") | ||||
| 	if rocmLibUsable(rocmTargetDir) { | ||||
| 		slog.Debug("detected ROCM next to ollama executable " + rocmTargetDir) | ||||
| 		return rocmTargetDir, nil | ||||
| 	} | ||||
|  | ||||
| 	// Prefer explicit HIP env var | ||||
|   | ||||
| @@ -77,8 +77,7 @@ func AMDGetGPUInfo() ([]RocmGPUInfo, error) { | ||||
|  | ||||
| 	gfxOverride := envconfig.HsaOverrideGfxVersion() | ||||
| 	var supported []string | ||||
| 	depPaths := LibraryDirs() | ||||
| 	libDir := "" | ||||
| 	var libDir string | ||||
|  | ||||
| 	// The amdgpu driver always exposes the host CPU(s) first, but we have to skip them and subtract | ||||
| 	// from the other IDs to get alignment with the HIP libraries expectations (zero is the first GPU, not the CPU) | ||||
| @@ -353,9 +352,8 @@ func AMDGetGPUInfo() ([]RocmGPUInfo, error) { | ||||
| 				}) | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			depPaths = append(depPaths, libDir) | ||||
| 		} | ||||
| 		gpuInfo.DependencyPath = depPaths | ||||
| 		gpuInfo.DependencyPath = []string{libDir} | ||||
|  | ||||
| 		if gfxOverride == "" { | ||||
| 			// Only load supported list once | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"log/slog" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"slices" | ||||
| 	"strconv" | ||||
| @@ -50,14 +49,13 @@ func AMDGetGPUInfo() ([]RocmGPUInfo, error) { | ||||
| 		slog.Info(err.Error()) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	depPaths := LibraryDirs() | ||||
|  | ||||
| 	libDir, err := AMDValidateLibDir() | ||||
| 	if err != nil { | ||||
| 		err = fmt.Errorf("unable to verify rocm library: %w", err) | ||||
| 		slog.Warn(err.Error()) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	depPaths = append(depPaths, libDir) | ||||
|  | ||||
| 	var supported []string | ||||
| 	gfxOverride := envconfig.HsaOverrideGfxVersion() | ||||
| @@ -113,7 +111,7 @@ func AMDGetGPUInfo() ([]RocmGPUInfo, error) { | ||||
| 				UnreliableFreeMemory: true, | ||||
|  | ||||
| 				ID:             strconv.Itoa(i), // TODO this is probably wrong if we specify visible devices | ||||
| 				DependencyPath: depPaths, | ||||
| 				DependencyPath: []string{libDir}, | ||||
| 				MinimumMemory:  rocmMinimumMemory, | ||||
| 				Name:           name, | ||||
| 				Compute:        gfx, | ||||
| @@ -164,9 +162,7 @@ func AMDValidateLibDir() (string, error) { | ||||
| 	} | ||||
|  | ||||
| 	// Installer payload (if we're running from some other location) | ||||
| 	localAppData := os.Getenv("LOCALAPPDATA") | ||||
| 	appDir := filepath.Join(localAppData, "Programs", "Ollama") | ||||
| 	rocmTargetDir := filepath.Join(appDir, envconfig.LibRelativeToExe(), "lib", "ollama") | ||||
| 	rocmTargetDir := filepath.Join(LibOllamaPath, "rocm") | ||||
| 	if rocmLibUsable(rocmTargetDir) { | ||||
| 		slog.Debug("detected ollama installed ROCm at " + rocmTargetDir) | ||||
| 		return rocmTargetDir, nil | ||||
|   | ||||
| @@ -12,7 +12,7 @@ func IsNUMA() bool { | ||||
| 		// numa support in llama.cpp is linux only | ||||
| 		return false | ||||
| 	} | ||||
| 	ids := map[string]interface{}{} | ||||
| 	ids := map[string]any{} | ||||
| 	packageIds, _ := filepath.Glob("/sys/devices/system/cpu/cpu*/topology/physical_package_id") | ||||
| 	for _, packageId := range packageIds { | ||||
| 		id, err := os.ReadFile(packageId) | ||||
|   | ||||
| @@ -57,7 +57,8 @@ func cudaVariant(gpuInfo CudaGPUInfo) string { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if gpuInfo.computeMajor < 6 || gpuInfo.DriverMajor < 12 || (gpuInfo.DriverMajor == 12 && gpuInfo.DriverMinor == 0) { | ||||
| 	// driver 12.0 has problems with the cuda v12 library, so run v11 on those older drivers | ||||
| 	if gpuInfo.DriverMajor < 12 || (gpuInfo.DriverMajor == 12 && gpuInfo.DriverMinor == 0) { | ||||
| 		return "v11" | ||||
| 	} | ||||
| 	return "v12" | ||||
|   | ||||
| @@ -23,7 +23,6 @@ import ( | ||||
|  | ||||
| 	"github.com/ollama/ollama/envconfig" | ||||
| 	"github.com/ollama/ollama/format" | ||||
| 	"github.com/ollama/ollama/runners" | ||||
| ) | ||||
|  | ||||
| type cudaHandles struct { | ||||
| @@ -101,15 +100,7 @@ func initCudaHandles() *cudaHandles { | ||||
|  | ||||
| 	// Aligned with driver, we can't carry as payloads | ||||
| 	nvcudaMgmtPatterns := NvcudaGlobs | ||||
|  | ||||
| 	if runtime.GOOS == "windows" { | ||||
| 		localAppData := os.Getenv("LOCALAPPDATA") | ||||
| 		cudartMgmtPatterns = []string{filepath.Join(localAppData, "Programs", "Ollama", CudartMgmtName)} | ||||
| 	} | ||||
| 	libDirs := LibraryDirs() | ||||
| 	for _, d := range libDirs { | ||||
| 		cudartMgmtPatterns = append(cudartMgmtPatterns, filepath.Join(d, CudartMgmtName)) | ||||
| 	} | ||||
| 	cudartMgmtPatterns = append(cudartMgmtPatterns, filepath.Join(LibOllamaPath, "cuda_v*", CudartMgmtName)) | ||||
| 	cudartMgmtPatterns = append(cudartMgmtPatterns, CudartGlobs...) | ||||
|  | ||||
| 	if len(NvmlGlobs) > 0 { | ||||
| @@ -240,7 +231,7 @@ func GetGPUInfo() GpuInfoList { | ||||
| 		if err != nil { | ||||
| 			slog.Warn("error looking up system memory", "error", err) | ||||
| 		} | ||||
| 		depPaths := LibraryDirs() | ||||
|  | ||||
| 		details, err := GetCPUDetails() | ||||
| 		if err != nil { | ||||
| 			slog.Warn("failed to lookup CPU details", "error", err) | ||||
| @@ -248,11 +239,9 @@ func GetGPUInfo() GpuInfoList { | ||||
| 		cpus = []CPUInfo{ | ||||
| 			{ | ||||
| 				GpuInfo: GpuInfo{ | ||||
| 					memInfo:        mem, | ||||
| 					Library:        "cpu", | ||||
| 					Variant:        runners.GetCPUCapability().String(), | ||||
| 					ID:             "0", | ||||
| 					DependencyPath: depPaths, | ||||
| 					memInfo: mem, | ||||
| 					Library: "cpu", | ||||
| 					ID:      "0", | ||||
| 				}, | ||||
| 				CPUs: details, | ||||
| 			}, | ||||
| @@ -294,17 +283,13 @@ func GetGPUInfo() GpuInfoList { | ||||
| 				gpuInfo.DriverMajor = driverMajor | ||||
| 				gpuInfo.DriverMinor = driverMinor | ||||
| 				variant := cudaVariant(gpuInfo) | ||||
| 				if depPaths != nil { | ||||
| 					gpuInfo.DependencyPath = depPaths | ||||
| 					// Check for variant specific directory | ||||
| 					if variant != "" { | ||||
| 						for _, d := range depPaths { | ||||
| 							if _, err := os.Stat(filepath.Join(d, "cuda_"+variant)); err == nil { | ||||
| 								// Put the variant directory first in the search path to avoid runtime linking to the wrong library | ||||
| 								gpuInfo.DependencyPath = append([]string{filepath.Join(d, "cuda_"+variant)}, gpuInfo.DependencyPath...) | ||||
| 								break | ||||
| 							} | ||||
| 						} | ||||
|  | ||||
| 				// Start with our bundled libraries | ||||
| 				if variant != "" { | ||||
| 					variantPath := filepath.Join(LibOllamaPath, "cuda_"+variant) | ||||
| 					if _, err := os.Stat(variantPath); err == nil { | ||||
| 						// Put the variant directory first in the search path to avoid runtime linking to the wrong library | ||||
| 						gpuInfo.DependencyPath = append([]string{variantPath}, gpuInfo.DependencyPath...) | ||||
| 					} | ||||
| 				} | ||||
| 				gpuInfo.Name = C.GoString(&memInfo.gpu_name[0]) | ||||
| @@ -376,7 +361,7 @@ func GetGPUInfo() GpuInfoList { | ||||
| 						gpuInfo.FreeMemory = uint64(memInfo.free) | ||||
| 						gpuInfo.ID = C.GoString(&memInfo.gpu_id[0]) | ||||
| 						gpuInfo.Name = C.GoString(&memInfo.gpu_name[0]) | ||||
| 						gpuInfo.DependencyPath = depPaths | ||||
| 						gpuInfo.DependencyPath = []string{LibOllamaPath} | ||||
| 						oneapiGPUs = append(oneapiGPUs, gpuInfo) | ||||
| 					} | ||||
| 				} | ||||
| @@ -512,33 +497,30 @@ func GetGPUInfo() GpuInfoList { | ||||
|  | ||||
| func FindGPULibs(baseLibName string, defaultPatterns []string) []string { | ||||
| 	// Multiple GPU libraries may exist, and some may not work, so keep trying until we exhaust them | ||||
| 	var ldPaths []string | ||||
| 	gpuLibPaths := []string{} | ||||
| 	slog.Debug("Searching for GPU library", "name", baseLibName) | ||||
|  | ||||
| 	// Start with our bundled libraries | ||||
| 	patterns := []string{} | ||||
| 	for _, d := range LibraryDirs() { | ||||
| 		patterns = append(patterns, filepath.Join(d, baseLibName)) | ||||
| 	} | ||||
| 	// search our bundled libraries first | ||||
| 	patterns := []string{filepath.Join(LibOllamaPath, baseLibName)} | ||||
|  | ||||
| 	var ldPaths []string | ||||
| 	switch runtime.GOOS { | ||||
| 	case "windows": | ||||
| 		ldPaths = strings.Split(os.Getenv("PATH"), ";") | ||||
| 		ldPaths = strings.Split(os.Getenv("PATH"), string(os.PathListSeparator)) | ||||
| 	case "linux": | ||||
| 		ldPaths = strings.Split(os.Getenv("LD_LIBRARY_PATH"), ":") | ||||
| 	default: | ||||
| 		return gpuLibPaths | ||||
| 		ldPaths = strings.Split(os.Getenv("LD_LIBRARY_PATH"), string(os.PathListSeparator)) | ||||
| 	} | ||||
|  | ||||
| 	// Then with whatever we find in the PATH/LD_LIBRARY_PATH | ||||
| 	for _, ldPath := range ldPaths { | ||||
| 		d, err := filepath.Abs(ldPath) | ||||
| 	// then search the system's LD_LIBRARY_PATH | ||||
| 	for _, p := range ldPaths { | ||||
| 		p, err := filepath.Abs(p) | ||||
| 		if err != nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		patterns = append(patterns, filepath.Join(d, baseLibName)) | ||||
| 		patterns = append(patterns, filepath.Join(p, baseLibName)) | ||||
| 	} | ||||
|  | ||||
| 	// finally, search the default patterns provided by the caller | ||||
| 	patterns = append(patterns, defaultPatterns...) | ||||
| 	slog.Debug("gpu library search", "globs", patterns) | ||||
| 	for _, pattern := range patterns { | ||||
| @@ -715,28 +697,6 @@ func (l GpuInfoList) GetVisibleDevicesEnv() (string, string) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func LibraryDirs() []string { | ||||
| 	// dependencies can exist wherever we found the runners (e.g. build tree for developers) and relative to the executable | ||||
| 	// This can be simplified once we no longer carry runners as payloads | ||||
| 	paths := []string{} | ||||
| 	appExe, err := os.Executable() | ||||
| 	if err != nil { | ||||
| 		slog.Warn("failed to lookup executable path", "error", err) | ||||
| 	} else { | ||||
| 		appRelative := filepath.Join(filepath.Dir(appExe), envconfig.LibRelativeToExe(), "lib", "ollama") | ||||
| 		if _, err := os.Stat(appRelative); err == nil { | ||||
| 			paths = append(paths, appRelative) | ||||
| 		} | ||||
| 	} | ||||
| 	rDir := runners.Locate() | ||||
| 	if err != nil { | ||||
| 		slog.Warn("unable to locate gpu dependency libraries", "error", err) | ||||
| 	} else { | ||||
| 		paths = append(paths, filepath.Dir(rDir)) | ||||
| 	} | ||||
| 	return paths | ||||
| } | ||||
|  | ||||
| func GetSystemInfo() SystemInfo { | ||||
| 	gpus := GetGPUInfo() | ||||
| 	gpuMutex.Lock() | ||||
|   | ||||
| @@ -15,7 +15,6 @@ import ( | ||||
| 	"syscall" | ||||
|  | ||||
| 	"github.com/ollama/ollama/format" | ||||
| 	"github.com/ollama/ollama/runners" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -28,7 +27,6 @@ func GetGPUInfo() GpuInfoList { | ||||
| 		return []GpuInfo{ | ||||
| 			{ | ||||
| 				Library: "cpu", | ||||
| 				Variant: runners.GetCPUCapability().String(), | ||||
| 				memInfo: mem, | ||||
| 			}, | ||||
| 		} | ||||
| @@ -51,7 +49,6 @@ func GetCPUInfo() GpuInfoList { | ||||
| 	return []GpuInfo{ | ||||
| 		{ | ||||
| 			Library: "cpu", | ||||
| 			Variant: runners.GetCPUCapability().String(), | ||||
| 			memInfo: mem, | ||||
| 		}, | ||||
| 	} | ||||
|   | ||||
| @@ -111,6 +111,7 @@ func GetCPUDetails() ([]CPU, error) { | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	defer file.Close() | ||||
| 	return linuxCPUDetails(file) | ||||
| } | ||||
|  | ||||
| @@ -168,13 +169,11 @@ func linuxCPUDetails(file io.Reader) ([]CPU, error) { | ||||
| 	for id, s := range socketByID { | ||||
| 		s.CoreCount = len(coreBySocket[id]) | ||||
| 		s.ThreadCount = 0 | ||||
| 		for _, tc := range threadsByCoreBySocket[id] { | ||||
| 			s.ThreadCount += tc | ||||
| 		} | ||||
|  | ||||
| 		// This only works if HT is enabled, consider a more reliable model, maybe cache size comparisons? | ||||
| 		efficiencyCoreCount := 0 | ||||
| 		for _, threads := range threadsByCoreBySocket[id] { | ||||
| 			s.ThreadCount += threads | ||||
| 			if threads == 1 { | ||||
| 				efficiencyCoreCount++ | ||||
| 			} | ||||
|   | ||||
							
								
								
									
										56
									
								
								discover/path.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								discover/path.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| package discover | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"runtime" | ||||
| ) | ||||
|  | ||||
| // LibPath is a path to lookup dynamic libraries | ||||
| // in development it's usually 'build/lib/ollama' | ||||
| // in distribution builds it's 'lib/ollama' on Windows | ||||
| // '../lib/ollama' on Linux and the executable's directory on macOS | ||||
| // note: distribution builds, additional GPU-specific libraries are | ||||
| // found in subdirectories of the returned path, such as | ||||
| // 'cuda_v11', 'cuda_v12', 'rocm', etc. | ||||
| var LibOllamaPath string = func() string { | ||||
| 	exe, err := os.Executable() | ||||
| 	if err != nil { | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	if eval, err := filepath.EvalSymlinks(exe); err == nil { | ||||
| 		exe = eval | ||||
| 	} | ||||
|  | ||||
| 	var libPath string | ||||
| 	switch runtime.GOOS { | ||||
| 	case "windows": | ||||
| 		libPath = filepath.Join(filepath.Dir(exe), "lib", "ollama") | ||||
| 	case "linux": | ||||
| 		libPath = filepath.Join(filepath.Dir(exe), "..", "lib", "ollama") | ||||
| 	case "darwin": | ||||
| 		libPath = filepath.Dir(exe) | ||||
| 	} | ||||
|  | ||||
| 	cwd, err := os.Getwd() | ||||
| 	if err != nil { | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	paths := []string{ | ||||
| 		libPath, | ||||
|  | ||||
| 		// build paths for development | ||||
| 		filepath.Join(filepath.Dir(exe), "build", "lib", "ollama"), | ||||
| 		filepath.Join(cwd, "build", "lib", "ollama"), | ||||
| 	} | ||||
|  | ||||
| 	for _, p := range paths { | ||||
| 		if _, err := os.Stat(p); err == nil { | ||||
| 			return p | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return filepath.Dir(exe) | ||||
| }() | ||||
| @@ -5,7 +5,6 @@ import ( | ||||
| 	"log/slog" | ||||
|  | ||||
| 	"github.com/ollama/ollama/format" | ||||
| 	"github.com/ollama/ollama/runners" | ||||
| ) | ||||
|  | ||||
| type memInfo struct { | ||||
| @@ -107,7 +106,7 @@ func (l GpuInfoList) ByLibrary() []GpuInfoList { | ||||
| 	for _, info := range l { | ||||
| 		found := false | ||||
| 		requested := info.Library | ||||
| 		if info.Variant != runners.CPUCapabilityNone.String() { | ||||
| 		if info.Variant != "" { | ||||
| 			requested += "_" + info.Variant | ||||
| 		} | ||||
| 		for i, lib := range libs { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| ### Getting Started | ||||
| * [Quickstart](../README.md#quickstart) | ||||
| * [Examples](../examples) | ||||
| * [Examples](./examples.md) | ||||
| * [Importing models](./import.md) | ||||
| * [Linux Documentation](./linux.md) | ||||
| * [Windows Documentation](./windows.md) | ||||
|   | ||||
							
								
								
									
										57
									
								
								docs/api.md
									
									
									
									
									
								
							
							
						
						
									
										57
									
								
								docs/api.md
									
									
									
									
									
								
							| @@ -31,7 +31,7 @@ Certain endpoints stream responses as JSON objects. Streaming can be disabled by | ||||
|  | ||||
| ## Generate a completion | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/generate | ||||
| ``` | ||||
|  | ||||
| @@ -306,7 +306,7 @@ curl http://localhost:11434/api/generate -d '{ | ||||
|  | ||||
| #### Response | ||||
|  | ||||
| ``` | ||||
| ```json | ||||
| { | ||||
|   "model": "llava", | ||||
|   "created_at": "2023-11-03T15:36:02.583064Z", | ||||
| @@ -485,7 +485,7 @@ A single JSON object is returned: | ||||
|  | ||||
| ## Generate a chat completion | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/chat | ||||
| ``` | ||||
|  | ||||
| @@ -495,14 +495,14 @@ Generate the next message in a chat with a provided model. This is a streaming e | ||||
|  | ||||
| - `model`: (required) the [model name](#model-names) | ||||
| - `messages`: the messages of the chat, this can be used to keep a chat memory | ||||
| - `tools`: tools for the model to use if supported. Requires `stream` to be set to `false` | ||||
| - `tools`: list of tools in JSON for the model to use if supported | ||||
|  | ||||
| The `message` object has the following fields: | ||||
|  | ||||
| - `role`: the role of the message, either `system`, `user`, `assistant`, or `tool` | ||||
| - `content`: the content of the message | ||||
| - `images` (optional): a list of images to include in the message (for multimodal models such as `llava`) | ||||
| - `tool_calls` (optional): a list of tools the model wants to use | ||||
| - `tool_calls` (optional): a list of tools in JSON that the model wants to use | ||||
|  | ||||
| Advanced parameters (optional): | ||||
|  | ||||
| @@ -558,6 +558,10 @@ Final response: | ||||
| { | ||||
|   "model": "llama3.2", | ||||
|   "created_at": "2023-08-04T19:22:45.499127Z", | ||||
|   "message": { | ||||
|     "role": "assistant", | ||||
|     "content": "" | ||||
|   }, | ||||
|   "done": true, | ||||
|   "total_duration": 4883583458, | ||||
|   "load_duration": 1334875, | ||||
| @@ -795,7 +799,7 @@ curl http://localhost:11434/api/chat -d '{ | ||||
|  | ||||
| ##### Request | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl http://localhost:11434/api/chat -d '{ | ||||
|   "model": "llama3.2", | ||||
|   "messages": [ | ||||
| @@ -870,7 +874,7 @@ If the messages array is empty, the model will be loaded into memory. | ||||
|  | ||||
| ##### Request | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl http://localhost:11434/api/chat -d '{ | ||||
|   "model": "llama3.2", | ||||
|   "messages": [] | ||||
| @@ -878,6 +882,7 @@ curl http://localhost:11434/api/chat -d '{ | ||||
| ``` | ||||
|  | ||||
| ##### Response | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "model": "llama3.2", | ||||
| @@ -897,7 +902,7 @@ If the messages array is empty and the `keep_alive` parameter is set to `0`, a m | ||||
|  | ||||
| ##### Request | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| curl http://localhost:11434/api/chat -d '{ | ||||
|   "model": "llama3.2", | ||||
|   "messages": [], | ||||
| @@ -924,7 +929,7 @@ A single JSON object is returned: | ||||
|  | ||||
| ## Create a Model | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/create | ||||
| ``` | ||||
|  | ||||
| @@ -1020,7 +1025,7 @@ curl http://localhost:11434/api/create -d '{ | ||||
|  | ||||
| A stream of JSON objects is returned: | ||||
|  | ||||
| ``` | ||||
| ```json | ||||
| {"status":"quantizing F16 model to Q4_K_M"} | ||||
| {"status":"creating new layer sha256:667b0c1932bc6ffc593ed1d03f895bf2dc8dc6df21db3042284a6f4416b06a29"} | ||||
| {"status":"using existing layer sha256:11ce4ee3e170f6adebac9a991c22e22ab3f8530e154ee669954c4bc73061c258"} | ||||
| @@ -1051,7 +1056,7 @@ curl http://localhost:11434/api/create -d '{ | ||||
|  | ||||
| A stream of JSON objects is returned: | ||||
|  | ||||
| ``` | ||||
| ```json | ||||
| {"status":"parsing GGUF"} | ||||
| {"status":"using existing layer sha256:432f310a77f4650a88d0fd59ecdd7cebed8d684bafea53cbff0473542964f0c3"} | ||||
| {"status":"writing manifest"} | ||||
| @@ -1118,7 +1123,7 @@ Return 200 OK if the blob exists, 404 Not Found if it does not. | ||||
|  | ||||
| ## Push a Blob | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/blobs/:digest | ||||
| ``` | ||||
|  | ||||
| @@ -1142,7 +1147,7 @@ Return 201 Created if the blob was successfully created, 400 Bad Request if the | ||||
|  | ||||
| ## List Local Models | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| GET /api/tags | ||||
| ``` | ||||
|  | ||||
| @@ -1195,7 +1200,7 @@ A single JSON object will be returned. | ||||
|  | ||||
| ## Show Model Information | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/show | ||||
| ``` | ||||
|  | ||||
| @@ -1212,7 +1217,7 @@ Show information about a model including details, modelfile, template, parameter | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/show -d '{ | ||||
|   "model": "llama3.2" | ||||
|   "model": "llava" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| @@ -1255,13 +1260,17 @@ curl http://localhost:11434/api/show -d '{ | ||||
|     "tokenizer.ggml.pre": "llama-bpe", | ||||
|     "tokenizer.ggml.token_type": [],        // populates if `verbose=true` | ||||
|     "tokenizer.ggml.tokens": []             // populates if `verbose=true` | ||||
|   } | ||||
|   }, | ||||
|   "capabilities": [ | ||||
|     "completion", | ||||
|     "vision" | ||||
|   ], | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Copy a Model | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/copy | ||||
| ``` | ||||
|  | ||||
| @@ -1284,7 +1293,7 @@ Returns a 200 OK if successful, or a 404 Not Found if the source model doesn't e | ||||
|  | ||||
| ## Delete a Model | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| DELETE /api/delete | ||||
| ``` | ||||
|  | ||||
| @@ -1310,7 +1319,7 @@ Returns a 200 OK if successful, 404 Not Found if the model to be deleted doesn't | ||||
|  | ||||
| ## Pull a Model | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/pull | ||||
| ``` | ||||
|  | ||||
| @@ -1382,7 +1391,7 @@ if `stream` is set to false, then the response is a single JSON object: | ||||
|  | ||||
| ## Push a Model | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/push | ||||
| ``` | ||||
|  | ||||
| @@ -1447,7 +1456,7 @@ If `stream` is set to `false`, then the response is a single JSON object: | ||||
|  | ||||
| ## Generate Embeddings | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/embed | ||||
| ``` | ||||
|  | ||||
| @@ -1515,7 +1524,7 @@ curl http://localhost:11434/api/embed -d '{ | ||||
| ``` | ||||
|  | ||||
| ## List Running Models | ||||
| ```shell | ||||
| ``` | ||||
| GET /api/ps | ||||
| ``` | ||||
|  | ||||
| @@ -1562,7 +1571,7 @@ A single JSON object will be returned. | ||||
|  | ||||
| > Note: this endpoint has been superseded by `/api/embed` | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| POST /api/embeddings | ||||
| ``` | ||||
|  | ||||
| @@ -1602,7 +1611,7 @@ curl http://localhost:11434/api/embeddings -d '{ | ||||
|  | ||||
| ## Version | ||||
|  | ||||
| ```shell | ||||
| ``` | ||||
| GET /api/version | ||||
| ``` | ||||
|  | ||||
|   | ||||
							
								
								
									
										59
									
								
								docs/benchmark.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								docs/benchmark.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| # Benchmark | ||||
|  | ||||
| Go benchmark tests that measure end-to-end performance of a running Ollama server. Run these tests to evaluate model inference performance on your hardware and measure the impact of code changes. | ||||
|  | ||||
| ## When to use | ||||
|  | ||||
| Run these benchmarks when: | ||||
| - Making changes to the model inference engine | ||||
| - Modifying model loading/unloading logic | ||||
| - Changing prompt processing or token generation code | ||||
| - Implementing a new model architecture | ||||
| - Testing performance across different hardware setups | ||||
|  | ||||
| ## Prerequisites | ||||
| - Ollama server running locally with `ollama serve` on `127.0.0.1:11434` | ||||
| ## Usage and Examples | ||||
|  | ||||
| >[!NOTE] | ||||
| >All commands must be run from the root directory of the Ollama project. | ||||
|  | ||||
| Basic syntax: | ||||
| ```bash | ||||
| go test -bench=. ./benchmark/... -m $MODEL_NAME | ||||
| ``` | ||||
|  | ||||
| Required flags: | ||||
| - `-bench=.`: Run all benchmarks | ||||
| - `-m`: Model name to benchmark | ||||
|  | ||||
| Optional flags: | ||||
| - `-count N`: Number of times to run the benchmark (useful for statistical analysis) | ||||
| - `-timeout T`: Maximum time for the benchmark to run (e.g. "10m" for 10 minutes) | ||||
|  | ||||
| Common usage patterns: | ||||
|  | ||||
| Single benchmark run with a model specified: | ||||
| ```bash | ||||
| go test -bench=. ./benchmark/... -m llama3.3 | ||||
| ``` | ||||
|  | ||||
| ## Output metrics | ||||
|  | ||||
| The benchmark reports several key metrics: | ||||
|  | ||||
| - `gen_tok/s`: Generated tokens per second | ||||
| - `prompt_tok/s`: Prompt processing tokens per second | ||||
| - `ttft_ms`: Time to first token in milliseconds | ||||
| - `load_ms`: Model load time in milliseconds | ||||
| - `gen_tokens`: Total tokens generated | ||||
| - `prompt_tokens`: Total prompt tokens processed | ||||
|  | ||||
| Each benchmark runs two scenarios: | ||||
| - Cold start: Model is loaded from disk for each test | ||||
| - Warm start: Model is pre-loaded in memory | ||||
|  | ||||
| Three prompt lengths are tested for each scenario: | ||||
| - Short prompt (100 tokens) | ||||
| - Medium prompt (500 tokens) | ||||
| - Long prompt (1000 tokens) | ||||
| @@ -1,165 +1,159 @@ | ||||
| # Development | ||||
|  | ||||
| Install required tools: | ||||
| Install prerequisites: | ||||
|  | ||||
| - go version 1.22 or higher | ||||
| - OS specific C/C++ compiler (see below) | ||||
| - GNU Make | ||||
| - [Go](https://go.dev/doc/install) | ||||
| - C/C++ Compiler e.g. Clang on macOS, [TDM-GCC](https://github.com/jmeubank/tdm-gcc/releases/latest) (Windows amd64) or [llvm-mingw](https://github.com/mstorsjo/llvm-mingw) (Windows arm64), GCC/Clang on Linux. | ||||
|  | ||||
| Then build and run Ollama from the root directory of the repository: | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| Ollama uses a mix of Go and C/C++ code to interface with GPUs.  The C/C++ code is compiled with both CGO and GPU library specific compilers.  A set of GNU Makefiles are used to compile the project.  GPU Libraries are auto-detected based on the typical environment variables used by the respective libraries, but can be overridden if necessary.  The default make target will build the runners and primary Go Ollama application that will run within the repo directory.  Throughout the examples below `-j 5` is suggested for 5 parallel jobs to speed up the build.  You can adjust the job count based on your CPU Core count to reduce build times.  If you want to relocate the built binaries, use the `dist` target and recursively copy the files in `./dist/$OS-$ARCH/` to your desired location. To learn more about the other make targets use `make help` | ||||
|  | ||||
| Once you have built the GPU/CPU runners, you can compile the main application with `go build .`  | ||||
|  | ||||
| ### MacOS | ||||
|  | ||||
| [Download Go](https://go.dev/dl/) | ||||
|  | ||||
| ```bash | ||||
| make -j 5 | ||||
| ```shell | ||||
| go run . serve | ||||
| ``` | ||||
|  | ||||
| Now you can run `ollama`: | ||||
| ## macOS (Apple Silicon) | ||||
|  | ||||
| ```bash | ||||
| ./ollama | ||||
| macOS Apple Silicon supports Metal which is built-in to the Ollama binary. No additional steps are required. | ||||
|  | ||||
| ## macOS (Intel) | ||||
|  | ||||
| Install prerequisites: | ||||
|  | ||||
| - [CMake](https://cmake.org/download/) or `brew install cmake` | ||||
|  | ||||
| Then, configure and build the project: | ||||
|  | ||||
| ```shell | ||||
| cmake -B build | ||||
| cmake --build build | ||||
| ``` | ||||
|  | ||||
| #### Xcode 15 warnings | ||||
| Lastly, run Ollama: | ||||
|  | ||||
| If you are using Xcode newer than version 14, you may see a warning during `go build` about `ld: warning: ignoring duplicate libraries: '-lobjc'` due to Golang issue https://github.com/golang/go/issues/67799 which can be safely ignored.  You can suppress the warning with `export CGO_LDFLAGS="-Wl,-no_warn_duplicate_libraries"` | ||||
|  | ||||
| ### Linux | ||||
|  | ||||
| #### Linux CUDA (NVIDIA) | ||||
|  | ||||
| _Your operating system distribution may already have packages for NVIDIA CUDA. Distro packages are often preferable, but instructions are distro-specific. Please consult distro-specific docs for dependencies if available!_ | ||||
|  | ||||
| Install `make`, `gcc` and `golang` as well as [NVIDIA CUDA](https://developer.nvidia.com/cuda-downloads) | ||||
| development and runtime packages. | ||||
|  | ||||
| Typically the makefile will auto-detect CUDA, however, if your Linux distro | ||||
| or installation approach uses alternative paths, you can specify the location by | ||||
| overriding `CUDA_PATH` to the location of the CUDA toolkit. You can customize | ||||
| a set of target CUDA architectures by setting `CUDA_ARCHITECTURES` (e.g. `CUDA_ARCHITECTURES=50;60;70`) | ||||
|  | ||||
| ``` | ||||
| make -j 5 | ||||
| ```shell | ||||
| go run . serve | ||||
| ``` | ||||
|  | ||||
| If both v11 and v12 tookkits are detected, runners for both major versions will be built by default.  You can build just v12 with `make cuda_v12` | ||||
| ## Windows | ||||
|  | ||||
| #### Older Linux CUDA (NVIDIA) | ||||
| Install prerequisites: | ||||
|  | ||||
| To support older GPUs with Compute Capability 3.5 or 3.7, you will need to use an older version of the Driver from [Unix Driver Archive](https://www.nvidia.com/en-us/drivers/unix/) (tested with 470) and [CUDA Toolkit Archive](https://developer.nvidia.com/cuda-toolkit-archive) (tested with cuda V11).  When you build Ollama, you will need to set two make variable to adjust the minimum compute capability Ollama supports via `make -j 5 CUDA_ARCHITECTURES="35;37;50;52" EXTRA_GOLDFLAGS="\"-X=github.com/ollama/ollama/discover.CudaComputeMajorMin=3\" \"-X=github.com/ollama/ollama/discover.CudaComputeMinorMin=5\""`.  To find the Compute Capability of your older GPU, refer to [GPU Compute Capability](https://developer.nvidia.com/cuda-gpus). | ||||
| - [CMake](https://cmake.org/download/) | ||||
| - [Visual Studio 2022](https://visualstudio.microsoft.com/downloads/) including the Native Desktop Workload | ||||
| - (Optional) AMD GPU support | ||||
|     - [ROCm](https://rocm.docs.amd.com/en/latest/) | ||||
|     - [Ninja](https://github.com/ninja-build/ninja/releases) | ||||
| - (Optional) NVIDIA GPU support | ||||
|     - [CUDA SDK](https://developer.nvidia.com/cuda-downloads?target_os=Windows&target_arch=x86_64&target_version=11&target_type=exe_network) | ||||
|  | ||||
| #### Linux ROCm (AMD) | ||||
| Then, configure and build the project: | ||||
|  | ||||
| _Your operating system distribution may already have packages for AMD ROCm. Distro packages are often preferable, but instructions are distro-specific. Please consult distro-specific docs for dependencies if available!_ | ||||
|  | ||||
| Install [ROCm](https://rocm.docs.amd.com/en/latest/) development packages first, as well as `make`, `gcc`, and `golang`. | ||||
|  | ||||
| Typically the build scripts will auto-detect ROCm, however, if your Linux distro | ||||
| or installation approach uses unusual paths, you can specify the location by | ||||
| specifying an environment variable `HIP_PATH` to the location of the ROCm | ||||
| install (typically `/opt/rocm`). You can also customize | ||||
| the AMD GPU targets by setting HIP_ARCHS (e.g. `HIP_ARCHS=gfx1101;gfx1102`) | ||||
|  | ||||
| ``` | ||||
| make -j 5 | ||||
| ```shell | ||||
| cmake -B build | ||||
| cmake --build build --config Release | ||||
| ``` | ||||
|  | ||||
| ROCm requires elevated privileges to access the GPU at runtime. On most distros you can add your user account to the `render` group, or run as root. | ||||
| > [!IMPORTANT] | ||||
| > Building for ROCm requires additional flags: | ||||
| > ``` | ||||
| > cmake -B build -G Ninja -DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++ | ||||
| > cmake --build build --config Release | ||||
| > ``` | ||||
|  | ||||
| #### Containerized Linux Build | ||||
|  | ||||
| If you have Docker and buildx available, you can build linux binaries with `./scripts/build_linux.sh` which has the CUDA and ROCm dependencies included. The resulting artifacts are placed in `./dist`  and by default the script builds both arm64 and amd64 binaries.  If you want to build only amd64, you can build with `PLATFORM=linux/amd64 ./scripts/build_linux.sh` | ||||
| Lastly, run Ollama: | ||||
|  | ||||
| ### Windows | ||||
|  | ||||
| The following tools are required as a minimal development environment to build CPU inference support. | ||||
|  | ||||
| - Go version 1.22 or higher | ||||
|   - https://go.dev/dl/ | ||||
| - Git | ||||
|   - https://git-scm.com/download/win | ||||
| - clang with gcc compat and Make.  There are multiple options on how to go about installing these tools on Windows.  We have verified the following, but others may work as well:   | ||||
|   - [MSYS2](https://www.msys2.org/) | ||||
|     - After installing, from an MSYS2 terminal, run `pacman -S mingw-w64-clang-x86_64-gcc-compat mingw-w64-clang-x86_64-clang make` to install the required tools | ||||
|   - Assuming you used the default install prefix for msys2 above, add `C:\msys64\clang64\bin` and `c:\msys64\usr\bin` to your environment variable `PATH` where you will perform the build steps below (e.g. system-wide, account-level, powershell, cmd, etc.) | ||||
|  | ||||
| > [!NOTE]   | ||||
| > Due to bugs in the GCC C++ library for unicode support, Ollama should be built with clang on windows. | ||||
|  | ||||
| ``` | ||||
| make -j 5 | ||||
| ```shell | ||||
| go run . serve | ||||
| ``` | ||||
|  | ||||
| #### GPU Support | ||||
| ## Windows (ARM) | ||||
|  | ||||
| The GPU tools require the Microsoft native build tools.  To build either CUDA or ROCm, you must first install MSVC via Visual Studio: | ||||
| Windows ARM does not support additional acceleration libraries at this time.  Do not use cmake, simply `go run` or `go build`. | ||||
|  | ||||
| - Make sure to select `Desktop development with C++` as a Workload during the Visual Studio install | ||||
| - You must complete the Visual Studio install and run it once **BEFORE** installing CUDA or ROCm for the tools to properly register | ||||
| - Add the location of the **64 bit (x64)** compiler (`cl.exe`) to your `PATH` | ||||
| - Note: the default Developer Shell may configure the 32 bit (x86) compiler which will lead to build failures.  Ollama requires a 64 bit toolchain. | ||||
| ## Linux | ||||
|  | ||||
| #### Windows CUDA (NVIDIA) | ||||
| Install prerequisites: | ||||
|  | ||||
| In addition to the common Windows development tools and MSVC described above: | ||||
| - [CMake](https://cmake.org/download/) or `sudo apt install cmake` or `sudo dnf install cmake` | ||||
| - (Optional) AMD GPU support | ||||
|     - [ROCm](https://rocm.docs.amd.com/projects/install-on-linux/en/latest/install/quick-start.html) | ||||
| - (Optional) NVIDIA GPU support | ||||
|     - [CUDA SDK](https://developer.nvidia.com/cuda-downloads) | ||||
|  | ||||
| - [NVIDIA CUDA](https://docs.nvidia.com/cuda/cuda-installation-guide-microsoft-windows/index.html) | ||||
| > [!IMPORTANT] | ||||
| > Ensure prerequisites are in `PATH` before running CMake. | ||||
|  | ||||
| #### Windows ROCm (AMD Radeon) | ||||
|  | ||||
| In addition to the common Windows development tools and MSVC described above: | ||||
| Then, configure and build the project: | ||||
|  | ||||
| - [AMD HIP](https://www.amd.com/en/developer/resources/rocm-hub/hip-sdk.html) | ||||
|  | ||||
| #### Windows arm64 | ||||
|  | ||||
| The default `Developer PowerShell for VS 2022` may default to x86 which is not what you want.  To ensure you get an arm64 development environment, start a plain PowerShell terminal and run: | ||||
|  | ||||
| ```powershell | ||||
| import-module 'C:\\Program Files\\Microsoft Visual Studio\\2022\\Community\\Common7\\Tools\\Microsoft.VisualStudio.DevShell.dll' | ||||
| Enter-VsDevShell -Arch arm64 -vsinstallpath 'C:\\Program Files\\Microsoft Visual Studio\\2022\\Community' -skipautomaticlocation | ||||
| ```shell | ||||
| cmake -B build | ||||
| cmake --build build | ||||
| ``` | ||||
|  | ||||
| You can confirm with `write-host $env:VSCMD_ARG_TGT_ARCH` | ||||
| Lastly, run Ollama: | ||||
|  | ||||
| Follow the instructions at https://www.msys2.org/wiki/arm64/ to set up an arm64 msys2 environment.  Ollama requires gcc and mingw32-make to compile, which is not currently available on Windows arm64, but a gcc compatibility adapter is available via `mingw-w64-clang-aarch64-gcc-compat`. At a minimum you will need to install the following: | ||||
|  | ||||
| ``` | ||||
| pacman -S mingw-w64-clang-aarch64-clang mingw-w64-clang-aarch64-gcc-compat mingw-w64-clang-aarch64-make make | ||||
| ```shell | ||||
| go run . serve | ||||
| ``` | ||||
|  | ||||
| You will need to ensure your PATH includes go, cmake, gcc and clang mingw32-make to build ollama from source. (typically `C:\msys64\clangarm64\bin\`) | ||||
| ## Docker | ||||
|  | ||||
|  | ||||
| ## Advanced CPU Vector Settings | ||||
|  | ||||
| On x86, running `make` will compile several CPU runners which can run on different CPU families. At runtime, Ollama will auto-detect the best variation to load.  If GPU libraries are present at build time, Ollama also compiles GPU runners with the `AVX` CPU vector feature enabled.  This provides a good performance balance when loading large models that split across GPU and CPU with broad compatibility.  Some users may prefer no vector extensions (e.g. older Xeon/Celeron processors, or hypervisors that mask the vector features) while other users may prefer turning on many more vector extensions to further improve performance for split model loads. | ||||
|  | ||||
| To customize the set of CPU vector features enabled for a CPU runner and all GPU runners, use CUSTOM_CPU_FLAGS during the build. | ||||
|  | ||||
| To build without any vector flags: | ||||
|  | ||||
| ``` | ||||
| make CUSTOM_CPU_FLAGS="" | ||||
| ```shell | ||||
| docker build . | ||||
| ``` | ||||
|  | ||||
| To build with both AVX and AVX2: | ||||
| ``` | ||||
| make CUSTOM_CPU_FLAGS=avx,avx2 | ||||
| ### ROCm | ||||
|  | ||||
| ```shell | ||||
| docker build --build-arg FLAVOR=rocm . | ||||
| ``` | ||||
|  | ||||
| To build with AVX512 features turned on: | ||||
| ## Running tests | ||||
|  | ||||
| ``` | ||||
| make CUSTOM_CPU_FLAGS=avx,avx2,avx512,avx512vbmi,avx512vnni,avx512bf16 | ||||
| To run tests, use `go test`: | ||||
|  | ||||
| ```shell | ||||
| go test ./... | ||||
| ``` | ||||
|  | ||||
| > [!NOTE]   | ||||
| > If you are experimenting with different flags, make sure to do a `make clean` between each change to ensure everything is rebuilt with the new compiler flags | ||||
| > NOTE: In rare cirumstances, you may nedd to change a package using the new | ||||
| > "synctest" package in go1.24. | ||||
| > | ||||
| > If you do not have the "synctest" package enabled, you will not see build or | ||||
| > test failures resulting from your change(s), if any, locally, but CI will | ||||
| > break. | ||||
| > | ||||
| > If you see failures in CI, you can either keep pushing changes to see if the | ||||
| > CI build passes, or you can enable the "synctest" package locally to see the | ||||
| > failures before pushing. | ||||
| > | ||||
| > To enable the "synctest" package for testing, run the following command: | ||||
| > | ||||
| > ```shell | ||||
| > GOEXPERIMENT=synctest go test ./... | ||||
| > ``` | ||||
| > | ||||
| > If you wish to enable synctest for all go commands, you can set the | ||||
| > `GOEXPERIMENT` environment variable in your shell profile or by using: | ||||
| > | ||||
| > ```shell | ||||
| > go env -w GOEXPERIMENT=synctest | ||||
| > ``` | ||||
| > | ||||
| > Which will enable the "synctest" package for all go commands without needing | ||||
| > to set it for all shell sessions. | ||||
| > | ||||
| > The synctest package is not required for production builds. | ||||
|  | ||||
| ## Library detection | ||||
|  | ||||
| Ollama looks for acceleration libraries in the following paths relative to the `ollama` executable: | ||||
|  | ||||
| * `./lib/ollama` (Windows) | ||||
| * `../lib/ollama` (Linux) | ||||
| * `.` (macOS) | ||||
| * `build/lib/ollama` (for development) | ||||
|  | ||||
| If the libraries are not found, Ollama will not run with any acceleration libraries. | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| ### CPU only | ||||
|  | ||||
| ```bash | ||||
| ```shell | ||||
| docker run -d -v ollama:/root/.ollama -p 11434:11434 --name ollama ollama/ollama | ||||
| ``` | ||||
|  | ||||
| @@ -11,42 +11,46 @@ Install the [NVIDIA Container Toolkit](https://docs.nvidia.com/datacenter/cloud- | ||||
|  | ||||
| #### Install with Apt | ||||
| 1.  Configure the repository | ||||
| ```bash | ||||
| curl -fsSL https://nvidia.github.io/libnvidia-container/gpgkey \ | ||||
|     | sudo gpg --dearmor -o /usr/share/keyrings/nvidia-container-toolkit-keyring.gpg | ||||
| curl -s -L https://nvidia.github.io/libnvidia-container/stable/deb/nvidia-container-toolkit.list \ | ||||
|     | sed 's#deb https://#deb [signed-by=/usr/share/keyrings/nvidia-container-toolkit-keyring.gpg] https://#g' \ | ||||
|     | sudo tee /etc/apt/sources.list.d/nvidia-container-toolkit.list | ||||
| sudo apt-get update | ||||
| ``` | ||||
|  | ||||
|     ```shell | ||||
|     curl -fsSL https://nvidia.github.io/libnvidia-container/gpgkey \ | ||||
|         | sudo gpg --dearmor -o /usr/share/keyrings/nvidia-container-toolkit-keyring.gpg | ||||
|     curl -s -L https://nvidia.github.io/libnvidia-container/stable/deb/nvidia-container-toolkit.list \ | ||||
|         | sed 's#deb https://#deb [signed-by=/usr/share/keyrings/nvidia-container-toolkit-keyring.gpg] https://#g' \ | ||||
|         | sudo tee /etc/apt/sources.list.d/nvidia-container-toolkit.list | ||||
|     sudo apt-get update | ||||
|     ``` | ||||
|  | ||||
| 2.  Install the NVIDIA Container Toolkit packages | ||||
| ```bash | ||||
| sudo apt-get install -y nvidia-container-toolkit | ||||
| ``` | ||||
|  | ||||
|     ```shell | ||||
|     sudo apt-get install -y nvidia-container-toolkit | ||||
|     ``` | ||||
|  | ||||
| #### Install with Yum or Dnf | ||||
| 1.  Configure the repository | ||||
|  | ||||
| ```bash | ||||
| curl -s -L https://nvidia.github.io/libnvidia-container/stable/rpm/nvidia-container-toolkit.repo \ | ||||
|     | sudo tee /etc/yum.repos.d/nvidia-container-toolkit.repo | ||||
| ``` | ||||
|     ```shell | ||||
|     curl -s -L https://nvidia.github.io/libnvidia-container/stable/rpm/nvidia-container-toolkit.repo \ | ||||
|         | sudo tee /etc/yum.repos.d/nvidia-container-toolkit.repo | ||||
|     ``` | ||||
|  | ||||
| 2. Install the NVIDIA Container Toolkit packages | ||||
|  | ||||
| ```bash | ||||
| sudo yum install -y nvidia-container-toolkit | ||||
| ``` | ||||
|     ```shell | ||||
|     sudo yum install -y nvidia-container-toolkit | ||||
|     ``` | ||||
|  | ||||
| #### Configure Docker to use Nvidia driver | ||||
| ``` | ||||
|  | ||||
| ```shell | ||||
| sudo nvidia-ctk runtime configure --runtime=docker | ||||
| sudo systemctl restart docker | ||||
| ``` | ||||
|  | ||||
| #### Start the container | ||||
|  | ||||
| ```bash | ||||
| ```shell | ||||
| docker run -d --gpus=all -v ollama:/root/.ollama -p 11434:11434 --name ollama ollama/ollama | ||||
| ``` | ||||
|  | ||||
| @@ -57,7 +61,7 @@ docker run -d --gpus=all -v ollama:/root/.ollama -p 11434:11434 --name ollama ol | ||||
|  | ||||
| To run Ollama using Docker with AMD GPUs, use the `rocm` tag and the following command: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| docker run -d --device /dev/kfd --device /dev/dri -v ollama:/root/.ollama -p 11434:11434 --name ollama ollama/ollama:rocm | ||||
| ``` | ||||
|  | ||||
| @@ -65,7 +69,7 @@ docker run -d --device /dev/kfd --device /dev/dri -v ollama:/root/.ollama -p 114 | ||||
|  | ||||
| Now you can run a model: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| docker exec -it ollama ollama run llama3.2 | ||||
| ``` | ||||
|  | ||||
|   | ||||
							
								
								
									
										37
									
								
								docs/faq.md
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								docs/faq.md
									
									
									
									
									
								
							| @@ -20,11 +20,17 @@ Please refer to the [GPU docs](./gpu.md). | ||||
|  | ||||
| ## How can I specify the context window size? | ||||
|  | ||||
| By default, Ollama uses a context window size of 2048 tokens. | ||||
| By default, Ollama uses a context window size of 2048 tokens.  | ||||
|  | ||||
| This can be overridden with the `OLLAMA_CONTEXT_LENGTH` environment variable. For example, to set the default context window to 8K, use:  | ||||
|  | ||||
| ```shell | ||||
| OLLAMA_CONTEXT_LENGTH=8192 ollama serve | ||||
| ``` | ||||
|  | ||||
| To change this when using `ollama run`, use `/set parameter`: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| /set parameter num_ctx 4096 | ||||
| ``` | ||||
|  | ||||
| @@ -46,10 +52,15 @@ Use the `ollama ps` command to see what models are currently loaded into memory. | ||||
|  | ||||
| ```shell | ||||
| ollama ps | ||||
| NAME      	ID          	SIZE 	PROCESSOR	UNTIL | ||||
| llama3:70b	bcfb190ca3a7	42 GB	100% GPU 	4 minutes from now | ||||
| ``` | ||||
|  | ||||
| > **Output**: | ||||
| > | ||||
| > ``` | ||||
| > NAME      	ID          	SIZE 	PROCESSOR	UNTIL | ||||
| > llama3:70b	bcfb190ca3a7	42 GB	100% GPU 	4 minutes from now | ||||
| > ``` | ||||
|  | ||||
| The `Processor` column will show which memory the model was loaded in to: | ||||
| * `100% GPU` means the model was loaded entirely into the GPU | ||||
| * `100% CPU` means the model was loaded entirely in system memory | ||||
| @@ -66,7 +77,7 @@ If Ollama is run as a macOS application, environment variables should be set usi | ||||
| 1. For each environment variable, call `launchctl setenv`. | ||||
|  | ||||
|     ```bash | ||||
|     launchctl setenv OLLAMA_HOST "0.0.0.0" | ||||
|     launchctl setenv OLLAMA_HOST "0.0.0.0:11434" | ||||
|     ``` | ||||
|  | ||||
| 2. Restart Ollama application. | ||||
| @@ -81,14 +92,14 @@ If Ollama is run as a systemd service, environment variables should be set using | ||||
|  | ||||
|     ```ini | ||||
|     [Service] | ||||
|     Environment="OLLAMA_HOST=0.0.0.0" | ||||
|     Environment="OLLAMA_HOST=0.0.0.0:11434" | ||||
|     ``` | ||||
|  | ||||
| 3. Save and exit. | ||||
|  | ||||
| 4. Reload `systemd` and restart Ollama: | ||||
|  | ||||
|    ```bash | ||||
|    ```shell | ||||
|    systemctl daemon-reload | ||||
|    systemctl restart ollama | ||||
|    ``` | ||||
| @@ -182,6 +193,13 @@ cloudflared tunnel --url http://localhost:11434 --http-host-header="localhost:11 | ||||
|  | ||||
| Ollama allows cross-origin requests from `127.0.0.1` and `0.0.0.0` by default. Additional origins can be configured with `OLLAMA_ORIGINS`. | ||||
|  | ||||
| For browser extensions, you'll need to explicitly allow the extension's origin pattern. Set `OLLAMA_ORIGINS` to include `chrome-extension://*`, `moz-extension://*`, and `safari-web-extension://*` if you wish to allow all browser extensions access, or specific extensions as needed: | ||||
|  | ||||
| ``` | ||||
| # Allow all Chrome, Firefox, and Safari extensions | ||||
| OLLAMA_ORIGINS=chrome-extension://*,moz-extension://*,safari-web-extension://* ollama serve | ||||
| ``` | ||||
|  | ||||
| Refer to the section [above](#how-do-i-configure-ollama-server) for how to set environment variables on your platform. | ||||
|  | ||||
| ## Where are models stored? | ||||
| @@ -221,16 +239,19 @@ properties. | ||||
| If you are using the API you can preload a model by sending the Ollama server an empty request. This works with both the `/api/generate` and `/api/chat` API endpoints. | ||||
|  | ||||
| To preload the mistral model using the generate endpoint, use: | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/generate -d '{"model": "mistral"}' | ||||
| ``` | ||||
|  | ||||
| To use the chat completions endpoint, use: | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/chat -d '{"model": "mistral"}' | ||||
| ``` | ||||
|  | ||||
| To preload a model using the CLI, use the command: | ||||
|  | ||||
| ```shell | ||||
| ollama run llama3.2 "" | ||||
| ``` | ||||
| @@ -250,11 +271,13 @@ If you're using the API, use the `keep_alive` parameter with the `/api/generate` | ||||
| * '0' which will unload the model immediately after generating a response | ||||
|  | ||||
| For example, to preload a model and leave it in memory use: | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/generate -d '{"model": "llama3.2", "keep_alive": -1}' | ||||
| ``` | ||||
|  | ||||
| To unload the model and free up memory use: | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/generate -d '{"model": "llama3.2", "keep_alive": 0}' | ||||
| ``` | ||||
|   | ||||
| @@ -7,7 +7,7 @@ Check your compute compatibility to see if your card is supported: | ||||
|  | ||||
| | Compute Capability | Family              | Cards                                                                                                       | | ||||
| | ------------------ | ------------------- | ----------------------------------------------------------------------------------------------------------- | | ||||
| | 9.0                | NVIDIA              | `H100`                                                                                                      | | ||||
| | 9.0                | NVIDIA              | `H200` `H100`                                                                                               | | ||||
| | 8.9                | GeForce RTX 40xx    | `RTX 4090` `RTX 4080 SUPER` `RTX 4080` `RTX 4070 Ti SUPER` `RTX 4070 Ti` `RTX 4070 SUPER` `RTX 4070` `RTX 4060 Ti` `RTX 4060`  | | ||||
| |                    | NVIDIA Professional | `L4` `L40` `RTX 6000`                                                                                       | | ||||
| | 8.6                | GeForce RTX 30xx    | `RTX 3090 Ti` `RTX 3090` `RTX 3080 Ti` `RTX 3080` `RTX 3070 Ti` `RTX 3070` `RTX 3060 Ti` `RTX 3060` `RTX 3050 Ti` `RTX 3050`   | | ||||
| @@ -38,7 +38,7 @@ Numeric IDs may be used, however ordering may vary, so UUIDs are more reliable. | ||||
| You can discover the UUID of your GPUs by running `nvidia-smi -L` If you want to | ||||
| ignore the GPUs and force CPU usage, use an invalid GPU ID (e.g., "-1") | ||||
|  | ||||
| ### Laptop Suspend Resume | ||||
| ### Linux Suspend Resume | ||||
|  | ||||
| On linux, after a suspend/resume cycle, sometimes Ollama will fail to discover | ||||
| your NVIDIA GPU, and fallback to running on the CPU.  You can workaround this | ||||
|   | ||||
| @@ -20,13 +20,13 @@ Make sure that you use the same base model in the `FROM` command as you used to | ||||
|  | ||||
| Now run `ollama create` from the directory where the `Modelfile` was created: | ||||
|  | ||||
| ```bash | ||||
| ```shell | ||||
| ollama create my-model | ||||
| ``` | ||||
|  | ||||
| Lastly, test the model: | ||||
|  | ||||
| ```bash | ||||
| ```shell | ||||
| ollama run my-model | ||||
| ``` | ||||
|  | ||||
|   | ||||
| @@ -75,7 +75,7 @@ RestartSec=3 | ||||
| Environment="PATH=$PATH" | ||||
|  | ||||
| [Install] | ||||
| WantedBy=default.target | ||||
| WantedBy=multi-user.target | ||||
| ``` | ||||
|  | ||||
| Then start the service: | ||||
| @@ -119,7 +119,7 @@ sudo systemctl status ollama | ||||
|  | ||||
| To customize the installation of Ollama, you can edit the systemd service file or the environment variables by running: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| sudo systemctl edit ollama | ||||
| ``` | ||||
|  | ||||
| @@ -152,7 +152,7 @@ Use `OLLAMA_VERSION` environment variable with the install script to install a s | ||||
| For example: | ||||
|  | ||||
| ```shell | ||||
| curl -fsSL https://ollama.com/install.sh | OLLAMA_VERSION=0.3.9 sh | ||||
| curl -fsSL https://ollama.com/install.sh | OLLAMA_VERSION=0.5.7 sh | ||||
| ``` | ||||
|  | ||||
| ## Viewing logs | ||||
| @@ -186,3 +186,9 @@ sudo rm -r /usr/share/ollama | ||||
| sudo userdel ollama | ||||
| sudo groupdel ollama | ||||
| ``` | ||||
|  | ||||
| Remove installed libraries: | ||||
|  | ||||
| ```shell | ||||
| sudo rm -rf /usr/local/lib/ollama | ||||
| ``` | ||||
|   | ||||
| @@ -28,7 +28,7 @@ A model file is the blueprint to create and share models with Ollama. | ||||
|  | ||||
| The format of the `Modelfile`: | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| # comment | ||||
| INSTRUCTION arguments | ||||
| ``` | ||||
| @@ -49,7 +49,7 @@ INSTRUCTION arguments | ||||
|  | ||||
| An example of a `Modelfile` creating a mario blueprint: | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM llama3.2 | ||||
| # sets the temperature to 1 [higher is more creative, lower is more coherent] | ||||
| PARAMETER temperature 1 | ||||
| @@ -67,28 +67,32 @@ To use this: | ||||
| 3. `ollama run choose-a-model-name` | ||||
| 4. Start using the model! | ||||
|  | ||||
| More examples are available in the [examples directory](../examples). | ||||
|  | ||||
| To view the Modelfile of a given model, use the `ollama show --modelfile` command. | ||||
|  | ||||
|   ```bash | ||||
|   > ollama show --modelfile llama3.2 | ||||
|   # Modelfile generated by "ollama show" | ||||
|   # To build a new Modelfile based on this one, replace the FROM line with: | ||||
|   # FROM llama3.2:latest | ||||
|   FROM /Users/pdevine/.ollama/models/blobs/sha256-00e1317cbf74d901080d7100f57580ba8dd8de57203072dc6f668324ba545f29 | ||||
|   TEMPLATE """{{ if .System }}<|start_header_id|>system<|end_header_id|> | ||||
| ```shell | ||||
| ollama show --modelfile llama3.2 | ||||
| ``` | ||||
|  | ||||
|   {{ .System }}<|eot_id|>{{ end }}{{ if .Prompt }}<|start_header_id|>user<|end_header_id|> | ||||
| > **Output**: | ||||
| > | ||||
| > ``` | ||||
| > # Modelfile generated by "ollama show" | ||||
| > # To build a new Modelfile based on this one, replace the FROM line with: | ||||
| > # FROM llama3.2:latest | ||||
| > FROM /Users/pdevine/.ollama/models/blobs/sha256-00e1317cbf74d901080d7100f57580ba8dd8de57203072dc6f668324ba545f29 | ||||
| > TEMPLATE """{{ if .System }}<|start_header_id|>system<|end_header_id|> | ||||
| > | ||||
| > {{ .System }}<|eot_id|>{{ end }}{{ if .Prompt }}<|start_header_id|>user<|end_header_id|> | ||||
| > | ||||
| > {{ .Prompt }}<|eot_id|>{{ end }}<|start_header_id|>assistant<|end_header_id|> | ||||
| > | ||||
| > {{ .Response }}<|eot_id|>""" | ||||
| > PARAMETER stop "<|start_header_id|>" | ||||
| > PARAMETER stop "<|end_header_id|>" | ||||
| > PARAMETER stop "<|eot_id|>" | ||||
| > PARAMETER stop "<|reserved_special_token" | ||||
| > ``` | ||||
|  | ||||
|   {{ .Prompt }}<|eot_id|>{{ end }}<|start_header_id|>assistant<|end_header_id|> | ||||
|  | ||||
|   {{ .Response }}<|eot_id|>""" | ||||
|   PARAMETER stop "<|start_header_id|>" | ||||
|   PARAMETER stop "<|end_header_id|>" | ||||
|   PARAMETER stop "<|eot_id|>" | ||||
|   PARAMETER stop "<|reserved_special_token" | ||||
|   ``` | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| @@ -96,13 +100,13 @@ To view the Modelfile of a given model, use the `ollama show --modelfile` comman | ||||
|  | ||||
| The `FROM` instruction defines the base model to use when creating a model. | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM <model name>:<tag> | ||||
| ``` | ||||
|  | ||||
| #### Build from existing model | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM llama3.2 | ||||
| ``` | ||||
|  | ||||
| @@ -113,7 +117,7 @@ Additional models can be found at: | ||||
|  | ||||
| #### Build from a Safetensors model | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM <model directory> | ||||
| ``` | ||||
|  | ||||
| @@ -127,7 +131,7 @@ Currently supported model architectures: | ||||
|  | ||||
| #### Build from a GGUF file | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM ./ollama-model.gguf | ||||
| ``` | ||||
|  | ||||
| @@ -138,7 +142,7 @@ The GGUF file location should be specified as an absolute path or relative to th | ||||
|  | ||||
| The `PARAMETER` instruction defines a parameter that can be set when the model is run. | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| PARAMETER <parameter> <parametervalue> | ||||
| ``` | ||||
|  | ||||
| @@ -155,7 +159,6 @@ PARAMETER <parameter> <parametervalue> | ||||
| | temperature    | The temperature of the model. Increasing the temperature will make the model answer more creatively. (Default: 0.8)                                                                                                                                     | float      | temperature 0.7      | | ||||
| | seed           | Sets the random number seed to use for generation. Setting this to a specific number will make the model generate the same text for the same prompt. (Default: 0)                                                                                       | int        | seed 42              | | ||||
| | stop           | Sets the stop sequences to use. When this pattern is encountered the LLM will stop generating text and return. Multiple stop patterns may be set by specifying multiple separate `stop` parameters in a modelfile.                                      | string     | stop "AI assistant:" | | ||||
| | tfs_z          | Tail free sampling is used to reduce the impact of less probable tokens from the output. A higher value (e.g., 2.0) will reduce the impact more, while a value of 1.0 disables this setting. (default: 1)                                               | float      | tfs_z 1              | | ||||
| | num_predict    | Maximum number of tokens to predict when generating text. (Default: -1, infinite generation)                                                                                                                                   | int        | num_predict 42       | | ||||
| | top_k          | Reduces the probability of generating nonsense. A higher value (e.g. 100) will give more diverse answers, while a lower value (e.g. 10) will be more conservative. (Default: 40)                                                                        | int        | top_k 40             | | ||||
| | top_p          | Works together with top-k. A higher value (e.g., 0.95) will lead to more diverse text, while a lower value (e.g., 0.5) will generate more focused and conservative text. (Default: 0.9)                                                                 | float      | top_p 0.9            | | ||||
| @@ -186,7 +189,7 @@ TEMPLATE """{{ if .System }}<|im_start|>system | ||||
|  | ||||
| The `SYSTEM` instruction specifies the system message to be used in the template, if applicable. | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| SYSTEM """<system message>""" | ||||
| ``` | ||||
|  | ||||
| @@ -196,7 +199,7 @@ The `ADAPTER` instruction specifies a fine tuned LoRA adapter that should apply | ||||
|  | ||||
| #### Safetensor adapter | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| ADAPTER <path to safetensor adapter> | ||||
| ``` | ||||
|  | ||||
| @@ -207,7 +210,7 @@ Currently supported Safetensor adapters: | ||||
|  | ||||
| #### GGUF adapter | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| ADAPTER ./ollama-lora.gguf | ||||
| ``` | ||||
|  | ||||
| @@ -215,7 +218,7 @@ ADAPTER ./ollama-lora.gguf | ||||
|  | ||||
| The `LICENSE` instruction allows you to specify the legal license under which the model used with this Modelfile is shared or distributed. | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| LICENSE """ | ||||
| <license text> | ||||
| """ | ||||
| @@ -225,7 +228,7 @@ LICENSE """ | ||||
|  | ||||
| The `MESSAGE` instruction allows you to specify a message history for the model to use when responding. Use multiple iterations of the MESSAGE command to build up a conversation which will guide the model to answer in a similar way. | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| MESSAGE <role> <message> | ||||
| ``` | ||||
|  | ||||
| @@ -240,7 +243,7 @@ MESSAGE <role> <message> | ||||
|  | ||||
| #### Example conversation | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| MESSAGE user Is Toronto in Canada? | ||||
| MESSAGE assistant yes | ||||
| MESSAGE user Is Sacramento in Canada? | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| # OpenAI compatibility | ||||
|  | ||||
| > **Note:** OpenAI compatibility is experimental and is subject to major adjustments including breaking changes. For fully-featured access to the Ollama API, see the Ollama [Python library](https://github.com/ollama/ollama-python), [JavaScript library](https://github.com/ollama/ollama-js) and [REST API](https://github.com/ollama/ollama/blob/main/docs/api.md). | ||||
| > [!NOTE] | ||||
| > OpenAI compatibility is experimental and is subject to major adjustments including breaking changes. For fully-featured access to the Ollama API, see the Ollama [Python library](https://github.com/ollama/ollama-python), [JavaScript library](https://github.com/ollama/ollama-js) and [REST API](https://github.com/ollama/ollama/blob/main/docs/api.md). | ||||
|  | ||||
| Ollama provides experimental compatibility with parts of the [OpenAI API](https://platform.openai.com/docs/api-reference) to help connect existing applications to Ollama. | ||||
|  | ||||
| @@ -59,8 +60,10 @@ embeddings = client.embeddings.create( | ||||
|     input=["why is the sky blue?", "why is the grass green?"], | ||||
| ) | ||||
| ``` | ||||
|  | ||||
| #### Structured outputs | ||||
| ```py | ||||
|  | ||||
| ```python | ||||
| from pydantic import BaseModel | ||||
| from openai import OpenAI | ||||
|  | ||||
| @@ -144,7 +147,7 @@ const embedding = await openai.embeddings.create({ | ||||
|  | ||||
| ### `curl` | ||||
|  | ||||
| ``` shell | ||||
| ```shell | ||||
| curl http://localhost:11434/v1/chat/completions \ | ||||
|     -H "Content-Type: application/json" \ | ||||
|     -d '{ | ||||
| @@ -319,7 +322,7 @@ ollama pull llama3.2 | ||||
|  | ||||
| For tooling that relies on default OpenAI model names such as `gpt-3.5-turbo`, use `ollama cp` to copy an existing model name to a temporary name: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| ollama cp llama3.2 gpt-3.5-turbo | ||||
| ``` | ||||
|  | ||||
| @@ -343,7 +346,7 @@ curl http://localhost:11434/v1/chat/completions \ | ||||
|  | ||||
| The OpenAI API does not have a way of setting the context size for a model. If you need to change the context size, create a `Modelfile` which looks like: | ||||
|  | ||||
| ```modelfile | ||||
| ``` | ||||
| FROM <some model> | ||||
| PARAMETER num_ctx <context size> | ||||
| ``` | ||||
|   | ||||
| @@ -9,7 +9,7 @@ cat ~/.ollama/logs/server.log | ||||
| On **Linux** systems with systemd, the logs can be found with this command: | ||||
|  | ||||
| ```shell | ||||
| journalctl -u ollama --no-pager | ||||
| journalctl -u ollama --no-pager --follow --pager-end  | ||||
| ``` | ||||
|  | ||||
| When you run Ollama in a **container**, the logs go to stdout/stderr in the container: | ||||
| @@ -17,6 +17,7 @@ When you run Ollama in a **container**, the logs go to stdout/stderr in the cont | ||||
| ```shell | ||||
| docker logs <container-name> | ||||
| ``` | ||||
|  | ||||
| (Use `docker ps` to find the container name) | ||||
|  | ||||
| If manually running `ollama serve` in a terminal, the logs will be on that terminal. | ||||
| @@ -25,9 +26,9 @@ When you run Ollama on **Windows**, there are a few different locations. You can | ||||
| - `explorer %LOCALAPPDATA%\Ollama` to view logs.  The most recent server logs will be in `server.log` and older logs will be in `server-#.log`  | ||||
| - `explorer %LOCALAPPDATA%\Programs\Ollama` to browse the binaries (The installer adds this to your user PATH) | ||||
| - `explorer %HOMEPATH%\.ollama` to browse where models and configuration is stored | ||||
| - `explorer %TEMP%` where temporary executable files are stored in one or more `ollama*` directories | ||||
|  | ||||
| To enable additional debug logging to help troubleshoot problems, first **Quit the running app from the tray menu** then in a powershell terminal | ||||
|  | ||||
| ```powershell | ||||
| $env:OLLAMA_DEBUG="1" | ||||
| & "ollama app.exe" | ||||
| @@ -49,12 +50,13 @@ Dynamic LLM libraries [rocm_v6 cpu cpu_avx cpu_avx2 cuda_v11 rocm_v5] | ||||
|  | ||||
| You can set OLLAMA_LLM_LIBRARY to any of the available LLM libraries to bypass autodetection, so for example, if you have a CUDA card, but want to force the CPU LLM library with AVX2 vector support, use: | ||||
|  | ||||
| ``` | ||||
| ```shell | ||||
| OLLAMA_LLM_LIBRARY="cpu_avx2" ollama serve | ||||
| ``` | ||||
|  | ||||
| You can see what features your CPU has with the following. | ||||
| ``` | ||||
|  | ||||
| ```shell | ||||
| cat /proc/cpuinfo| grep flags | head -1 | ||||
| ``` | ||||
|  | ||||
| @@ -62,13 +64,13 @@ cat /proc/cpuinfo| grep flags | head -1 | ||||
|  | ||||
| If you run into problems on Linux and want to install an older version, or you'd like to try out a pre-release before it's officially released, you can tell the install script which version to install. | ||||
|  | ||||
| ```sh | ||||
| curl -fsSL https://ollama.com/install.sh | OLLAMA_VERSION="0.1.29" sh | ||||
| ```shell | ||||
| curl -fsSL https://ollama.com/install.sh | OLLAMA_VERSION=0.5.7 sh | ||||
| ``` | ||||
|  | ||||
| ## Linux tmp noexec  | ||||
| ## Linux docker | ||||
|  | ||||
| If your system is configured with the "noexec" flag where Ollama stores its temporary executable files, you can specify an alternate location by setting OLLAMA_TMPDIR to a location writable by the user ollama runs as. For example OLLAMA_TMPDIR=/usr/share/ollama/ | ||||
| If Ollama initially works on the GPU in a docker container, but then switches to running on CPU after some period of time with errors in the server log reporting GPU discovery failures, this can be resolved by disabling systemd cgroup management in Docker.  Edit `/etc/docker/daemon.json` on the host and add `"exec-opts": ["native.cgroupdriver=cgroupfs"]` to the docker configuration. | ||||
|  | ||||
| ## NVIDIA GPU Discovery | ||||
|  | ||||
| @@ -97,8 +99,6 @@ On linux, AMD GPU access typically requires `video` and/or `render` group member | ||||
|  | ||||
| When running in a container, in some Linux distributions and container runtimes, the ollama process may be unable to access the GPU.  Use `ls -lnd /dev/kfd /dev/dri /dev/dri/*` on the host system to determine the **numeric** group IDs on your system, and pass additional `--group-add ...` arguments to the container so it can access the required devices.   For example, in the following output `crw-rw---- 1 0  44 226,   0 Sep 16 16:55 /dev/dri/card0` the group ID column is `44`  | ||||
|  | ||||
| If Ollama initially works on the GPU in a docker container, but then switches to running on CPU after some period of time with errors in the server log reporting GPU discovery failures, this can be resolved by disabling systemd cgroup management in Docker.  Edit `/etc/docker/daemon.json` on the host and add `"exec-opts": ["native.cgroupdriver=cgroupfs"]` to the docker configuration. | ||||
|  | ||||
| If you are experiencing problems getting Ollama to correctly discover or use your GPU for inference, the following may help isolate the failure. | ||||
| - `AMD_LOG_LEVEL=3` Enable info log levels in the AMD HIP/ROCm libraries.  This can help show more detailed error codes that can help troubleshoot problems | ||||
| - `OLLAMA_DEBUG=1` During GPU discovery additional information will be reported | ||||
|   | ||||
| @@ -47,6 +47,7 @@ If Ollama is already running, Quit the tray application and relaunch it from the | ||||
| ## API Access | ||||
|  | ||||
| Here's a quick example showing API access from `powershell` | ||||
|  | ||||
| ```powershell | ||||
| (Invoke-WebRequest -method POST -Body '{"model":"llama3.2", "prompt":"Why is the sky blue?", "stream": false}' -uri http://localhost:11434/api/generate ).Content | ConvertFrom-json | ||||
| ``` | ||||
| @@ -54,14 +55,13 @@ Here's a quick example showing API access from `powershell` | ||||
| ## Troubleshooting | ||||
|  | ||||
| Ollama on Windows stores files in a few different locations.  You can view them in | ||||
| the explorer window by hitting `<cmd>+R` and type in: | ||||
| the explorer window by hitting `<Ctrl>+R` and type in: | ||||
| - `explorer %LOCALAPPDATA%\Ollama` contains logs, and downloaded updates | ||||
|     - *app.log* contains most resent logs from the GUI application | ||||
|     - *server.log* contains the most recent server logs | ||||
|     - *upgrade.log* contains log output for upgrades | ||||
| - `explorer %LOCALAPPDATA%\Programs\Ollama` contains the binaries (The installer adds this to your user PATH) | ||||
| - `explorer %HOMEPATH%\.ollama` contains models and configuration | ||||
| - `explorer %TEMP%` contains temporary executable files in one or more `ollama*` directories | ||||
|  | ||||
| ## Uninstall | ||||
|  | ||||
| @@ -80,9 +80,11 @@ help you keep up to date. | ||||
|  | ||||
| If you'd like to install or integrate Ollama as a service, a standalone | ||||
| `ollama-windows-amd64.zip` zip file is available containing only the Ollama CLI | ||||
| and GPU library dependencies for Nvidia and AMD. This allows for embedding | ||||
| Ollama in existing applications, or running it as a system service via `ollama | ||||
| serve` with tools such as [NSSM](https://nssm.cc/). | ||||
| and GPU library dependencies for Nvidia.  If you have an AMD GPU, also download | ||||
| and extract the additional ROCm package `ollama-windows-amd64-rocm.zip` into the | ||||
| same directory.  This allows for embedding Ollama in existing applications, or | ||||
| running it as a system service via `ollama serve` with tools such as | ||||
| [NSSM](https://nssm.cc/).  | ||||
|  | ||||
| > [!NOTE]   | ||||
| > If you are upgrading from a prior version, you should remove the old directories first. | ||||
|   | ||||
| @@ -53,8 +53,8 @@ func Host() *url.URL { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Origins returns a list of allowed origins. Origins can be configured via the OLLAMA_ORIGINS environment variable. | ||||
| func Origins() (origins []string) { | ||||
| // AllowedOrigins returns a list of allowed origins. AllowedOrigins can be configured via the OLLAMA_ORIGINS environment variable. | ||||
| func AllowedOrigins() (origins []string) { | ||||
| 	if s := Var("OLLAMA_ORIGINS"); s != "" { | ||||
| 		origins = strings.Split(s, ",") | ||||
| 	} | ||||
| @@ -73,6 +73,7 @@ func Origins() (origins []string) { | ||||
| 		"file://*", | ||||
| 		"tauri://*", | ||||
| 		"vscode-webview://*", | ||||
| 		"vscode-file://*", | ||||
| 	) | ||||
|  | ||||
| 	return origins | ||||
| @@ -165,6 +166,10 @@ var ( | ||||
| 	IntelGPU = Bool("OLLAMA_INTEL_GPU") | ||||
| 	// MultiUserCache optimizes prompt caching for multi-user scenarios | ||||
| 	MultiUserCache = Bool("OLLAMA_MULTIUSER_CACHE") | ||||
| 	// Enable the new Ollama engine | ||||
| 	NewEngine = Bool("OLLAMA_NEW_ENGINE") | ||||
| 	// ContextLength sets the default context length | ||||
| 	ContextLength = Uint("OLLAMA_CONTEXT_LENGTH", 2048) | ||||
| ) | ||||
|  | ||||
| func String(s string) func() string { | ||||
| @@ -247,9 +252,11 @@ func AsMap() map[string]EnvVar { | ||||
| 		"OLLAMA_NOHISTORY":         {"OLLAMA_NOHISTORY", NoHistory(), "Do not preserve readline history"}, | ||||
| 		"OLLAMA_NOPRUNE":           {"OLLAMA_NOPRUNE", NoPrune(), "Do not prune model blobs on startup"}, | ||||
| 		"OLLAMA_NUM_PARALLEL":      {"OLLAMA_NUM_PARALLEL", NumParallel(), "Maximum number of parallel requests"}, | ||||
| 		"OLLAMA_ORIGINS":           {"OLLAMA_ORIGINS", Origins(), "A comma separated list of allowed origins"}, | ||||
| 		"OLLAMA_ORIGINS":           {"OLLAMA_ORIGINS", AllowedOrigins(), "A comma separated list of allowed origins"}, | ||||
| 		"OLLAMA_SCHED_SPREAD":      {"OLLAMA_SCHED_SPREAD", SchedSpread(), "Always schedule model across all GPUs"}, | ||||
| 		"OLLAMA_MULTIUSER_CACHE":   {"OLLAMA_MULTIUSER_CACHE", MultiUserCache(), "Optimize prompt caching for multi-user scenarios"}, | ||||
| 		"OLLAMA_CONTEXT_LENGTH":    {"OLLAMA_CONTEXT_LENGTH", ContextLength(), "Context length to use unless otherwise specified (default: 2048)"}, | ||||
| 		"OLLAMA_NEW_ENGINE":        {"OLLAMA_NEW_ENGINE", NewEngine(), "Enable the new Ollama engine"}, | ||||
|  | ||||
| 		// Informational | ||||
| 		"HTTP_PROXY":  {"HTTP_PROXY", String("HTTP_PROXY")(), "HTTP proxy"}, | ||||
| @@ -288,12 +295,3 @@ func Values() map[string]string { | ||||
| func Var(key string) string { | ||||
| 	return strings.Trim(strings.TrimSpace(os.Getenv(key)), "\"'") | ||||
| } | ||||
|  | ||||
| // On windows, we keep the binary at the top directory, but | ||||
| // other platforms use a "bin" directory, so this returns ".." | ||||
| func LibRelativeToExe() string { | ||||
| 	if runtime.GOOS == "windows" { | ||||
| 		return "." | ||||
| 	} | ||||
| 	return ".." | ||||
| } | ||||
|   | ||||
| @@ -69,6 +69,7 @@ func TestOrigins(t *testing.T) { | ||||
| 			"file://*", | ||||
| 			"tauri://*", | ||||
| 			"vscode-webview://*", | ||||
| 			"vscode-file://*", | ||||
| 		}}, | ||||
| 		{"http://10.0.0.1", []string{ | ||||
| 			"http://10.0.0.1", | ||||
| @@ -88,6 +89,7 @@ func TestOrigins(t *testing.T) { | ||||
| 			"file://*", | ||||
| 			"tauri://*", | ||||
| 			"vscode-webview://*", | ||||
| 			"vscode-file://*", | ||||
| 		}}, | ||||
| 		{"http://172.16.0.1,https://192.168.0.1", []string{ | ||||
| 			"http://172.16.0.1", | ||||
| @@ -108,6 +110,7 @@ func TestOrigins(t *testing.T) { | ||||
| 			"file://*", | ||||
| 			"tauri://*", | ||||
| 			"vscode-webview://*", | ||||
| 			"vscode-file://*", | ||||
| 		}}, | ||||
| 		{"http://totally.safe,http://definitely.legit", []string{ | ||||
| 			"http://totally.safe", | ||||
| @@ -128,13 +131,14 @@ func TestOrigins(t *testing.T) { | ||||
| 			"file://*", | ||||
| 			"tauri://*", | ||||
| 			"vscode-webview://*", | ||||
| 			"vscode-file://*", | ||||
| 		}}, | ||||
| 	} | ||||
| 	for _, tt := range cases { | ||||
| 		t.Run(tt.value, func(t *testing.T) { | ||||
| 			t.Setenv("OLLAMA_ORIGINS", tt.value) | ||||
|  | ||||
| 			if diff := cmp.Diff(Origins(), tt.expect); diff != "" { | ||||
| 			if diff := cmp.Diff(AllowedOrigins(), tt.expect); diff != "" { | ||||
| 				t.Errorf("%s: mismatch (-want +got):\n%s", tt.value, diff) | ||||
| 			} | ||||
| 		}) | ||||
| @@ -272,3 +276,19 @@ func TestVar(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestContextLength(t *testing.T) { | ||||
| 	cases := map[string]uint{ | ||||
| 		"":     2048, | ||||
| 		"4096": 4096, | ||||
| 	} | ||||
|  | ||||
| 	for k, v := range cases { | ||||
| 		t.Run(k, func(t *testing.T) { | ||||
| 			t.Setenv("OLLAMA_CONTEXT_LENGTH", k) | ||||
| 			if i := ContextLength(); i != v { | ||||
| 				t.Errorf("%s: expected %d, got %d", k, v, i) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -40,8 +40,6 @@ func HumanBytes(b int64) string { | ||||
| 	} | ||||
|  | ||||
| 	switch { | ||||
| 	case value >= 100: | ||||
| 		return fmt.Sprintf("%d %s", int(value), unit) | ||||
| 	case value >= 10: | ||||
| 		return fmt.Sprintf("%d %s", int(value), unit) | ||||
| 	case value != math.Trunc(value): | ||||
|   | ||||
							
								
								
									
										91
									
								
								format/bytes_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								format/bytes_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| package format | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestHumanBytes(t *testing.T) { | ||||
| 	type testCase struct { | ||||
| 		input    int64 | ||||
| 		expected string | ||||
| 	} | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		// Test bytes (B) | ||||
| 		{0, "0 B"}, | ||||
| 		{1, "1 B"}, | ||||
| 		{999, "999 B"}, | ||||
|  | ||||
| 		// Test kilobytes (KB) | ||||
| 		{1000, "1 KB"}, | ||||
| 		{1500, "1.5 KB"}, | ||||
| 		{999999, "999 KB"}, | ||||
|  | ||||
| 		// Test megabytes (MB) | ||||
| 		{1000000, "1 MB"}, | ||||
| 		{1500000, "1.5 MB"}, | ||||
| 		{999999999, "999 MB"}, | ||||
|  | ||||
| 		// Test gigabytes (GB) | ||||
| 		{1000000000, "1 GB"}, | ||||
| 		{1500000000, "1.5 GB"}, | ||||
| 		{999999999999, "999 GB"}, | ||||
|  | ||||
| 		// Test terabytes (TB) | ||||
| 		{1000000000000, "1 TB"}, | ||||
| 		{1500000000000, "1.5 TB"}, | ||||
| 		{1999999999999, "2.0 TB"}, | ||||
|  | ||||
| 		// Test fractional values | ||||
| 		{1234, "1.2 KB"}, | ||||
| 		{1234567, "1.2 MB"}, | ||||
| 		{1234567890, "1.2 GB"}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		t.Run(tc.expected, func(t *testing.T) { | ||||
| 			result := HumanBytes(tc.input) | ||||
| 			if result != tc.expected { | ||||
| 				t.Errorf("Expected %s, got %s", tc.expected, result) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestHumanBytes2(t *testing.T) { | ||||
| 	type testCase struct { | ||||
| 		input    uint64 | ||||
| 		expected string | ||||
| 	} | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		// Test bytes (B) | ||||
| 		{0, "0 B"}, | ||||
| 		{1, "1 B"}, | ||||
| 		{1023, "1023 B"}, | ||||
|  | ||||
| 		// Test kibibytes (KiB) | ||||
| 		{1024, "1.0 KiB"}, | ||||
| 		{1536, "1.5 KiB"}, | ||||
| 		{1048575, "1024.0 KiB"}, | ||||
|  | ||||
| 		// Test mebibytes (MiB) | ||||
| 		{1048576, "1.0 MiB"}, | ||||
| 		{1572864, "1.5 MiB"}, | ||||
| 		{1073741823, "1024.0 MiB"}, | ||||
|  | ||||
| 		// Test gibibytes (GiB) | ||||
| 		{1073741824, "1.0 GiB"}, | ||||
| 		{1610612736, "1.5 GiB"}, | ||||
| 		{2147483648, "2.0 GiB"}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		t.Run(tc.expected, func(t *testing.T) { | ||||
| 			result := HumanBytes2(tc.input) | ||||
| 			if result != tc.expected { | ||||
| 				t.Errorf("Expected %s, got %s", tc.expected, result) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -12,6 +12,9 @@ func TestHumanNumber(t *testing.T) { | ||||
|  | ||||
| 	testCases := []testCase{ | ||||
| 		{0, "0"}, | ||||
| 		{999, "999"}, | ||||
| 		{1000, "1K"}, | ||||
| 		{1001, "1K"}, | ||||
| 		{1000000, "1M"}, | ||||
| 		{125000000, "125M"}, | ||||
| 		{500500000, "500.50M"}, | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func assertEqual(t *testing.T, a interface{}, b interface{}) { | ||||
| func assertEqual(t *testing.T, a any, b any) { | ||||
| 	if a != b { | ||||
| 		t.Errorf("Assert failed, expected %v, got %v", b, a) | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										13
									
								
								fs/config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								fs/config.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| package fs | ||||
|  | ||||
| type Config interface { | ||||
| 	Architecture() string | ||||
| 	String(string, ...string) string | ||||
| 	Uint(string, ...uint32) uint32 | ||||
| 	Float(string, ...float32) float32 | ||||
| 	Bool(string, ...bool) bool | ||||
|  | ||||
| 	Strings(string, ...[]string) []string | ||||
| 	Uints(string, ...[]uint32) []uint32 | ||||
| 	Floats(string, ...[]float32) []float32 | ||||
| } | ||||
| @@ -1,15 +1,15 @@ | ||||
| package llm | ||||
| package ggml | ||||
| 
 | ||||
| import ( | ||||
| 	"encoding/binary" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"log/slog" | ||||
| 	"slices" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 
 | ||||
| 	"github.com/ollama/ollama/util/bufioutil" | ||||
| 	"github.com/ollama/ollama/fs/util/bufioutil" | ||||
| ) | ||||
| 
 | ||||
| type GGML struct { | ||||
| @@ -19,145 +19,186 @@ type GGML struct { | ||||
| 
 | ||||
| type model interface { | ||||
| 	KV() KV | ||||
| 	Tensors() *Tensors | ||||
| 	Tensors() Tensors | ||||
| } | ||||
| 
 | ||||
| type KV map[string]any | ||||
| 
 | ||||
| func (kv KV) u64(key string) uint64 { | ||||
| 	switch v := kv[key].(type) { | ||||
| 	case uint64: | ||||
| 		return v | ||||
| 	case uint32: | ||||
| 		return uint64(v) | ||||
| 	case float64: | ||||
| 		return uint64(v) | ||||
| 	default: | ||||
| 		return 0 | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Architecture() string { | ||||
| 	if s, ok := kv["general.architecture"].(string); ok { | ||||
| 		return s | ||||
| 	} | ||||
| 
 | ||||
| 	return "unknown" | ||||
| 	return kv.String("general.architecture", "unknown") | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Kind() string { | ||||
| 	if s, ok := kv["general.type"].(string); ok { | ||||
| 		return s | ||||
| 	} | ||||
| 
 | ||||
| 	return "unknown" | ||||
| 	return kv.String("general.type", "unknown") | ||||
| } | ||||
| 
 | ||||
| func (kv KV) ParameterCount() uint64 { | ||||
| 	return kv.u64("general.parameter_count") | ||||
| 	return keyValue[uint64](kv, "general.parameter_count") | ||||
| } | ||||
| 
 | ||||
| func (kv KV) FileType() fileType { | ||||
| 	if u64 := kv.u64("general.file_type"); u64 > 0 { | ||||
| 		return fileType(uint32(u64)) | ||||
| 	if t := kv.Uint("general.file_type"); t > 0 { | ||||
| 		return fileType(t) | ||||
| 	} | ||||
| 
 | ||||
| 	return fileTypeUnknown | ||||
| } | ||||
| 
 | ||||
| func (kv KV) BlockCount() uint64 { | ||||
| 	return kv.u64(fmt.Sprintf("%s.block_count", kv.Architecture())) | ||||
| 	return uint64(kv.Uint("block_count")) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) EmbeddingLength() uint64 { | ||||
| 	return uint64(kv.Uint("embedding_length")) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) HeadCount() uint64 { | ||||
| 	return kv.u64(fmt.Sprintf("%s.attention.head_count", kv.Architecture())) | ||||
| 	return uint64(kv.Uint("attention.head_count")) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) HeadCountKV() uint64 { | ||||
| 	if headCountKV := kv.u64(fmt.Sprintf("%s.attention.head_count_kv", kv.Architecture())); headCountKV > 0 { | ||||
| 		return headCountKV | ||||
| 	} | ||||
| 
 | ||||
| 	return 1 | ||||
| 	return uint64(kv.Uint("attention.head_count_kv", 1)) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) EmbeddingHeadCount() uint64 { | ||||
| 	if heads := kv.HeadCount(); heads > 0 { | ||||
| 		return kv.EmbeddingLength() / kv.HeadCount() | ||||
| 		return kv.EmbeddingLength() / heads | ||||
| 	} | ||||
| 
 | ||||
| 	return 0 | ||||
| } | ||||
| 
 | ||||
| func (kv KV) EmbeddingHeadCountK() uint64 { | ||||
| 	if k := kv.u64(fmt.Sprintf("%s.attention.key_length", kv.Architecture())); k > 0 { | ||||
| 		return k | ||||
| 	} | ||||
| 
 | ||||
| 	return kv.EmbeddingHeadCount() | ||||
| 	return uint64(kv.Uint("attention.key_length", uint32(kv.EmbeddingHeadCount()))) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) EmbeddingHeadCountV() uint64 { | ||||
| 	if v := kv.u64(fmt.Sprintf("%s.attention.value_length", kv.Architecture())); v > 0 { | ||||
| 		return v | ||||
| 	} | ||||
| 
 | ||||
| 	return kv.EmbeddingHeadCount() | ||||
| 	return uint64(kv.Uint("attention.value_length", uint32(kv.EmbeddingHeadCount()))) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) GQA() uint64 { | ||||
| 	return kv.HeadCount() / kv.HeadCountKV() | ||||
| } | ||||
| 
 | ||||
| func (kv KV) EmbeddingLength() uint64 { | ||||
| 	return kv.u64(fmt.Sprintf("%s.embedding_length", kv.Architecture())) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) ContextLength() uint64 { | ||||
| 	return kv.u64(fmt.Sprintf("%s.context_length", kv.Architecture())) | ||||
| 	return uint64(kv.Uint("context_length")) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) ChatTemplate() string { | ||||
| 	s, _ := kv["tokenizer.chat_template"].(string) | ||||
| 	return kv.String("tokenizer.chat_template") | ||||
| } | ||||
| 
 | ||||
| func (kv KV) String(key string, defaultValue ...string) string { | ||||
| 	return keyValue(kv, key, append(defaultValue, "")...) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Uint(key string, defaultValue ...uint32) uint32 { | ||||
| 	return keyValue(kv, key, append(defaultValue, 0)...) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Float(key string, defaultValue ...float32) float32 { | ||||
| 	return keyValue(kv, key, append(defaultValue, 0)...) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Bool(key string, defaultValue ...bool) bool { | ||||
| 	return keyValue(kv, key, append(defaultValue, false)...) | ||||
| } | ||||
| 
 | ||||
| func (kv KV) Strings(key string, defaultValue ...[]string) []string { | ||||
| 	r := keyValue(kv, key, &array{}) | ||||
| 	s := make([]string, r.size) | ||||
| 	for i := range r.size { | ||||
| 		s[i] = r.values[i].(string) | ||||
| 	} | ||||
| 
 | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| type Tensors struct { | ||||
| 	Items  []*Tensor | ||||
| 	Offset uint64 | ||||
| func (kv KV) Uints(key string, defaultValue ...[]uint32) []uint32 { | ||||
| 	r := keyValue(kv, key, &array{}) | ||||
| 	s := make([]uint32, r.size) | ||||
| 	for i := range r.size { | ||||
| 		s[i] = uint32(r.values[i].(int32)) | ||||
| 	} | ||||
| 
 | ||||
| 	layers     map[string]Layer | ||||
| 	layersOnce sync.Once | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| func (ts *Tensors) Layers() map[string]Layer { | ||||
| 	ts.layersOnce.Do(func() { | ||||
| 		ts.layers = make(map[string]Layer) | ||||
| 		for _, t := range ts.Items { | ||||
| 			parts := strings.Split(t.Name, ".") | ||||
| 			if index := slices.IndexFunc(parts, func(s string) bool { return s == "blk" || s == "mm" }); index != -1 { | ||||
| 				if len(parts) > index+2 { | ||||
| 					// blk and mm should have a number after them, join it | ||||
| 					parts = append( | ||||
| 						[]string{strings.Join(parts[:index+2], ".")}, | ||||
| 						parts[index+2:]...) | ||||
| 				} | ||||
| 			} | ||||
| func (kv KV) Floats(key string, defaultValue ...[]float32) []float32 { | ||||
| 	r := keyValue(kv, key, &array{}) | ||||
| 	s := make([]float32, r.size) | ||||
| 	for i := range r.size { | ||||
| 		s[i] = float32(r.values[i].(float32)) | ||||
| 	} | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| 			if _, ok := ts.layers[parts[0]]; !ok { | ||||
| 				ts.layers[parts[0]] = make(Layer) | ||||
| 			} | ||||
| func (kv KV) OllamaEngineRequired() bool { | ||||
| 	return slices.Contains([]string{ | ||||
| 		"gemma3", | ||||
| 		"mistral3", | ||||
| 	}, kv.Architecture()) | ||||
| } | ||||
| 
 | ||||
| 			ts.layers[parts[0]][strings.Join(parts[1:], ".")] = t | ||||
| func keyValue[T string | uint32 | uint64 | float32 | *array | bool](kv KV, key string, defaultValue ...T) T { | ||||
| 	if !strings.HasPrefix(key, "tokenizer.") && !strings.HasPrefix(key, "general.") { | ||||
| 		key = kv.Architecture() + "." + key | ||||
| 	} | ||||
| 
 | ||||
| 	if val, ok := kv[key]; ok { | ||||
| 		return val.(T) | ||||
| 	} | ||||
| 
 | ||||
| 	slog.Warn("key not found", "key", key, "default", defaultValue[0]) | ||||
| 	return defaultValue[0] | ||||
| } | ||||
| 
 | ||||
| type Tensors struct { | ||||
| 	items  []*Tensor | ||||
| 	Offset uint64 | ||||
| } | ||||
| 
 | ||||
| func (s Tensors) Items(prefix ...string) []*Tensor { | ||||
| 	if len(prefix) == 0 { | ||||
| 		return s.items | ||||
| 	} | ||||
| 
 | ||||
| 	var items []*Tensor | ||||
| 	for _, t := range s.items { | ||||
| 		if strings.HasPrefix(t.Name, prefix[0]) { | ||||
| 			items = append(items, t) | ||||
| 		} | ||||
| 	}) | ||||
| 	} | ||||
| 
 | ||||
| 	return ts.layers | ||||
| 	return items | ||||
| } | ||||
| 
 | ||||
| func (ts Tensors) GroupLayers() map[string]Layer { | ||||
| 	layers := make(map[string]Layer) | ||||
| 	for _, t := range ts.items { | ||||
| 		parts := strings.Split(t.Name, ".") | ||||
| 		if index := slices.IndexFunc(parts, func(s string) bool { return s == "blk" || s == "mm" }); index != -1 { | ||||
| 			if len(parts) > index+2 { | ||||
| 				// blk and mm should have a number after them, join it | ||||
| 				parts = append( | ||||
| 					[]string{strings.Join(parts[:index+2], ".")}, | ||||
| 					parts[index+2:]...) | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if _, ok := layers[parts[0]]; !ok { | ||||
| 			layers[parts[0]] = make(Layer) | ||||
| 		} | ||||
| 
 | ||||
| 		layers[parts[0]][strings.Join(parts[1:], ".")] = t | ||||
| 	} | ||||
| 
 | ||||
| 	return layers | ||||
| } | ||||
| 
 | ||||
| type Layer map[string]*Tensor | ||||
| 
 | ||||
| func (l Layer) size() (size uint64) { | ||||
| func (l Layer) Size() (size uint64) { | ||||
| 	for _, t := range l { | ||||
| 		size += t.Size() | ||||
| 	} | ||||
| @@ -186,11 +227,26 @@ func (t Tensor) block() (n int) { | ||||
| 
 | ||||
| func (t Tensor) blockSize() uint64 { | ||||
| 	switch t.Kind { | ||||
| 	case 0, 1, 24, 25, 26, 27, 28, 30: // F32, F16, I8, I16, I32, I64, F64, BF16 | ||||
| 	case | ||||
| 		0,  // F32 | ||||
| 		1,  // F16 | ||||
| 		24, // I8 | ||||
| 		25, // I16 | ||||
| 		26, // I32 | ||||
| 		27, // I64 | ||||
| 		28, // F64 | ||||
| 		30: // BF16 | ||||
| 		return 1 | ||||
| 	case 2, 3, 4, 5, 6, 7, 8, 9, 20: // Q4_0, Q4_1, Q5_0, Q5_1, Q8_0, Q8_1, IQ4_NL | ||||
| 	case | ||||
| 		2,  // Q4_0 | ||||
| 		3,  // Q4_1 | ||||
| 		6,  // Q5_0 | ||||
| 		7,  // Q5_1 | ||||
| 		8,  // Q8_0 | ||||
| 		9,  // Q8_1 | ||||
| 		20: // IQ4_NL | ||||
| 		return 32 | ||||
| 	default: // All others | ||||
| 	default: | ||||
| 		return 256 | ||||
| 	} | ||||
| } | ||||
| @@ -214,7 +270,7 @@ func (t Tensor) typeSize() uint64 { | ||||
| 	case 8: // Q8_0 | ||||
| 		return 2 + blockSize | ||||
| 	case 9: // Q8_1 | ||||
| 		return 4 + 4 + blockSize | ||||
| 		return 2 + 2 + blockSize | ||||
| 	case 10: // Q2_K | ||||
| 		return blockSize/16 + blockSize/4 + 2 + 2 | ||||
| 	case 11: // Q3_K | ||||
| @@ -226,7 +282,7 @@ func (t Tensor) typeSize() uint64 { | ||||
| 	case 14: // Q6_K | ||||
| 		return blockSize/2 + blockSize/4 + blockSize/16 + 2 | ||||
| 	case 15: // Q8_K | ||||
| 		return 2 + blockSize + 2*blockSize/16 | ||||
| 		return 4 + blockSize + 2*blockSize/16 | ||||
| 	case 16: // IQ2_XXS | ||||
| 		return 2 + 2*blockSize/8 | ||||
| 	case 17: // IQ2_XS | ||||
| @@ -274,6 +330,10 @@ func (t Tensor) Size() uint64 { | ||||
| 	return t.parameters() * t.typeSize() / t.blockSize() | ||||
| } | ||||
| 
 | ||||
| func (t Tensor) Type() string { | ||||
| 	return fileType(t.Kind).String() | ||||
| } | ||||
| 
 | ||||
| type container interface { | ||||
| 	Name() string | ||||
| 	Decode(io.ReadSeeker) (model, error) | ||||
| @@ -295,7 +355,7 @@ const ( | ||||
| 
 | ||||
| var ErrUnsupportedFormat = errors.New("unsupported model format") | ||||
| 
 | ||||
| func DetectGGMLType(b []byte) string { | ||||
| func DetectContentType(b []byte) string { | ||||
| 	switch binary.LittleEndian.Uint32(b[:4]) { | ||||
| 	case FILE_MAGIC_GGML: | ||||
| 		return "ggml" | ||||
| @@ -312,12 +372,12 @@ func DetectGGMLType(b []byte) string { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // DecodeGGML decodes a GGML model from the given reader. | ||||
| // Decode decodes a GGML model from the given reader. | ||||
| // | ||||
| // It collects array values for arrays with a size less than or equal to | ||||
| // maxArraySize. If maxArraySize is 0, the default value of 1024 is used. If | ||||
| // the maxArraySize is negative, all arrays are collected. | ||||
| func DecodeGGML(rs io.ReadSeeker, maxArraySize int) (*GGML, int64, error) { | ||||
| func Decode(rs io.ReadSeeker, maxArraySize int) (*GGML, int64, error) { | ||||
| 	if maxArraySize == 0 { | ||||
| 		maxArraySize = 1024 | ||||
| 	} | ||||
| @@ -331,10 +391,6 @@ func DecodeGGML(rs io.ReadSeeker, maxArraySize int) (*GGML, int64, error) { | ||||
| 
 | ||||
| 	var c container | ||||
| 	switch magic { | ||||
| 	case FILE_MAGIC_GGML, FILE_MAGIC_GGMF, FILE_MAGIC_GGJT: | ||||
| 		return nil, 0, ErrUnsupportedFormat | ||||
| 	case FILE_MAGIC_GGLA: | ||||
| 		c = &containerGGLA{} | ||||
| 	case FILE_MAGIC_GGUF_LE: | ||||
| 		c = &containerGGUF{ByteOrder: binary.LittleEndian, maxArraySize: maxArraySize} | ||||
| 	case FILE_MAGIC_GGUF_BE: | ||||
| @@ -360,22 +416,25 @@ func DecodeGGML(rs io.ReadSeeker, maxArraySize int) (*GGML, int64, error) { | ||||
| 	}, offset, nil | ||||
| } | ||||
| 
 | ||||
| func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partialOffload, fullOffload uint64) { | ||||
| 	embedding := llm.KV().EmbeddingLength() | ||||
| 	heads := llm.KV().HeadCount() | ||||
| 	headsKV := llm.KV().HeadCountKV() | ||||
| 	vocab := uint64(llm.KV()["tokenizer.ggml.tokens"].(*array).size) | ||||
| func (f GGML) GraphSize(context, batch uint64, numParallel int, kvCacheType string) (kv []uint64, partialOffload, fullOffload uint64) { | ||||
| 	embedding := f.KV().EmbeddingLength() | ||||
| 	heads := f.KV().HeadCount() | ||||
| 	headsKV := f.KV().HeadCountKV() | ||||
| 	vocab := uint64(f.KV()["tokenizer.ggml.tokens"].(*array).size) | ||||
| 
 | ||||
| 	embeddingHeads := llm.KV().EmbeddingHeadCount() | ||||
| 	embeddingHeadsK := llm.KV().EmbeddingHeadCountK() | ||||
| 	embeddingHeadsV := llm.KV().EmbeddingHeadCountV() | ||||
| 	embeddingHeads := f.KV().EmbeddingHeadCount() | ||||
| 	embeddingHeadsK := f.KV().EmbeddingHeadCountK() | ||||
| 	embeddingHeadsV := f.KV().EmbeddingHeadCountV() | ||||
| 
 | ||||
| 	layers := llm.Tensors().Layers() | ||||
| 	layers := f.Tensors().GroupLayers() | ||||
| 
 | ||||
| 	bytesPerElement := kvCacheBytesPerElement(kvCacheType) | ||||
| 	kv = uint64(float64(context*llm.KV().BlockCount()*(embeddingHeadsK+embeddingHeadsV)*headsKV) * bytesPerElement) | ||||
| 	kv = make([]uint64, f.KV().BlockCount()) | ||||
| 	for i := range kv { | ||||
| 		kv[i] = uint64(float64(context*(embeddingHeadsK+embeddingHeadsV)*headsKV) * bytesPerElement) | ||||
| 	} | ||||
| 
 | ||||
| 	switch llm.KV().Architecture() { | ||||
| 	switch f.KV().Architecture() { | ||||
| 	case "llama": | ||||
| 		fullOffload = max( | ||||
| 			4*batch*(1+4*embedding+context*(1+heads)), | ||||
| @@ -390,7 +449,7 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 
 | ||||
| 		if ffnGateExpsWeight, ok := layers["blk.0"]["ffn_gate_exps.weight"]; ok { | ||||
| 			// mixtral 8x22b | ||||
| 			ff := uint64(llm.KV()["llama.feed_forward_length"].(uint32)) | ||||
| 			ff := uint64(f.KV()["llama.feed_forward_length"].(uint32)) | ||||
| 			partialOffload = max( | ||||
| 				3*ffnGateExpsWeight.Size()+4*batch*(2*ff+headsKV+embedding+context+embeddingHeads*headsKV), | ||||
| 				4*(context*batch*heads+context*embeddingHeads*headsKV+batch*1024+embeddingHeads*headsKV*batch), | ||||
| @@ -407,16 +466,14 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 	case "mllama": | ||||
| 		var visionTokens, tiles uint64 = 1601, 4 | ||||
| 
 | ||||
| 		if crossAttentionLayers, ok := llm.KV()["mllama.attention.cross_attention_layers"].(*array); ok { | ||||
| 			kv = headsKV * | ||||
| 				(embeddingHeadsK + embeddingHeadsV) * // one for K, one for V | ||||
| 				(2* // sizeof(float16) | ||||
| 					(llm.KV().BlockCount()-uint64(crossAttentionLayers.size))* // num non-cross attention layers | ||||
| 					context + | ||||
| 					4* // sizeof(float32) | ||||
| 						uint64(crossAttentionLayers.size)* // num cross attention layers | ||||
| 						visionTokens* | ||||
| 						tiles) | ||||
| 		crossAttentionLayers := f.KV().Uints("attention.cross_attention_layers") | ||||
| 		for i := range kv { | ||||
| 			if slices.Contains(crossAttentionLayers, uint32(i)) { | ||||
| 				kv[i] = headsKV * (embeddingHeadsK + embeddingHeadsV) * | ||||
| 					4 * // sizeof(float32) | ||||
| 					visionTokens * | ||||
| 					tiles | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		fullOffload = max( | ||||
| @@ -426,7 +483,7 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 		) | ||||
| 
 | ||||
| 		var ropeFreqsCount uint64 | ||||
| 		if ropeFreqs, ok := llm.Tensors().Layers()["rope_freqs"]; ok { | ||||
| 		if ropeFreqs, ok := f.Tensors().GroupLayers()["rope_freqs"]; ok { | ||||
| 			if ropeFreqsWeights, ok := ropeFreqs["weights"]; ok { | ||||
| 				ropeFreqsCount = ropeFreqsWeights.parameters() | ||||
| 			} | ||||
| @@ -440,7 +497,7 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 			// vocab graph | ||||
| 			4*batch*(embedding+vocab)+embedding*vocab*105/128, | ||||
| 		) | ||||
| 	case "gemma", "gemma2": | ||||
| 	case "gemma", "gemma2", "gemma3": | ||||
| 		fullOffload = max( | ||||
| 			4*batch*(embedding+vocab), | ||||
| 			4*batch*(2+context+context*heads+2*embedding+2*embeddingHeadsK*heads), | ||||
| @@ -452,6 +509,20 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 				4*embeddingHeadsK*context*8+ | ||||
| 				embedding*embeddingHeadsK*heads*9/16, | ||||
| 		) | ||||
| 
 | ||||
| 		// Gemma2 also has sliding window attention but we only have an optimized implementation in the Ollama | ||||
| 		// engine. Gemma3 always uses the Ollama engine. | ||||
| 		if f.KV().Architecture() == "gemma3" { | ||||
| 			const gemma3GlobalCacheCount = 6 | ||||
| 			slidingWindow := (uint64(numParallel) * uint64(f.KV().Uint("attention.sliding_window"))) + batch | ||||
| 			for i := range kv { | ||||
| 				// Every 6th layer is a global layer, which is the full context size that has already been set. The other | ||||
| 				// layers are the smaller local (sliding) layers. | ||||
| 				if (i+1)%gemma3GlobalCacheCount != 0 { | ||||
| 					kv[i] = uint64(float64(slidingWindow*(embeddingHeadsK+embeddingHeadsV)*headsKV) * bytesPerElement) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	case "command-r": | ||||
| 		fullOffload = max( | ||||
| 			4*batch*(embedding+vocab), | ||||
| @@ -529,22 +600,71 @@ func (llm GGML) GraphSize(context, batch uint64, kvCacheType string) (kv, partia | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func (llm GGML) VisionGraphSize() (weights, graphSize uint64) { | ||||
| 	if llm.KV().Uint("vision.block_count") == 0 { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	for name, layer := range llm.Tensors().GroupLayers() { | ||||
| 		if name == "v" || strings.HasPrefix(name, "v.") { | ||||
| 			for _, tensor := range layer { | ||||
| 				weights += tensor.Size() | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	imageSize := uint64(llm.KV().Uint("vision.image_size")) | ||||
| 	patchSize := uint64(llm.KV().Uint("vision.patch_size")) | ||||
| 	if patchSize == 0 { | ||||
| 		slog.Warn("unknown patch size for vision model") | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	numChannels := uint64(llm.KV().Uint("vision.num_channels")) | ||||
| 
 | ||||
| 	numPatches := (imageSize / patchSize) * (imageSize / patchSize) | ||||
| 	if _, ok := llm.Tensors().GroupLayers()["v"]["class_embd"]; ok { | ||||
| 		numPatches++ | ||||
| 	} | ||||
| 
 | ||||
| 	headCount := uint64(llm.KV().Uint("vision.attention.head_count")) | ||||
| 	embeddingLength := uint64(llm.KV().Uint("vision.embedding_length")) | ||||
| 
 | ||||
| 	switch llm.KV().Architecture() { | ||||
| 	case "mllama": | ||||
| 		numPaddedPatches := numPatches + 8 - (numPatches%8)%8 | ||||
| 
 | ||||
| 		maxNumTiles := uint64(llm.KV().Uint("vision.max_num_tiles")) | ||||
| 
 | ||||
| 		graphSize = 4 * (8 + | ||||
| 			imageSize*imageSize*numChannels*maxNumTiles + | ||||
| 			embeddingLength*numPatches*maxNumTiles + | ||||
| 			9*embeddingLength*numPaddedPatches*maxNumTiles + | ||||
| 			numPaddedPatches*maxNumTiles*numPaddedPatches*maxNumTiles*headCount) | ||||
| 	case "gemma3", "mistral3": | ||||
| 		graphSize = 4 * (imageSize*imageSize*numChannels + | ||||
| 			embeddingLength*patchSize + | ||||
| 			numPatches*numPatches*headCount) | ||||
| 	} | ||||
| 
 | ||||
| 	return weights, graphSize | ||||
| } | ||||
| 
 | ||||
| // SupportsKVCacheType checks if the requested cache type is supported | ||||
| func (ggml GGML) SupportsKVCacheType(cacheType string) bool { | ||||
| 	validKVCacheTypes := []string{"f16", "q8_0", "q4_0"} | ||||
| 	return slices.Contains(validKVCacheTypes, cacheType) | ||||
| func (f GGML) SupportsKVCacheType(cacheType string) bool { | ||||
| 	return slices.Contains([]string{"f16", "q8_0", "q4_0"}, cacheType) | ||||
| } | ||||
| 
 | ||||
| // SupportsFlashAttention checks if the model supports flash attention | ||||
| func (ggml GGML) SupportsFlashAttention() bool { | ||||
| 	_, isEmbedding := ggml.KV()[fmt.Sprintf("%s.pooling_type", ggml.KV().Architecture())] | ||||
| func (f GGML) SupportsFlashAttention() bool { | ||||
| 	_, isEmbedding := f.KV()[fmt.Sprintf("%s.pooling_type", f.KV().Architecture())] | ||||
| 	if isEmbedding { | ||||
| 		return false | ||||
| 	} | ||||
| 
 | ||||
| 	// Check head counts match and are non-zero | ||||
| 	headCountK := ggml.KV().EmbeddingHeadCountK() | ||||
| 	headCountV := ggml.KV().EmbeddingHeadCountV() | ||||
| 	headCountK := f.KV().EmbeddingHeadCountK() | ||||
| 	headCountV := f.KV().EmbeddingHeadCountV() | ||||
| 	return headCountK != 0 && headCountV != 0 && headCountK == headCountV | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										212
									
								
								fs/ggml/ggml_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										212
									
								
								fs/ggml/ggml_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,212 @@ | ||||
| package ggml | ||||
|  | ||||
| import ( | ||||
| 	"maps" | ||||
| 	"slices" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| ) | ||||
|  | ||||
| func TestTensorLayers(t *testing.T) { | ||||
| 	tensors := make(map[string]*Tensor) | ||||
| 	for _, name := range []string{ | ||||
| 		"token_embd.weight", | ||||
| 		"blk.0.attn_k.weight", | ||||
| 		"blk.0.attn_output.weight", | ||||
| 		"blk.0.attn_q.weight", | ||||
| 		"blk.0.attn_v.weight", | ||||
| 		"blk.0.attn_norm.weight", | ||||
| 		"blk.0.ffn_down.weight", | ||||
| 		"blk.0.ffn_gate.weight", | ||||
| 		"blk.0.ffn_up.weight", | ||||
| 		"blk.0.ffn_norm.weight", | ||||
| 		"output_norm.weight", | ||||
| 		"mm.0.bias", | ||||
| 		"mm.0.weight", | ||||
| 		"v.blk.0.attn_k.weight", | ||||
| 		"v.blk.0.attn_output.weight", | ||||
| 		"v.blk.0.attn_q.weight", | ||||
| 		"v.blk.0.attn_v.weight", | ||||
| 		"v.blk.0.attn_norm.weight", | ||||
| 		"v.blk.0.ffn_down.weight", | ||||
| 		"v.blk.0.ffn_gate.weight", | ||||
| 		"v.blk.0.ffn_up.weight", | ||||
| 		"v.blk.0.ffn_norm.weight", | ||||
| 		"v.patch_embd.weight", | ||||
| 		"v.position_embd.gate", | ||||
| 		"v.position_embd.weight", | ||||
| 	} { | ||||
| 		tensors[name] = &Tensor{Name: name} | ||||
| 	} | ||||
|  | ||||
| 	cases := []struct { | ||||
| 		name  string | ||||
| 		items []*Tensor | ||||
| 		want  map[string]Layer | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "text", | ||||
| 			items: slices.Collect(func(yield func(*Tensor) bool) { | ||||
| 				for k, v := range tensors { | ||||
| 					if !strings.HasPrefix(k, "mm.") && !strings.HasPrefix(k, "v.") { | ||||
| 						if !yield(v) { | ||||
| 							return | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}), | ||||
| 			want: map[string]Layer{ | ||||
| 				"blk.0": { | ||||
| 					"attn_k.weight":      tensors["blk.0.attn_k.weight"], | ||||
| 					"attn_q.weight":      tensors["blk.0.attn_q.weight"], | ||||
| 					"attn_v.weight":      tensors["blk.0.attn_v.weight"], | ||||
| 					"attn_output.weight": tensors["blk.0.attn_output.weight"], | ||||
| 					"attn_norm.weight":   tensors["blk.0.attn_norm.weight"], | ||||
| 					"ffn_down.weight":    tensors["blk.0.ffn_down.weight"], | ||||
| 					"ffn_gate.weight":    tensors["blk.0.ffn_gate.weight"], | ||||
| 					"ffn_up.weight":      tensors["blk.0.ffn_up.weight"], | ||||
| 					"ffn_norm.weight":    tensors["blk.0.ffn_norm.weight"], | ||||
| 				}, | ||||
| 				"token_embd":  {"weight": tensors["token_embd.weight"]}, | ||||
| 				"output_norm": {"weight": tensors["output_norm.weight"]}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "vision", | ||||
| 			items: slices.Collect(func(yield func(*Tensor) bool) { | ||||
| 				for k, v := range tensors { | ||||
| 					if strings.HasPrefix(k, "mm.") || strings.HasPrefix(k, "v.") { | ||||
| 						if !yield(v) { | ||||
| 							return | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}), | ||||
| 			want: map[string]Layer{ | ||||
| 				"mm.0": { | ||||
| 					"bias":   tensors["mm.0.bias"], | ||||
| 					"weight": tensors["mm.0.weight"], | ||||
| 				}, | ||||
| 				"v.blk.0": { | ||||
| 					"attn_k.weight":      tensors["v.blk.0.attn_k.weight"], | ||||
| 					"attn_q.weight":      tensors["v.blk.0.attn_q.weight"], | ||||
| 					"attn_v.weight":      tensors["v.blk.0.attn_v.weight"], | ||||
| 					"attn_output.weight": tensors["v.blk.0.attn_output.weight"], | ||||
| 					"attn_norm.weight":   tensors["v.blk.0.attn_norm.weight"], | ||||
| 					"ffn_down.weight":    tensors["v.blk.0.ffn_down.weight"], | ||||
| 					"ffn_gate.weight":    tensors["v.blk.0.ffn_gate.weight"], | ||||
| 					"ffn_up.weight":      tensors["v.blk.0.ffn_up.weight"], | ||||
| 					"ffn_norm.weight":    tensors["v.blk.0.ffn_norm.weight"], | ||||
| 				}, | ||||
| 				"v": { | ||||
| 					"patch_embd.weight":    tensors["v.patch_embd.weight"], | ||||
| 					"position_embd.gate":   tensors["v.position_embd.gate"], | ||||
| 					"position_embd.weight": tensors["v.position_embd.weight"], | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:  "vision and text", | ||||
| 			items: slices.Collect(maps.Values(tensors)), | ||||
| 			want: map[string]Layer{ | ||||
| 				"blk.0": { | ||||
| 					"attn_k.weight":      tensors["blk.0.attn_k.weight"], | ||||
| 					"attn_q.weight":      tensors["blk.0.attn_q.weight"], | ||||
| 					"attn_v.weight":      tensors["blk.0.attn_v.weight"], | ||||
| 					"attn_output.weight": tensors["blk.0.attn_output.weight"], | ||||
| 					"attn_norm.weight":   tensors["blk.0.attn_norm.weight"], | ||||
| 					"ffn_down.weight":    tensors["blk.0.ffn_down.weight"], | ||||
| 					"ffn_gate.weight":    tensors["blk.0.ffn_gate.weight"], | ||||
| 					"ffn_up.weight":      tensors["blk.0.ffn_up.weight"], | ||||
| 					"ffn_norm.weight":    tensors["blk.0.ffn_norm.weight"], | ||||
| 				}, | ||||
| 				"token_embd":  {"weight": tensors["token_embd.weight"]}, | ||||
| 				"output_norm": {"weight": tensors["output_norm.weight"]}, | ||||
| 				"mm.0": { | ||||
| 					"bias":   tensors["mm.0.bias"], | ||||
| 					"weight": tensors["mm.0.weight"], | ||||
| 				}, | ||||
| 				"v.blk.0": { | ||||
| 					"attn_k.weight":      tensors["v.blk.0.attn_k.weight"], | ||||
| 					"attn_q.weight":      tensors["v.blk.0.attn_q.weight"], | ||||
| 					"attn_v.weight":      tensors["v.blk.0.attn_v.weight"], | ||||
| 					"attn_output.weight": tensors["v.blk.0.attn_output.weight"], | ||||
| 					"attn_norm.weight":   tensors["v.blk.0.attn_norm.weight"], | ||||
| 					"ffn_down.weight":    tensors["v.blk.0.ffn_down.weight"], | ||||
| 					"ffn_gate.weight":    tensors["v.blk.0.ffn_gate.weight"], | ||||
| 					"ffn_up.weight":      tensors["v.blk.0.ffn_up.weight"], | ||||
| 					"ffn_norm.weight":    tensors["v.blk.0.ffn_norm.weight"], | ||||
| 				}, | ||||
| 				"v": { | ||||
| 					"patch_embd.weight":    tensors["v.patch_embd.weight"], | ||||
| 					"position_embd.gate":   tensors["v.position_embd.gate"], | ||||
| 					"position_embd.weight": tensors["v.position_embd.weight"], | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range cases { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			got := Tensors{items: tt.items}.GroupLayers() | ||||
| 			if diff := cmp.Diff(got, tt.want); diff != "" { | ||||
| 				t.Errorf("unexpected layers (-got +want):\n%s", diff) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // ref: https://github.com/ggml-org/llama.cpp/blob/a82c9e7c23ef6db48cebfa194dc9cebbc4ac3552/ggml/src/ggml.c#L572 | ||||
| func TestTensorTypes(t *testing.T) { | ||||
| 	cases := []struct { | ||||
| 		kind      uint32 | ||||
| 		blockSize uint64 | ||||
| 		typeSize  uint64 | ||||
| 	}{ | ||||
| 		{0, 1, 4}, | ||||
| 		{1, 1, 2}, | ||||
| 		{2, 32, 18}, | ||||
| 		{3, 32, 20}, | ||||
| 		{6, 32, 22}, | ||||
| 		{7, 32, 24}, | ||||
| 		{8, 32, 34}, | ||||
| 		{9, 32, 36}, | ||||
| 		{10, 256, 84}, | ||||
| 		{11, 256, 110}, | ||||
| 		{12, 256, 144}, | ||||
| 		{13, 256, 176}, | ||||
| 		{14, 256, 210}, | ||||
| 		{15, 256, 292}, | ||||
| 		{16, 256, 66}, | ||||
| 		{17, 256, 74}, | ||||
| 		{18, 256, 98}, | ||||
| 		{19, 256, 50}, | ||||
| 		{20, 32, 18}, | ||||
| 		{21, 256, 110}, | ||||
| 		{22, 256, 82}, | ||||
| 		{23, 256, 136}, | ||||
| 		{24, 1, 1}, | ||||
| 		{25, 1, 2}, | ||||
| 		{26, 1, 4}, | ||||
| 		{27, 1, 8}, | ||||
| 		{28, 1, 8}, | ||||
| 		{29, 256, 56}, | ||||
| 		{30, 1, 2}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range cases { | ||||
| 		t.Run(strconv.Itoa(int(tt.kind)), func(t *testing.T) { | ||||
| 			tensor := Tensor{Kind: tt.kind} | ||||
| 			if tensor.blockSize() != tt.blockSize { | ||||
| 				t.Errorf("unexpected block size: got=%d want=%d", tensor.blockSize(), tt.blockSize) | ||||
| 			} | ||||
|  | ||||
| 			if tensor.typeSize() != tt.typeSize { | ||||
| 				t.Errorf("unexpected type size: got=%d want=%d", tensor.typeSize(), tt.typeSize) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -1,4 +1,4 @@ | ||||
| package llm | ||||
| package ggml | ||||
| 
 | ||||
| import ( | ||||
| 	"bytes" | ||||
| @@ -8,10 +8,9 @@ import ( | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"log/slog" | ||||
| 	"maps" | ||||
| 	"slices" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"golang.org/x/exp/maps" | ||||
| ) | ||||
| 
 | ||||
| type containerGGUF struct { | ||||
| @@ -110,9 +109,9 @@ func (llm *gguf) KV() KV { | ||||
| 	return llm.kv | ||||
| } | ||||
| 
 | ||||
| func (llm *gguf) Tensors() *Tensors { | ||||
| 	return &Tensors{ | ||||
| 		Items:  llm.tensors, | ||||
| func (llm *gguf) Tensors() Tensors { | ||||
| 	return Tensors{ | ||||
| 		items:  llm.tensors, | ||||
| 		Offset: llm.tensorOffset, | ||||
| 	} | ||||
| } | ||||
| @@ -523,7 +522,7 @@ func WriteGGUF(ws io.WriteSeeker, kv KV, ts []Tensor) error { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	keys := maps.Keys(kv) | ||||
| 	keys := slices.Collect(maps.Keys(kv)) | ||||
| 	slices.Sort(keys) | ||||
| 
 | ||||
| 	for _, key := range keys { | ||||
| @@ -1,4 +1,4 @@ | ||||
| package llm | ||||
| package ggml | ||||
| 
 | ||||
| import "fmt" | ||||
| 
 | ||||
| @@ -98,10 +98,10 @@ func ParseFileType(s string) (fileType, error) { | ||||
| 		return fileTypeIQ3_M, nil | ||||
| 	case "IQ2_S": | ||||
| 		return fileTypeIQ2_S, nil | ||||
| 	case "IQ4_XS": | ||||
| 		return fileTypeIQ4_XS, nil | ||||
| 	case "IQ2_M": | ||||
| 		return fileTypeIQ2_M, nil | ||||
| 	case "IQ4_XS": | ||||
| 		return fileTypeIQ4_XS, nil | ||||
| 	case "IQ1_M": | ||||
| 		return fileTypeIQ1_M, nil | ||||
| 	case "BF16": | ||||
							
								
								
									
										18
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| module github.com/ollama/ollama | ||||
|  | ||||
| go 1.23.4 | ||||
| go 1.24.0 | ||||
|  | ||||
| require ( | ||||
| 	github.com/containerd/console v1.0.3 | ||||
| @@ -11,18 +11,20 @@ require ( | ||||
| 	github.com/spf13/cobra v1.7.0 | ||||
| 	github.com/stretchr/testify v1.9.0 | ||||
| 	github.com/x448/float16 v0.8.4 | ||||
| 	golang.org/x/sync v0.10.0 | ||||
| 	golang.org/x/sync v0.11.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/agnivade/levenshtein v1.1.1 | ||||
| 	github.com/d4l3k/go-bfloat16 v0.0.0-20211005043715-690c3bdd05f1 | ||||
| 	github.com/dlclark/regexp2 v1.11.4 | ||||
| 	github.com/emirpasic/gods/v2 v2.0.0-alpha | ||||
| 	github.com/google/go-cmp v0.6.0 | ||||
| 	github.com/mattn/go-runewidth v0.0.14 | ||||
| 	github.com/nlpodyssey/gopickle v0.3.0 | ||||
| 	github.com/pdevine/tensor v0.0.0-20240510204454-f88f4562727c | ||||
| 	golang.org/x/image v0.22.0 | ||||
| 	golang.org/x/tools v0.30.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| @@ -68,12 +70,12 @@ require ( | ||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.12 // indirect | ||||
| 	golang.org/x/arch v0.8.0 // indirect | ||||
| 	golang.org/x/crypto v0.31.0 | ||||
| 	golang.org/x/exp v0.0.0-20231110203233-9a3e6036ecaa | ||||
| 	golang.org/x/net v0.25.0 // indirect | ||||
| 	golang.org/x/sys v0.28.0 | ||||
| 	golang.org/x/term v0.27.0 | ||||
| 	golang.org/x/text v0.21.0 | ||||
| 	golang.org/x/crypto v0.33.0 | ||||
| 	golang.org/x/exp v0.0.0-20250218142911-aa4b98e5adaa | ||||
| 	golang.org/x/net v0.35.0 // indirect | ||||
| 	golang.org/x/sys v0.30.0 | ||||
| 	golang.org/x/term v0.29.0 | ||||
| 	golang.org/x/text v0.22.0 | ||||
| 	google.golang.org/protobuf v1.34.1 | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										32
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								go.sum
									
									
									
									
									
								
							| @@ -42,6 +42,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/trifles v0.0.0-20200323201526-dd97f9abfb48 h1:fRzb/w+pyskVMQ+UbP35JkH8yB7MYb4q/qhBarqZE6g= | ||||
| github.com/dgryski/trifles v0.0.0-20200323201526-dd97f9abfb48/go.mod h1:if7Fbed8SFyPtHLHbg49SI7NAdJiC5WIA09pe59rfAA= | ||||
| github.com/dlclark/regexp2 v1.11.4 h1:rPYF9/LECdNymJufQKmri9gV604RvvABwgOA8un7yAo= | ||||
| github.com/dlclark/regexp2 v1.11.4/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= | ||||
| github.com/emirpasic/gods/v2 v2.0.0-alpha h1:dwFlh8pBg1VMOXWGipNMRt8v96dKAIvBehtCt6OtunU= | ||||
| github.com/emirpasic/gods/v2 v2.0.0-alpha/go.mod h1:W0y4M2dtBB9U5z3YlghmpuUhiaZT2h6yoeE+C1sCp6A= | ||||
| github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| @@ -212,16 +214,16 @@ golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACk | ||||
| golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.31.0 h1:ihbySMvVjLAeSH1IbfcRTkD/iNscyz8rGzjF/E5hV6U= | ||||
| golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= | ||||
| golang.org/x/crypto v0.33.0 h1:IOBPskki6Lysi0lo9qQvbxiQ+FvsCC/YWOecCHAixus= | ||||
| golang.org/x/crypto v0.33.0/go.mod h1:bVdXmD7IV/4GdElGPozy6U7lWdRXA4qyRVGJV57uQ5M= | ||||
| golang.org/x/exp v0.0.0-20180321215751-8460e604b9de/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20180807140117-3d87b88a115f/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190125153040-c74c464bbbf2/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20191002040644-a1355ae1e2c3/go.mod h1:NOZ3BPKG0ec/BKJQgnvsSFpcKLM5xXVWnvZS97DWHgE= | ||||
| golang.org/x/exp v0.0.0-20231110203233-9a3e6036ecaa h1:FRnLl4eNAQl8hwxVVC17teOw8kdjVDVAiFMtgUdTSRQ= | ||||
| golang.org/x/exp v0.0.0-20231110203233-9a3e6036ecaa/go.mod h1:zk2irFbV9DP96SEBUUAy67IdHUaZuSnrz1n472HUCLE= | ||||
| golang.org/x/exp v0.0.0-20250218142911-aa4b98e5adaa h1:t2QcU6V556bFjYgu4L6C+6VrCPyJZ+eyRsABUPs1mz4= | ||||
| golang.org/x/exp v0.0.0-20250218142911-aa4b98e5adaa/go.mod h1:BHOTPb3L19zxehTsLoJXVaTktb06DFgmdW6Wb9s8jqk= | ||||
| golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= | ||||
| golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= | ||||
| golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| @@ -255,8 +257,8 @@ golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81R | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| golang.org/x/net v0.0.0-20210614182718-04defd469f4e/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.25.0 h1:d/OCCoBEUq33pjydKrGQhw7IlUPI2Oylr+8qLx49kac= | ||||
| golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= | ||||
| golang.org/x/net v0.35.0 h1:T5GQRQb2y08kTAByq9L4/bz8cipCdA8FbRTXewonqY8= | ||||
| golang.org/x/net v0.35.0/go.mod h1:EglIi67kWsHKlRzzVMUD93VMSWGFOMSZgxFjparz1Qk= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| @@ -266,8 +268,8 @@ golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJ | ||||
| golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= | ||||
| golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= | ||||
| golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -283,17 +285,17 @@ golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBc | ||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.28.0 h1:Fksou7UEQUWlKvIdsqzJmUmCX3cZuD2+P3XyyzwMhlA= | ||||
| golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= | ||||
| golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.27.0 h1:WP60Sv1nlK1T6SupCHbXzSaN0b9wUmsPoRS9b61A23Q= | ||||
| golang.org/x/term v0.27.0/go.mod h1:iMsnZpn0cago0GOrHO2+Y7u7JPn5AylBrcoWkElMTSM= | ||||
| golang.org/x/term v0.29.0 h1:L6pJp37ocefwRRtYPKSWOWzOtWSxVajvz2ldH/xi3iU= | ||||
| golang.org/x/term v0.29.0/go.mod h1:6bl4lRlvVuDgSf3179VpIxBF0o10JUpXWOnI7nErv7s= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo= | ||||
| golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ= | ||||
| golang.org/x/text v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM= | ||||
| golang.org/x/text v0.22.0/go.mod h1:YRoo4H8PVmsu+E3Ou7cqLVH8oXWIHVoX0jqUWALQhfY= | ||||
| golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| @@ -307,6 +309,8 @@ golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapK | ||||
| golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= | ||||
| golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= | ||||
| golang.org/x/tools v0.30.0 h1:BgcpHewrV5AUp2G9MebG4XPFI1E2W41zU1SaqVA9vJY= | ||||
| golang.org/x/tools v0.30.0/go.mod h1:c347cR/OJfw5TI+GfX7RUPNMdDRRbjvYTS0jPyvsVtY= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
|   | ||||
| @@ -22,7 +22,7 @@ func TestOrcaMiniBlueSky(t *testing.T) { | ||||
| 		Model:  "orca-mini", | ||||
| 		Prompt: "why is the sky blue?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 		}, | ||||
| @@ -39,7 +39,7 @@ func TestUnicode(t *testing.T) { | ||||
| 		Model:  "deepseek-coder-v2:16b-lite-instruct-q2_K", | ||||
| 		Prompt: "天空为什么是蓝色的?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 			// Workaround deepseek context shifting bug | ||||
| @@ -61,7 +61,7 @@ func TestExtendedUnicodeOutput(t *testing.T) { | ||||
| 		Model:  "gemma2:2b", | ||||
| 		Prompt: "Output some smily face emoji", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 		}, | ||||
| @@ -96,7 +96,7 @@ func TestUnicodeModelDir(t *testing.T) { | ||||
| 		Model:  "orca-mini", | ||||
| 		Prompt: "why is the sky blue?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 		}, | ||||
|   | ||||
| @@ -25,7 +25,7 @@ func TestMultiModelConcurrency(t *testing.T) { | ||||
| 				Prompt:    "why is the ocean blue?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
| @@ -34,7 +34,7 @@ func TestMultiModelConcurrency(t *testing.T) { | ||||
| 				Prompt:    "what is the origin of the us thanksgiving holiday?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
|   | ||||
| @@ -23,7 +23,7 @@ func TestLongInputContext(t *testing.T) { | ||||
| 		Model:  "llama2", | ||||
| 		Prompt: "Oh, don’t speak to me of Austria. Perhaps I don’t understand things, but Austria never has wished, and does not wish, for war. She is betraying us! Russia alone must save Europe. Our gracious sovereign recognizes his high vocation and will be true to it. That is the one thing I have faith in! Our good and wonderful sovereign has to perform the noblest role on earth, and he is so virtuous and noble that God will not forsake him. He will fulfill his vocation and crush the hydra of revolution, which has become more terrible than ever in the person of this murderer and villain! We alone must avenge the blood of the just one.... Whom, I ask you, can we rely on?... England with her commercial spirit will not and cannot understand the Emperor Alexander’s loftiness of soul. She has refused to evacuate Malta. She wanted to find, and still seeks, some secret motive in our actions. What answer did Novosíltsev get? None. The English have not understood and cannot understand the self-abnegation of our Emperor who wants nothing for himself, but only desires the good of mankind. And what have they promised? Nothing! And what little they have promised they will not perform! Prussia has always declared that Buonaparte is invincible, and that all Europe is powerless before him.... And I don’t believe a word that Hardenburg says, or Haugwitz either. This famous Prussian neutrality is just a trap. I have faith only in God and the lofty destiny of our adored monarch. He will save Europe! What country is this referring to?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 			"num_ctx":     128, | ||||
| @@ -50,7 +50,7 @@ func TestContextExhaustion(t *testing.T) { | ||||
| 		Model:  "llama2", | ||||
| 		Prompt: "Write me a story with a ton of emojis?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"temperature": 0, | ||||
| 			"seed":        123, | ||||
| 			"num_ctx":     128, | ||||
|   | ||||
| @@ -19,7 +19,7 @@ func TestIntegrationLlava(t *testing.T) { | ||||
| 		Model:  "llava:7b", | ||||
| 		Prompt: "what does the text in this image say?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"seed":        42, | ||||
| 			"temperature": 0.0, | ||||
| 		}, | ||||
| @@ -47,7 +47,7 @@ func TestIntegrationMllama(t *testing.T) { | ||||
| 		Model:  "x/llama3.2-vision", | ||||
| 		Prompt: "what does the text in this image say?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"seed":        42, | ||||
| 			"temperature": 0.0, | ||||
| 		}, | ||||
| @@ -66,6 +66,35 @@ func TestIntegrationMllama(t *testing.T) { | ||||
| 	DoGenerate(ctx, t, client, req, []string{resp}, 240*time.Second, 30*time.Second) | ||||
| } | ||||
|  | ||||
| func TestIntegrationSplitBatch(t *testing.T) { | ||||
| 	image, err := base64.StdEncoding.DecodeString(imageEncoding) | ||||
| 	require.NoError(t, err) | ||||
| 	req := api.GenerateRequest{ | ||||
| 		Model: "gemma3:4b", | ||||
| 		// Fill up a chunk of the batch so the image will partially spill over into the next one | ||||
| 		System: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed aliquet, justo in malesuada lobortis, odio ligula volutpat quam, quis faucibus ipsum magna quis sapien. Aliquam in venenatis diam, eu viverra magna. Phasellus imperdiet hendrerit volutpat. Vivamus sem ex, facilisis placerat felis non, dictum elementum est. Phasellus aliquam imperdiet lacus, eget placerat ligula sodales vel. Pellentesque nec auctor mi. Curabitur arcu nisi, faucibus eget nunc id, viverra interdum mi. Curabitur ornare ipsum ex, ac euismod ex aliquam in. Vestibulum id magna at purus accumsan fermentum. Proin scelerisque posuere nunc quis interdum. Maecenas sed mollis nisl. Etiam vitae ipsum interdum, placerat est quis, tincidunt velit. Nullam tempor nibh non lorem volutpat efficitur. Cras laoreet diam imperdiet ipsum auctor bibendum. Suspendisse ultrices urna sed metus sagittis suscipit. Quisque ullamcorper aliquam nibh ut mollis. Aenean dapibus mauris pharetra, venenatis elit ac, hendrerit odio. Cras vestibulum erat tempor, lobortis justo eu, lobortis ipsum. Nam laoreet dapibus sem. Proin vel diam ultrices, elementum ante et, ornare lectus. Proin eu accumsan nisl. Praesent ac ex vitae ipsum vulputate tristique facilisis sit amet lacus. Nullam faucibus magna a pellentesque pretium. Nunc lacinia ullamcorper sollicitudin. Donec vitae accumsan turpis, sed porttitor est. Donec porttitor mi vitae augue faucibus, vel mollis diam tincidunt.", | ||||
| 		Prompt: "what does the text in this image say?", | ||||
| 		Stream: &stream, | ||||
| 		Options: map[string]any{ | ||||
| 			"seed":        42, | ||||
| 			"temperature": 0.0, | ||||
| 		}, | ||||
| 		Images: []api.ImageData{ | ||||
| 			image, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	// Note: sometimes it returns "the ollamas" sometimes "the ollams" | ||||
| 	resp := "the ollam" | ||||
| 	ctx, cancel := context.WithTimeout(context.Background(), 3*time.Minute) | ||||
| 	defer cancel() | ||||
| 	client, _, cleanup := InitServerConnection(ctx, t) | ||||
| 	defer cleanup() | ||||
| 	require.NoError(t, PullIfMissing(ctx, client, req.Model)) | ||||
| 	// llava models on CPU can be quite slow to start, | ||||
| 	DoGenerate(ctx, t, client, req, []string{resp}, 120*time.Second, 30*time.Second) | ||||
| } | ||||
|  | ||||
| const imageEncoding = `iVBORw0KGgoAAAANSUhEUgAAANIAAAB4CAYAAACHHqzKAAAAAXNSR0IArs4c6QAAAIRlWElmTU0AKgAAAAgABQESAAMAAAABAAEAAAEaAAUAAAABAAAASgEb | ||||
| AAUAAAABAAAAUgEoAAMAAAABAAIAAIdpAAQAAAABAAAAWgAAAAAAAABIAAAAAQAAAEgAAAABAAOgAQADAAAAAQABAACgAgAEAAAAAQAAANKgAwAEAAAAAQAA | ||||
| AHgAAAAAXdsepgAAAAlwSFlzAAALEwAACxMBAJqcGAAAAVlpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6 | ||||
|   | ||||
| @@ -20,7 +20,7 @@ var ( | ||||
| 			Model:  "orca-mini", | ||||
| 			Prompt: "why is the ocean blue?", | ||||
| 			Stream: &stream, | ||||
| 			Options: map[string]interface{}{ | ||||
| 			Options: map[string]any{ | ||||
| 				"seed":        42, | ||||
| 				"temperature": 0.0, | ||||
| 			}, | ||||
| @@ -28,7 +28,7 @@ var ( | ||||
| 			Model:  "orca-mini", | ||||
| 			Prompt: "what is the origin of the us thanksgiving holiday?", | ||||
| 			Stream: &stream, | ||||
| 			Options: map[string]interface{}{ | ||||
| 			Options: map[string]any{ | ||||
| 				"seed":        42, | ||||
| 				"temperature": 0.0, | ||||
| 			}, | ||||
|   | ||||
| @@ -32,7 +32,7 @@ func TestMaxQueue(t *testing.T) { | ||||
| 	req := api.GenerateRequest{ | ||||
| 		Model:  "orca-mini", | ||||
| 		Prompt: "write a long historical fiction story about christopher columbus.  use at least 10 facts from his actual journey", | ||||
| 		Options: map[string]interface{}{ | ||||
| 		Options: map[string]any{ | ||||
| 			"seed":        42, | ||||
| 			"temperature": 0.0, | ||||
| 		}, | ||||
| @@ -52,8 +52,8 @@ func TestMaxQueue(t *testing.T) { | ||||
| 	embedCtx := ctx | ||||
|  | ||||
| 	var genwg sync.WaitGroup | ||||
| 	genwg.Add(1) | ||||
| 	go func() { | ||||
| 		genwg.Add(1) | ||||
| 		defer genwg.Done() | ||||
| 		slog.Info("Starting generate request") | ||||
| 		DoGenerate(ctx, t, client, req, resp, 45*time.Second, 5*time.Second) | ||||
| @@ -71,8 +71,8 @@ func TestMaxQueue(t *testing.T) { | ||||
| 	counterMu := sync.Mutex{} | ||||
| 	var embedwg sync.WaitGroup | ||||
| 	for i := 0; i < threadCount; i++ { | ||||
| 		embedwg.Add(1) | ||||
| 		go func(i int) { | ||||
| 			embedwg.Add(1) | ||||
| 			defer embedwg.Done() | ||||
| 			slog.Info("embed started", "id", i) | ||||
| 			embedReq := api.EmbeddingRequest{ | ||||
|   | ||||
| @@ -291,7 +291,7 @@ func GenerateRequests() ([]api.GenerateRequest, [][]string) { | ||||
| 				Prompt:    "why is the ocean blue?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
| @@ -300,7 +300,7 @@ func GenerateRequests() ([]api.GenerateRequest, [][]string) { | ||||
| 				Prompt:    "why is the color of dirt brown?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
| @@ -309,7 +309,7 @@ func GenerateRequests() ([]api.GenerateRequest, [][]string) { | ||||
| 				Prompt:    "what is the origin of the us thanksgiving holiday?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
| @@ -318,7 +318,7 @@ func GenerateRequests() ([]api.GenerateRequest, [][]string) { | ||||
| 				Prompt:    "what is the origin of independence day?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
| @@ -327,7 +327,7 @@ func GenerateRequests() ([]api.GenerateRequest, [][]string) { | ||||
| 				Prompt:    "what is the composition of air?", | ||||
| 				Stream:    &stream, | ||||
| 				KeepAlive: &api.Duration{Duration: 10 * time.Second}, | ||||
| 				Options: map[string]interface{}{ | ||||
| 				Options: map[string]any{ | ||||
| 					"seed":        42, | ||||
| 					"temperature": 0.0, | ||||
| 				}, | ||||
|   | ||||
							
								
								
									
										77
									
								
								kvcache/cache.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								kvcache/cache.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| package kvcache | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
|  | ||||
| 	"github.com/ollama/ollama/ml" | ||||
| 	"github.com/ollama/ollama/model/input" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	ErrKvCacheFull  = errors.New("could not find a kv cache slot") | ||||
| 	ErrNotSupported = errors.New("model does not support operation") | ||||
| ) | ||||
|  | ||||
| type Cache interface { | ||||
| 	// ** used by model implementations ** | ||||
|  | ||||
| 	// SetLayer sets the active layer of the cache | ||||
| 	SetLayer(layer int) | ||||
|  | ||||
| 	// Get returns the history of key and value tensors plus a mask | ||||
| 	// | ||||
| 	// The shape of the tensors is documented in the specific | ||||
| 	// cache implementation used. | ||||
| 	Get(ctx ml.Context) (ml.Tensor, ml.Tensor, ml.Tensor) | ||||
|  | ||||
| 	// Put stores a batch of key and value in the cache | ||||
| 	// | ||||
| 	// The shape of the tensors is documented in the specific | ||||
| 	// cache implementation used. | ||||
| 	Put(ctx ml.Context, key, value ml.Tensor) | ||||
|  | ||||
| 	// SetConfig controls optimizations (mostly backend-specific) that may transform | ||||
| 	// the output of the cache to work better with specific kernels. If not called, | ||||
| 	// the backend settings will be used. This works well when calling Attention. | ||||
| 	// | ||||
| 	// The config can be overridden by models, especially if they require vanilla | ||||
| 	// output when implementing their own version of attention. To do this, pass | ||||
| 	// an empty ml.CacheConfig. | ||||
| 	// | ||||
| 	// Most models will not need to use this. | ||||
| 	SetConfig(ml.CacheConfig) | ||||
|  | ||||
| 	// ** cache management ** | ||||
|  | ||||
| 	// Init sets up runtime parameters. | ||||
| 	// backend: Used to allocate cache data storage and execute management operations (such as defrag) | ||||
| 	// dtype: The data type for storing cache entries | ||||
| 	// maxSequences: The maximum number of sequences stored in the cache - across all batches | ||||
| 	// capacity: The number of cache entries to store, per sequence | ||||
| 	// maxBatch: The maximum number of tokens that can occur in a single batch | ||||
| 	Init(backend ml.Backend, dtype ml.DType, maxSequences, capacity, maxBatch int) | ||||
|  | ||||
| 	// Close closes the cache and frees resources associated with it | ||||
| 	Close() | ||||
|  | ||||
| 	// StartForward is called before the start of the model's forward pass. | ||||
| 	// For each token in the coming batch, there must be a corresponding | ||||
| 	// entry in positions and seqs. reserve is to preallocate memory | ||||
| 	// without actually storing data in the cache. | ||||
| 	StartForward(ctx ml.Context, batch input.Batch, reserve bool) error | ||||
|  | ||||
| 	// CopyPrefix copies tokens in the range [0, len) from srcSeq to dstSeq | ||||
| 	CopyPrefix(srcSeq, dstSeq int, len int32) | ||||
|  | ||||
| 	// CanResume returns true if the cache can continue with the next token at | ||||
| 	// the given position and sequence. Assumes that the caller has already | ||||
| 	// verified the contents of the cache. | ||||
| 	CanResume(seq int, pos int32) bool | ||||
|  | ||||
| 	// Remove deletes tokens in the range [beginIndex, endIndex) from seq. Set | ||||
| 	// endIndex to math.MaxInt32 to remove everything starting at beginIndex. | ||||
| 	// | ||||
| 	// If an error occurs, the entire context for the sequence should be | ||||
| 	// removed by calling Remove(seq, 0, math.MaxInt32) | ||||
| 	Remove(seq int, beginIndex, endIndex int32) error | ||||
| } | ||||
							
								
								
									
										726
									
								
								kvcache/causal.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										726
									
								
								kvcache/causal.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,726 @@ | ||||
| package kvcache | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"log/slog" | ||||
| 	"math" | ||||
| 	"slices" | ||||
|  | ||||
| 	"github.com/ollama/ollama/ml" | ||||
| 	"github.com/ollama/ollama/model/input" | ||||
| ) | ||||
|  | ||||
| type shiftFn func(ctx ml.Context, layer int, key, shift ml.Tensor) (ml.Tensor, error) | ||||
|  | ||||
| // Causal cache stores K and V tensors according to their position in the | ||||
| // sequence. Returns the history and a mask for attending to past tokens | ||||
| // | ||||
| // The tensors are of shape embed dim, kv heads, batch size | ||||
| // The mask is of shape history size, batch size | ||||
| type Causal struct { | ||||
| 	DType      ml.DType | ||||
| 	windowSize int32 | ||||
|  | ||||
| 	opts CausalOptions | ||||
|  | ||||
| 	// config controls mostly backend-specific optimizations | ||||
| 	config *ml.CacheConfig | ||||
|  | ||||
| 	// ** current forward pass ** | ||||
|  | ||||
| 	// the active layer for Get and Put | ||||
| 	curLayer int | ||||
|  | ||||
| 	// starting location for data storage for this batch | ||||
| 	curLoc int | ||||
|  | ||||
| 	// size of the current batch | ||||
| 	curBatchSize int | ||||
|  | ||||
| 	// mask of the cache as used by this batch | ||||
| 	curMask ml.Tensor | ||||
|  | ||||
| 	// locations in the cache that are needed for this batch | ||||
| 	curCellRange cellRange | ||||
|  | ||||
| 	// curSequences is the sequences corresponding to this pass's entries in the cache | ||||
| 	curSequences []int | ||||
|  | ||||
| 	// curPositions is the positions corresponding to this pass's entries in the cache | ||||
| 	curPositions []int32 | ||||
|  | ||||
| 	// ** cache metadata ** | ||||
|  | ||||
| 	// for each possible location in the cache, stores the position and set of sequences | ||||
| 	// that reference the data there | ||||
| 	cells []cacheCell | ||||
|  | ||||
| 	// maps from sequence to the range of locations where it is stored in the cache | ||||
| 	cellRanges map[int]cellRange | ||||
|  | ||||
| 	// ** cache data storage ** | ||||
|  | ||||
| 	shiftFn      shiftFn | ||||
| 	backend      ml.Backend | ||||
| 	ctxs         map[int]ml.Context | ||||
| 	keys, values map[int]ml.Tensor | ||||
| } | ||||
|  | ||||
| type cacheCell struct { | ||||
| 	pos       int32 | ||||
| 	sequences []int | ||||
| } | ||||
|  | ||||
| type cellRange struct { | ||||
| 	min int | ||||
| 	max int | ||||
| } | ||||
|  | ||||
| func NewCausalCache(shift shiftFn) *Causal { | ||||
| 	return &Causal{ | ||||
| 		windowSize: math.MaxInt32, | ||||
| 		shiftFn:    shift, | ||||
| 		ctxs:       make(map[int]ml.Context), | ||||
| 		keys:       make(map[int]ml.Tensor), | ||||
| 		values:     make(map[int]ml.Tensor), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func NewSWACache(windowSize int32, shift shiftFn) *Causal { | ||||
| 	return &Causal{ | ||||
| 		windowSize: windowSize, | ||||
| 		shiftFn:    shift, | ||||
| 		ctxs:       make(map[int]ml.Context), | ||||
| 		keys:       make(map[int]ml.Tensor), | ||||
| 		values:     make(map[int]ml.Tensor), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) Init(backend ml.Backend, dtype ml.DType, maxSequences, capacity, maxBatch int) { | ||||
| 	if c.config == nil { | ||||
| 		var config ml.CacheConfig | ||||
| 		if cc, ok := backend.(ml.BackendCacheConfig); ok { | ||||
| 			config = cc.CacheConfig() | ||||
| 		} | ||||
| 		c.config = &config | ||||
| 	} | ||||
|  | ||||
| 	if c.config.CachePadding == 0 { | ||||
| 		c.config.CachePadding = 1 | ||||
| 	} | ||||
|  | ||||
| 	if c.config.MaskBatchPadding == 0 { | ||||
| 		c.config.MaskBatchPadding = 1 | ||||
| 	} | ||||
|  | ||||
| 	if c.config.MaskDType == ml.DTypeOther { | ||||
| 		c.config.MaskDType = ml.DTypeF32 | ||||
| 	} | ||||
|  | ||||
| 	var cacheSize int | ||||
| 	if c.windowSize == math.MaxInt32 || capacity < int(c.windowSize) { | ||||
| 		cacheSize = maxSequences * capacity | ||||
| 	} else { | ||||
| 		cacheSize = (maxSequences * int(c.windowSize)) + maxBatch | ||||
| 	} | ||||
| 	cacheSize = roundUp(cacheSize, c.config.CachePadding) | ||||
| 	c.cells = make([]cacheCell, cacheSize) | ||||
|  | ||||
| 	c.DType = dtype | ||||
| 	c.cellRanges = make(map[int]cellRange) | ||||
| 	c.backend = backend | ||||
| } | ||||
|  | ||||
| func (c *Causal) SetConfig(config ml.CacheConfig) { | ||||
| 	if c.config != nil { | ||||
| 		panic("config cannot be changed after being previously set, either by the model or backend") | ||||
| 	} | ||||
|  | ||||
| 	c.config = &config | ||||
| } | ||||
|  | ||||
| func (c *Causal) Close() { | ||||
| 	for _, ctx := range c.ctxs { | ||||
| 		ctx.Close() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) StartForward(ctx ml.Context, batch input.Batch, reserve bool) error { | ||||
| 	c.curBatchSize = len(batch.Positions) | ||||
| 	c.curSequences = batch.Sequences | ||||
| 	c.curPositions = batch.Positions | ||||
| 	c.opts.Except = nil | ||||
|  | ||||
| 	if !reserve { | ||||
| 		c.updateSlidingWindow() | ||||
|  | ||||
| 		var err error | ||||
| 		c.curLoc, err = c.findStartLoc() | ||||
| 		if errors.Is(err, ErrKvCacheFull) { | ||||
| 			c.defrag() | ||||
| 			c.curLoc, err = c.findStartLoc() | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		c.curCellRange = newRange() | ||||
| 		for i, pos := range batch.Positions { | ||||
| 			seq := batch.Sequences[i] | ||||
|  | ||||
| 			c.cells[c.curLoc+i] = cacheCell{pos: pos, sequences: []int{seq}} | ||||
|  | ||||
| 			seqRange, ok := c.cellRanges[seq] | ||||
| 			if !ok { | ||||
| 				seqRange = newRange() | ||||
| 			} | ||||
|  | ||||
| 			if c.curLoc+i > seqRange.max { | ||||
| 				seqRange.max = c.curLoc + i | ||||
| 			} | ||||
| 			if seqRange.max > c.curCellRange.max { | ||||
| 				c.curCellRange.max = seqRange.max | ||||
| 			} | ||||
|  | ||||
| 			if c.curLoc+i < seqRange.min { | ||||
| 				seqRange.min = c.curLoc + i | ||||
| 			} | ||||
| 			if seqRange.min < c.curCellRange.min { | ||||
| 				c.curCellRange.min = seqRange.min | ||||
| 			} | ||||
| 			c.cellRanges[seq] = seqRange | ||||
| 		} | ||||
| 	} else { | ||||
| 		// If we are reserving memory, don't update any of the cache metadata but set the size | ||||
| 		// to the worst case. | ||||
| 		c.curLoc = 0 | ||||
| 		c.curCellRange.min = 0 | ||||
| 		c.curCellRange.max = len(c.cells) - 1 | ||||
| 	} | ||||
|  | ||||
| 	var err error | ||||
| 	c.curMask, err = c.buildMask(ctx) | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func newRange() cellRange { | ||||
| 	return cellRange{ | ||||
| 		min: math.MaxInt, | ||||
| 		max: 0, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Find the first contiguous block of at least curBatchSize | ||||
| func (c *Causal) findStartLoc() (int, error) { | ||||
| 	var start, count int | ||||
| 	for i := range c.cells { | ||||
| 		if len(c.cells[i].sequences) == 0 { | ||||
| 			count++ | ||||
| 			if count >= c.curBatchSize { | ||||
| 				return start, nil | ||||
| 			} | ||||
| 		} else { | ||||
| 			start = i + 1 | ||||
| 			count = 0 | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return 0, fmt.Errorf("%w (length: %v)", ErrKvCacheFull, len(c.cells)) | ||||
| } | ||||
|  | ||||
| func (c *Causal) updateSlidingWindow() { | ||||
| 	if c.windowSize == math.MaxInt32 { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	// create a map of unique sequences to the lowest position in that sequence | ||||
| 	lowestPos := make(map[int]int32) | ||||
| 	for i := range c.curPositions { | ||||
| 		seq := c.curSequences[i] | ||||
|  | ||||
| 		pos, ok := lowestPos[seq] | ||||
| 		if !ok { | ||||
| 			pos = c.curPositions[i] | ||||
| 		} else if c.curPositions[i] < pos { | ||||
| 			pos = c.curPositions[i] | ||||
| 		} | ||||
|  | ||||
| 		lowestPos[seq] = pos | ||||
| 	} | ||||
|  | ||||
| 	// delete any entries that are beyond the window of the oldest position in the sequence | ||||
| 	for seq, pos := range lowestPos { | ||||
| 		oldRange, ok := c.cellRanges[seq] | ||||
| 		if !ok { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		newRange := newRange() | ||||
|  | ||||
| 		for i := oldRange.min; i <= oldRange.max; i++ { | ||||
| 			if slices.Contains(c.cells[i].sequences, seq) { | ||||
| 				if c.cells[i].pos < pos-c.windowSize { | ||||
| 					c.cells[i].sequences = slices.DeleteFunc(c.cells[i].sequences, func(s int) bool { return s == seq }) | ||||
| 				} else { | ||||
| 					newRange.min = min(newRange.min, i) | ||||
| 					newRange.max = max(newRange.max, i) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		c.cellRanges[seq] = newRange | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func roundDown(length, pad int) int { | ||||
| 	return (length / pad) * pad | ||||
| } | ||||
|  | ||||
| func roundUp(length, pad int) int { | ||||
| 	return ((length + pad - 1) / pad) * pad | ||||
| } | ||||
|  | ||||
| // Builds a mask of history x batch indicating whether for each token in the batch the | ||||
| // token in the history should apply. This is based on both the sequence and causality (the | ||||
| // position of the history is not ahead of the token in the batch). | ||||
| func (c *Causal) buildMask(ctx ml.Context) (ml.Tensor, error) { | ||||
| 	// Align and pad the two dimensions as required by the backend | ||||
| 	batchSize := roundUp(c.curBatchSize, c.config.MaskBatchPadding) | ||||
|  | ||||
| 	c.curCellRange.min = roundDown(c.curCellRange.min, c.config.CachePadding) | ||||
| 	c.curCellRange.max = roundUp(c.curCellRange.max+1, c.config.CachePadding) - 1 | ||||
|  | ||||
| 	length := c.curCellRange.max - c.curCellRange.min + 1 | ||||
| 	mask := make([]float32, batchSize*length) | ||||
|  | ||||
| 	for i := range c.curBatchSize { | ||||
| 		enabled := !slices.Contains(c.opts.Except, i) | ||||
| 		for j := c.curCellRange.min; j <= c.curCellRange.max; j++ { | ||||
| 			if !slices.Contains(c.cells[j].sequences, c.curSequences[i]) || | ||||
| 				(enabled && c.cells[j].pos > c.curPositions[i]) || | ||||
| 				c.cells[j].pos < c.curPositions[i]-c.windowSize { | ||||
| 				mask[i*length+(j-c.curCellRange.min)] = float32(math.Inf(-1)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Mask out any padding tokens we added. For padding that we added to the cache history, this | ||||
| 	// has already been masked out because the sequence doesn't match. | ||||
| 	for i := c.curBatchSize * length; i < len(mask); i++ { | ||||
| 		mask[i] = float32(math.Inf(-1)) | ||||
| 	} | ||||
|  | ||||
| 	maskTensor, err := ctx.Input().FromFloatSlice(mask, length, batchSize) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	if c.config.MaskDType != ml.DTypeF32 { | ||||
| 		out := ctx.Input().Empty(c.config.MaskDType, maskTensor.Shape()...) | ||||
| 		ctx.Forward(maskTensor.Copy(ctx, out)) | ||||
| 		maskTensor = out | ||||
| 	} | ||||
|  | ||||
| 	return maskTensor, nil | ||||
| } | ||||
|  | ||||
| func (c *Causal) moveCells(ctx ml.Context, src, dst, length int) { | ||||
| 	for i, key := range c.keys { | ||||
| 		if key == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		kHeadDim := key.Dim(0) | ||||
| 		numKVHeads := key.Dim(1) | ||||
| 		rowSize := key.Stride(2) | ||||
|  | ||||
| 		kSrcView := key.View(ctx, rowSize*src, kHeadDim*numKVHeads*length) | ||||
| 		kDstView := key.View(ctx, rowSize*dst, kHeadDim*numKVHeads*length) | ||||
|  | ||||
| 		value := c.values[i] | ||||
| 		var vSrcView, vDstView ml.Tensor | ||||
| 		if c.config.PermutedV { | ||||
| 			vHeadDim := value.Dim(1) | ||||
| 			elemSize := value.Stride(0) | ||||
|  | ||||
| 			vSrcView = value.View(ctx, elemSize*src, length, len(c.cells)*elemSize, vHeadDim*numKVHeads) | ||||
| 			vDstView = value.View(ctx, elemSize*dst, length, len(c.cells)*elemSize, vHeadDim*numKVHeads) | ||||
| 		} else { | ||||
| 			vHeadDim := value.Dim(0) | ||||
| 			rowSize := value.Stride(2) | ||||
|  | ||||
| 			vSrcView = value.View(ctx, rowSize*src, vHeadDim*numKVHeads*length) | ||||
| 			vDstView = value.View(ctx, rowSize*dst, vHeadDim*numKVHeads*length) | ||||
| 		} | ||||
|  | ||||
| 		ctx.Forward( | ||||
| 			kSrcView.Copy(ctx, kDstView), | ||||
| 			vSrcView.Copy(ctx, vDstView), | ||||
| 		) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) defrag() { | ||||
| 	slog.Debug("defragmenting kv cache") | ||||
|  | ||||
| 	// Defrag strategy: | ||||
| 	// - Search for empty holes at the beginning of the cache, | ||||
| 	//   filling them with active data starting at the end | ||||
| 	// - If there are contiguous elements that need to be moved, | ||||
| 	//   combine them into a single operation by holding new moves | ||||
| 	//   until we see that the next one is non-contiguous | ||||
| 	// - Fill up the context with the maximum number of operations it | ||||
| 	//   can hold then compute that and continue with a new context | ||||
| 	// | ||||
| 	// We could try to optimize placement by grouping blocks from | ||||
| 	// the same sequences together but most likely the next forward | ||||
| 	// pass will disrupt this anyways, so the real world benefit | ||||
| 	// seems limited as this time. | ||||
|  | ||||
| 	ctx := c.backend.NewContext() | ||||
|  | ||||
| 	// For every move, 6 tensors are required per layer (2 views and a | ||||
| 	// copy for each of k and v). We also need to refer to the original | ||||
| 	// k and v cache tensors - once per layer, not per move. | ||||
| 	layers := 0 | ||||
| 	for _, key := range c.keys { | ||||
| 		if key == nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		layers++ | ||||
| 	} | ||||
|  | ||||
| 	maxMoves := (ctx.MaxGraphNodes() - 2*layers) / (6 * layers) | ||||
| 	moves := 0 | ||||
|  | ||||
| 	var pendingSrc, pendingDst, pendingLen int | ||||
| 	src := len(c.cells) - 1 | ||||
|  | ||||
| 	for dst := 0; dst < src; dst++ { | ||||
| 		if len(c.cells[dst].sequences) == 0 { | ||||
| 			for ; src > dst; src-- { | ||||
| 				if len(c.cells[src].sequences) != 0 { | ||||
| 					c.cells[dst] = c.cells[src] | ||||
| 					c.cells[src] = cacheCell{} | ||||
|  | ||||
| 					if pendingLen > 0 { | ||||
| 						if src == pendingSrc-pendingLen && dst == pendingDst+pendingLen { | ||||
| 							pendingSrc = src | ||||
| 							pendingLen++ | ||||
| 							break | ||||
| 						} else { | ||||
| 							c.moveCells(ctx, pendingSrc, pendingDst, pendingLen) | ||||
| 							moves++ | ||||
| 						} | ||||
| 					} | ||||
|  | ||||
| 					pendingSrc = src | ||||
| 					pendingDst = dst | ||||
| 					pendingLen = 1 | ||||
|  | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if moves >= maxMoves { | ||||
| 			ctx.Compute() | ||||
| 			ctx.Close() | ||||
| 			ctx = c.backend.NewContext() | ||||
|  | ||||
| 			moves = 0 | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if pendingLen > 0 { | ||||
| 		c.moveCells(ctx, pendingSrc, pendingDst, pendingLen) | ||||
| 		moves++ | ||||
| 	} | ||||
|  | ||||
| 	if moves > 0 { | ||||
| 		ctx.Compute() | ||||
| 	} | ||||
| 	ctx.Close() | ||||
|  | ||||
| 	// Reset range metadata | ||||
| 	for seq := range c.cellRanges { | ||||
| 		seqRange := newRange() | ||||
|  | ||||
| 		for i, cell := range c.cells { | ||||
| 			if slices.Contains(cell.sequences, seq) { | ||||
| 				if i < seqRange.min { | ||||
| 					seqRange.min = i | ||||
| 				} | ||||
| 				if i > seqRange.max { | ||||
| 					seqRange.max = i | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		c.cellRanges[seq] = seqRange | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) SetLayer(layer int) { | ||||
| 	c.curLayer = layer | ||||
| } | ||||
|  | ||||
| type CausalOptions struct { | ||||
| 	// Enabled controls whether the causal mask is generated for a particular index in a batch | ||||
| 	Except []int | ||||
| } | ||||
|  | ||||
| // SetCausal disables causal mask generation for a particular range of indicies in | ||||
| // the current batch for subsequent calls to Get. The state resets for the next forward pass. | ||||
| func (c *Causal) SetCausal(ctx ml.Context, opts CausalOptions) { | ||||
| 	if !slices.Equal(c.opts.Except, opts.Except) { | ||||
| 		c.opts = opts | ||||
| 		if ctx != nil { | ||||
| 			var err error | ||||
| 			c.curMask, err = c.buildMask(ctx) | ||||
| 			if err != nil { | ||||
| 				// This error should never occur because we have previously built a mask with the same shape | ||||
| 				panic(fmt.Errorf("SetCausal: %w", err)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) Get(ctx ml.Context) (ml.Tensor, ml.Tensor, ml.Tensor) { | ||||
| 	key := c.keys[c.curLayer] | ||||
| 	value := c.values[c.curLayer] | ||||
|  | ||||
| 	kHeadDim := key.Dim(0) | ||||
| 	numKVHeads := key.Dim(1) | ||||
| 	rowSize := key.Stride(2) | ||||
| 	cachedSize := c.curMask.Dim(0) | ||||
|  | ||||
| 	key = key.View(ctx, rowSize*c.curCellRange.min, | ||||
| 		kHeadDim, key.Stride(1), | ||||
| 		numKVHeads, key.Stride(2), | ||||
| 		cachedSize, | ||||
| 	) | ||||
|  | ||||
| 	if c.config.PermutedV { | ||||
| 		vHeadDim := value.Dim(1) | ||||
| 		elemSize := value.Stride(0) | ||||
|  | ||||
| 		value = value.View(ctx, elemSize*c.curCellRange.min, | ||||
| 			cachedSize, value.Stride(1), | ||||
| 			vHeadDim, value.Stride(2), | ||||
| 			numKVHeads, | ||||
| 		) | ||||
| 	} else { | ||||
| 		vHeadDim := value.Dim(0) | ||||
| 		rowSize := value.Stride(2) | ||||
|  | ||||
| 		value = value.View(ctx, rowSize*c.curCellRange.min, | ||||
| 			vHeadDim, value.Stride(1), | ||||
| 			numKVHeads, value.Stride(2), | ||||
| 			cachedSize, | ||||
| 		) | ||||
| 	} | ||||
|  | ||||
| 	return key, value, c.curMask | ||||
| } | ||||
|  | ||||
| func (c *Causal) Put(ctx ml.Context, key, value ml.Tensor) { | ||||
| 	kHeadDim := key.Dim(0) | ||||
| 	vHeadDim := value.Dim(0) | ||||
| 	numKVHeads := key.Dim(1) | ||||
| 	batchSize := key.Dim(2) | ||||
|  | ||||
| 	if c.curBatchSize != batchSize { | ||||
| 		panic(fmt.Errorf("inconsistent batch sizes (layer: %v, batch size: %v layer batch size: %v)", c.curLayer, c.curBatchSize, batchSize)) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.ctxs[c.curLayer]; !ok { | ||||
| 		c.ctxs[c.curLayer] = c.backend.NewContextSize(2).Layer(c.curLayer) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.keys[c.curLayer]; !ok { | ||||
| 		c.keys[c.curLayer] = c.ctxs[c.curLayer].Zeros(c.DType, kHeadDim, numKVHeads, len(c.cells)) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.values[c.curLayer]; !ok { | ||||
| 		if c.config.PermutedV { | ||||
| 			c.values[c.curLayer] = c.ctxs[c.curLayer].Zeros(c.DType, len(c.cells), vHeadDim, numKVHeads) | ||||
| 		} else { | ||||
| 			c.values[c.curLayer] = c.ctxs[c.curLayer].Zeros(c.DType, vHeadDim, numKVHeads, len(c.cells)) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	rowSize := c.keys[c.curLayer].Stride(2) | ||||
| 	ctx.Forward(key.Copy(ctx, c.keys[c.curLayer].View(ctx, rowSize*c.curLoc, kHeadDim*numKVHeads*batchSize))) | ||||
|  | ||||
| 	if c.config.PermutedV { | ||||
| 		elemSize := c.values[c.curLayer].Stride(0) | ||||
|  | ||||
| 		value = value.Permute(ctx, 1, 2, 0, 3) | ||||
| 		ctx.Forward(value.Copy(ctx, c.values[c.curLayer].View(ctx, elemSize*c.curLoc, batchSize, len(c.cells)*elemSize, vHeadDim*numKVHeads))) | ||||
| 	} else { | ||||
| 		rowSize := c.values[c.curLayer].Stride(2) | ||||
|  | ||||
| 		ctx.Forward(value.Copy(ctx, c.values[c.curLayer].View(ctx, rowSize*c.curLoc, vHeadDim*numKVHeads*batchSize))) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Causal) CopyPrefix(srcSeq, dstSeq int, len int32) { | ||||
| 	seqRange := newRange() | ||||
|  | ||||
| 	for i := range c.cells { | ||||
| 		// Remove the contents of dstSeq so that we only have the copied prefix, metadata will be reset at the end | ||||
| 		if slices.Contains(c.cells[i].sequences, dstSeq) { | ||||
| 			c.cells[i].sequences = slices.DeleteFunc(c.cells[i].sequences, func(s int) bool { return s == dstSeq }) | ||||
| 		} | ||||
|  | ||||
| 		if slices.Contains(c.cells[i].sequences, srcSeq) && c.cells[i].pos < len { | ||||
| 			c.cells[i].sequences = append(c.cells[i].sequences, dstSeq) | ||||
| 			if i < seqRange.min { | ||||
| 				seqRange.min = i | ||||
| 			} | ||||
| 			if i > seqRange.max { | ||||
| 				seqRange.max = i | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	c.cellRanges[dstSeq] = seqRange | ||||
| } | ||||
|  | ||||
| func (c *Causal) CanResume(seq int, pos int32) bool { | ||||
| 	if c.windowSize == math.MaxInt32 { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	seqRange, ok := c.cellRanges[seq] | ||||
| 	if !ok { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	// for sliding window, check that the window of the new sequence is contained in | ||||
| 	// the window of what we are storing | ||||
| 	var last int32 = -1 | ||||
| 	for i := seqRange.min; i <= seqRange.max; i++ { | ||||
| 		if slices.Contains(c.cells[i].sequences, seq) { | ||||
| 			last = max(last, c.cells[i].pos) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if last == -1 { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	lastWindowStart := max(0, last-c.windowSize) | ||||
| 	posWindowStart := max(0, pos-c.windowSize) | ||||
|  | ||||
| 	return posWindowStart >= lastWindowStart | ||||
| } | ||||
|  | ||||
| func (c *Causal) shift(seq int, beginIndex, offset int32) error { | ||||
| 	if c.shiftFn == nil { | ||||
| 		return ErrNotSupported | ||||
| 	} | ||||
|  | ||||
| 	ctx := c.backend.NewContext() | ||||
| 	defer ctx.Close() | ||||
|  | ||||
| 	seqRange := c.cellRanges[seq] | ||||
| 	size := seqRange.max - seqRange.min + 1 | ||||
|  | ||||
| 	offsets := make([]int32, size) | ||||
| 	for i := range offsets { | ||||
| 		cell := c.cells[seqRange.min+i] | ||||
|  | ||||
| 		if slices.Contains(cell.sequences, seq) && cell.pos >= beginIndex { | ||||
| 			offsets[i] = offset | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	kShift, err := ctx.Input().FromIntSlice(offsets, len(offsets)) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	for i, key := range c.keys { | ||||
| 		if key == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		kHeadDim := key.Dim(0) | ||||
| 		numKVHeads := key.Dim(1) | ||||
| 		rowSize := key.Stride(2) | ||||
|  | ||||
| 		key = key.View(ctx, rowSize*seqRange.min, | ||||
| 			kHeadDim, key.Stride(1), | ||||
| 			numKVHeads, key.Stride(2), | ||||
| 			size, | ||||
| 		) | ||||
|  | ||||
| 		roped, err := c.shiftFn(ctx, i, key, kShift) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		ctx.Forward(roped.Copy(ctx, key)) | ||||
| 	} | ||||
|  | ||||
| 	ctx.Compute() | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *Causal) Remove(seq int, beginIndex, endIndex int32) error { | ||||
| 	// TODO(jessegross): We should check to see if removing the middle of the sequence will | ||||
| 	// cause the sliding window to encompass tokens that we no longer have. If so, then we | ||||
| 	// should return an error, which will trigger the runner to evaluate the full history and | ||||
| 	// rebuild the window. However, if we have multimodal inputs in our history, this reuse | ||||
| 	// results in use after free, so we don't do it for now. | ||||
|  | ||||
| 	var offset int32 | ||||
| 	if endIndex != math.MaxInt32 { | ||||
| 		offset = beginIndex - endIndex | ||||
| 	} | ||||
|  | ||||
| 	seqRange := newRange() | ||||
|  | ||||
| 	for i := range c.cells { | ||||
| 		if slices.Contains(c.cells[i].sequences, seq) { | ||||
| 			if c.cells[i].pos >= beginIndex && c.cells[i].pos < endIndex { | ||||
| 				c.cells[i].sequences = slices.DeleteFunc(c.cells[i].sequences, func(s int) bool { return s == seq }) | ||||
| 			} else { | ||||
| 				if c.cells[i].pos >= endIndex { | ||||
| 					if slices.ContainsFunc(c.cells[i].sequences, func(s int) bool { return s != seq }) { | ||||
| 						return errors.New("shifting cells shared by multiple sequences not supported") | ||||
| 					} | ||||
|  | ||||
| 					c.cells[i].pos += offset | ||||
| 				} | ||||
| 				if i < seqRange.min { | ||||
| 					seqRange.min = i | ||||
| 				} | ||||
| 				if i > seqRange.max { | ||||
| 					seqRange.max = i | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if seqRange == newRange() { | ||||
| 		delete(c.cellRanges, seq) | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	c.cellRanges[seq] = seqRange | ||||
|  | ||||
| 	if endIndex != math.MaxInt32 { | ||||
| 		err := c.shift(seq, endIndex+offset, offset) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										521
									
								
								kvcache/causal_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										521
									
								
								kvcache/causal_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,521 @@ | ||||
| package kvcache | ||||
|  | ||||
| import ( | ||||
| 	"math" | ||||
| 	"slices" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/ollama/ollama/ml" | ||||
| 	"github.com/ollama/ollama/model/input" | ||||
| ) | ||||
|  | ||||
| type testCase struct { | ||||
| 	name          string | ||||
| 	in            []float32 | ||||
| 	inShape       []int | ||||
| 	seqs          []int | ||||
| 	pos           []int32 | ||||
| 	expected      []float32 | ||||
| 	expectedShape []int | ||||
| 	expectedMask  []float32 | ||||
| } | ||||
|  | ||||
| func TestStore(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewCausalCache(nil) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{111, 211, 121, 221, 131, 231, 112, 212, 122, 222, 132, 232, 113, 213, 123, 223, 133, 233, 114, 214, 124, 224, 134, 234}, | ||||
| 			inShape:       []int{2, 3, 4}, | ||||
| 			seqs:          []int{0, 0, 0, 0}, | ||||
| 			pos:           []int32{0, 1, 2, 3}, | ||||
| 			expected:      []float32{111, 211, 121, 221, 131, 231, 112, 212, 122, 222, 132, 232, 113, 213, 123, 223, 133, 233, 114, 214, 124, 224, 134, 234}, | ||||
| 			expectedShape: []int{2, 3, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, float32(math.Inf(-1)), 0, 0, 0, 0}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:          "SecondBatch", | ||||
| 			in:            []float32{115, 215, 125, 225, 135, 235}, | ||||
| 			inShape:       []int{2, 3, 1}, | ||||
| 			seqs:          []int{0}, | ||||
| 			pos:           []int32{4}, | ||||
| 			expected:      []float32{111, 211, 121, 221, 131, 231, 112, 212, 122, 222, 132, 232, 113, 213, 123, 223, 133, 233, 114, 214, 124, 224, 134, 234, 115, 215, 125, 225, 135, 235}, | ||||
| 			expectedShape: []int{2, 3, 5}, | ||||
| 			expectedMask:  []float32{0, 0, 0, 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func TestSWA(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewSWACache(1, nil) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{1, 2, 3, 4}, | ||||
| 			inShape:       []int{1, 1, 4}, | ||||
| 			seqs:          []int{0, 0, 0, 0}, | ||||
| 			pos:           []int32{0, 1, 2, 3}, | ||||
| 			expected:      []float32{1, 2, 3, 4}, | ||||
| 			expectedShape: []int{1, 1, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:          "SecondBatch", | ||||
| 			in:            []float32{5, 6}, | ||||
| 			inShape:       []int{1, 1, 2}, | ||||
| 			seqs:          []int{0, 0}, | ||||
| 			pos:           []int32{4, 5}, | ||||
| 			expected:      []float32{5, 6, 3, 4}, | ||||
| 			expectedShape: []int{1, 1, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1))}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func TestSequences(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewCausalCache(nil) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{1, 2, 3, 4}, | ||||
| 			inShape:       []int{1, 1, 4}, | ||||
| 			seqs:          []int{0, 0, 1, 1}, | ||||
| 			pos:           []int32{0, 1, 0, 1}, | ||||
| 			expected:      []float32{1, 2, 3, 4}, | ||||
| 			expectedShape: []int{1, 1, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:          "SecondBatch", | ||||
| 			in:            []float32{5, 6}, | ||||
| 			inShape:       []int{1, 1, 2}, | ||||
| 			seqs:          []int{0, 1}, | ||||
| 			pos:           []int32{2, 2}, | ||||
| 			expected:      []float32{1, 2, 3, 4, 5, 6}, | ||||
| 			expectedShape: []int{1, 1, 6}, | ||||
| 			expectedMask:  []float32{0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func TestRemove(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewCausalCache(func(ctx ml.Context, layer int, key, shift ml.Tensor) (ml.Tensor, error) { | ||||
| 		return key.Add(ctx, shift), nil | ||||
| 	}) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{1, 2, 3, 4}, | ||||
| 			inShape:       []int{1, 1, 4}, | ||||
| 			seqs:          []int{0, 0, 1, 1}, | ||||
| 			pos:           []int32{0, 1, 0, 1}, | ||||
| 			expected:      []float32{1, 2, 3, 4}, | ||||
| 			expectedShape: []int{1, 1, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
|  | ||||
| 	err := cache.Remove(0, 1, math.MaxInt32) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	tests = []testCase{ | ||||
| 		{ | ||||
| 			name:          "RemoveEnd", | ||||
| 			in:            []float32{5, 6}, | ||||
| 			inShape:       []int{1, 1, 2}, | ||||
| 			seqs:          []int{0, 1}, | ||||
| 			pos:           []int32{1, 2}, | ||||
| 			expected:      []float32{1, 2, 3, 4, 5, 6}, | ||||
| 			expectedShape: []int{1, 1, 6}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
|  | ||||
| 	err = cache.Remove(0, 0, 1) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	tests = []testCase{ | ||||
| 		{ | ||||
| 			name:          "RemoveMiddle", | ||||
| 			in:            []float32{7, 8}, | ||||
| 			inShape:       []int{1, 1, 2}, | ||||
| 			seqs:          []int{0, 0}, | ||||
| 			pos:           []int32{1, 2}, | ||||
| 			expected:      []float32{7, 8, 3, 4, 4}, | ||||
| 			expectedShape: []int{1, 1, 5}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func TestDefrag(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewCausalCache(func(ctx ml.Context, layer int, key, shift ml.Tensor) (ml.Tensor, error) { | ||||
| 		return key.Add(ctx, shift), nil | ||||
| 	}) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, | ||||
| 			inShape:       []int{1, 1, 16}, | ||||
| 			seqs:          []int{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||
| 			pos:           []int32{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15}, | ||||
| 			expected:      []float32{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, | ||||
| 			expectedShape: []int{1, 1, 16}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
|  | ||||
| 	err := cache.Remove(0, 2, 4) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	err = cache.Remove(0, 13, math.MaxInt32) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	tests = []testCase{ | ||||
| 		{ | ||||
| 			name:          "Defrag", | ||||
| 			in:            []float32{17, 18, 19}, | ||||
| 			inShape:       []int{1, 1, 3}, | ||||
| 			seqs:          []int{0, 0, 0}, | ||||
| 			pos:           []int32{16, 17, 18}, | ||||
| 			expected:      []float32{1, 2, 12, 13, 3, 4, 5, 6, 7, 8, 9, 10, 11, 17, 18, 19}, | ||||
| 			expectedShape: []int{1, 1, 16}, | ||||
| 			expectedMask:  []float32{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, float32(math.Inf(-1)), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func TestCopy(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	cache := NewCausalCache(func(ctx ml.Context, layer int, key, shift ml.Tensor) (ml.Tensor, error) { return key, nil }) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	tests := []testCase{ | ||||
| 		{ | ||||
| 			name:          "FirstBatch", | ||||
| 			in:            []float32{1, 2, 3, 4}, | ||||
| 			inShape:       []int{1, 1, 4}, | ||||
| 			seqs:          []int{0, 0, 0, 0}, | ||||
| 			pos:           []int32{0, 1, 2, 3}, | ||||
| 			expected:      []float32{1, 2, 3, 4}, | ||||
| 			expectedShape: []int{1, 1, 4}, | ||||
| 			expectedMask:  []float32{0, float32(math.Inf(-1)), float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0, 0, float32(math.Inf(-1)), 0, 0, 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
|  | ||||
| 	cache.CopyPrefix(0, 1, 2) | ||||
|  | ||||
| 	tests = []testCase{ | ||||
| 		{ | ||||
| 			name:          "Copy", | ||||
| 			in:            []float32{5, 6}, | ||||
| 			inShape:       []int{1, 1, 2}, | ||||
| 			seqs:          []int{1, 1}, | ||||
| 			pos:           []int32{3, 4}, | ||||
| 			expected:      []float32{1, 2, 3, 4, 5, 6}, | ||||
| 			expectedShape: []int{1, 1, 6}, | ||||
| 			expectedMask:  []float32{0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, float32(math.Inf(-1)), 0, 0, float32(math.Inf(-1)), float32(math.Inf(-1)), 0, 0}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	testCache(t, backend, cache, tests) | ||||
| } | ||||
|  | ||||
| func testCache(t *testing.T, backend ml.Backend, cache Cache, tests []testCase) { | ||||
| 	for _, test := range tests { | ||||
| 		t.Run(test.name, func(t *testing.T) { | ||||
| 			context := backend.NewContext() | ||||
| 			defer context.Close() | ||||
|  | ||||
| 			err := cache.StartForward(context, input.Batch{Positions: test.pos, Sequences: test.seqs}, false) | ||||
| 			if err != nil { | ||||
| 				panic(err) | ||||
| 			} | ||||
|  | ||||
| 			cache.SetLayer(0) | ||||
| 			tensor, _ := context.FromFloatSlice(test.in, test.inShape...) | ||||
| 			cache.Put(context, tensor, tensor) | ||||
|  | ||||
| 			out, _, mask := cache.Get(context) | ||||
|  | ||||
| 			context.Forward(out, mask).Compute(out, mask) | ||||
|  | ||||
| 			if !slices.Equal(out.Floats(), test.expected) || !slices.Equal(out.Shape(), test.expectedShape) || !slices.Equal(mask.Floats(), test.expectedMask) { | ||||
| 				t.Errorf("TestCache: have %v (shape %v); want %v (shape %v); mask: have %v (shape %v) want %v", out.Floats(), out.Shape(), test.expected, test.expectedShape, mask.Floats(), mask.Shape(), test.expectedMask) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCanResume(t *testing.T) { | ||||
| 	backend := &testBackend{} | ||||
| 	windowSize := int32(4) | ||||
| 	cache := NewSWACache(windowSize, nil) | ||||
| 	defer cache.Close() | ||||
|  | ||||
| 	cache.Init(backend, ml.DTypeF16, 1, 16, 16) | ||||
|  | ||||
| 	context := backend.NewContext() | ||||
| 	defer context.Close() | ||||
|  | ||||
| 	err := cache.StartForward(context, input.Batch{ | ||||
| 		Positions: []int32{0, 1, 2, 3}, | ||||
| 		Sequences: []int{0, 0, 0, 0}, | ||||
| 	}, false) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("StartForward failed: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	cache.SetLayer(0) | ||||
| 	tensor, _ := context.FromFloatSlice([]float32{1, 2, 3, 4}, 1, 1, 4) | ||||
| 	cache.Put(context, tensor, tensor) | ||||
|  | ||||
| 	// with window size 4, nothing has slid out of the window yet | ||||
| 	if !cache.CanResume(0, 0) { | ||||
| 		t.Errorf("CanResume(0, 0) = false, want true (within window)") | ||||
| 	} | ||||
| 	if !cache.CanResume(0, 1) { | ||||
| 		t.Errorf("CanResume(0, 1) = false, want true (within window)") | ||||
| 	} | ||||
| 	if !cache.CanResume(0, 2) { | ||||
| 		t.Errorf("CanResume(0, 2) = false, want true (within window)") | ||||
| 	} | ||||
| 	if !cache.CanResume(0, 3) { | ||||
| 		t.Errorf("CanResume(0, 3) = false, want true (latest position)") | ||||
| 	} | ||||
|  | ||||
| 	// shift window by adding position 4 | ||||
| 	err = cache.StartForward(context, input.Batch{ | ||||
| 		Positions: []int32{4, 5}, | ||||
| 		Sequences: []int{0, 0}, | ||||
| 	}, false) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("StartForward failed: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	cache.SetLayer(0) | ||||
| 	tensor, _ = context.FromFloatSlice([]float32{5, 6}, 1, 1, 2) | ||||
| 	cache.Put(context, tensor, tensor) | ||||
|  | ||||
| 	// only the latest position has overlapping windows | ||||
| 	if cache.CanResume(0, 0) { | ||||
| 		t.Errorf("after shift: CanResume(0, 0) = true, want false (outside window)") | ||||
| 	} | ||||
| 	if cache.CanResume(0, 1) { | ||||
| 		t.Errorf("after shift: CanResume(0, 1) = true, want false (outside window)") | ||||
| 	} | ||||
| 	if cache.CanResume(0, 2) { | ||||
| 		t.Errorf("after shift: CanResume(0, 2) = true, want false (outside window)") | ||||
| 	} | ||||
| 	if cache.CanResume(0, 3) { | ||||
| 		t.Errorf("after shift: CanResume(0, 3) = true, want false (outside window)") | ||||
| 	} | ||||
| 	if cache.CanResume(0, 4) { | ||||
| 		t.Errorf("after shift: CanResume(0, 4) = true, want false (outside window)") | ||||
| 	} | ||||
| 	if !cache.CanResume(0, 5) { | ||||
| 		t.Errorf("after shift: CanResume(0, 5) = false, want true (latest position)") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type testBackend struct { | ||||
| 	ml.Backend | ||||
| } | ||||
|  | ||||
| func (b *testBackend) NewContext() ml.Context { | ||||
| 	return &testContext{} | ||||
| } | ||||
|  | ||||
| func (b *testBackend) NewContextSize(int) ml.Context { | ||||
| 	return &testContext{} | ||||
| } | ||||
|  | ||||
| type testContext struct { | ||||
| 	ml.Context | ||||
| } | ||||
|  | ||||
| func (c *testContext) Empty(dtype ml.DType, shape ...int) ml.Tensor { | ||||
| 	total := 0 | ||||
|  | ||||
| 	if len(shape) > 0 { | ||||
| 		total = 1 | ||||
| 		for _, s := range shape { | ||||
| 			total *= s | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &testTensor{dtype: dtype, elementSize: 4, data: make([]float32, total), shape: shape} | ||||
| } | ||||
|  | ||||
| func (c *testContext) Zeros(dtype ml.DType, shape ...int) ml.Tensor { | ||||
| 	return c.Empty(dtype, shape...) | ||||
| } | ||||
|  | ||||
| func (c *testContext) FromFloatSlice(s []float32, shape ...int) (ml.Tensor, error) { | ||||
| 	t := c.Empty(ml.DTypeF32, shape...).(*testTensor) | ||||
|  | ||||
| 	copy(t.data, s) | ||||
|  | ||||
| 	return t, nil | ||||
| } | ||||
|  | ||||
| func (c *testContext) FromIntSlice(s []int32, shape ...int) (ml.Tensor, error) { | ||||
| 	f := make([]float32, len(s)) | ||||
| 	for i := range f { | ||||
| 		f[i] = float32(s[i]) | ||||
| 	} | ||||
|  | ||||
| 	out, _ := c.FromFloatSlice(f, shape...) | ||||
| 	out.(*testTensor).dtype = ml.DTypeI32 | ||||
|  | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *testContext) Input() ml.Context    { return c } | ||||
| func (c *testContext) Layer(int) ml.Context { return c } | ||||
|  | ||||
| func (c *testContext) Forward(...ml.Tensor) ml.Context { return c } | ||||
|  | ||||
| func (c *testContext) Compute(...ml.Tensor) {} | ||||
|  | ||||
| func (c *testContext) Reserve() error { return nil } | ||||
|  | ||||
| func (c *testContext) MaxGraphNodes() int { | ||||
| 	return 10 | ||||
| } | ||||
|  | ||||
| func (c *testContext) Close() {} | ||||
|  | ||||
| type testTensor struct { | ||||
| 	ml.Tensor | ||||
|  | ||||
| 	dtype       ml.DType | ||||
| 	elementSize int | ||||
| 	data        []float32 | ||||
| 	shape       []int | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Dim(n int) int { | ||||
| 	return t.shape[n] | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Stride(n int) int { | ||||
| 	stride := t.elementSize | ||||
| 	for i := range n { | ||||
| 		stride *= t.shape[i] | ||||
| 	} | ||||
|  | ||||
| 	return stride | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Shape() []int { | ||||
| 	return t.shape | ||||
| } | ||||
|  | ||||
| func (t *testTensor) DType() ml.DType { | ||||
| 	return t.dtype | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Floats() []float32 { | ||||
| 	out := make([]float32, len(t.data)) | ||||
| 	copy(out, t.data) | ||||
| 	return out | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Neg(ctx ml.Context) ml.Tensor { | ||||
| 	out := ctx.Empty(t.DType(), t.Shape()...).(*testTensor) | ||||
| 	for i := range out.data { | ||||
| 		out.data[i] = -t.data[i] | ||||
| 	} | ||||
| 	return out | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Add(ctx ml.Context, t2 ml.Tensor) ml.Tensor { | ||||
| 	out := ctx.Empty(t.DType(), t.Shape()...).(*testTensor) | ||||
|  | ||||
| 	for i := range out.data { | ||||
| 		out.data[i] = t.data[i] + t2.(*testTensor).data[i] | ||||
| 	} | ||||
|  | ||||
| 	return out | ||||
| } | ||||
|  | ||||
| func (t *testTensor) View(ctx ml.Context, offset int, shape ...int) ml.Tensor { | ||||
| 	offset /= t.elementSize | ||||
|  | ||||
| 	var s []int | ||||
|  | ||||
| 	switch len(shape) { | ||||
| 	case 1: | ||||
| 		s = []int{shape[0]} | ||||
| 	case 5: | ||||
| 		s = []int{shape[0], shape[2], shape[4]} | ||||
| 	default: | ||||
| 		panic("unsupported number of dimensions") | ||||
| 	} | ||||
|  | ||||
| 	context := &testContext{} | ||||
|  | ||||
| 	view := context.Empty(t.dtype, s...).(*testTensor) | ||||
| 	view.data = t.data[offset : offset+len(view.data)] | ||||
|  | ||||
| 	return view | ||||
| } | ||||
|  | ||||
| func (t *testTensor) Copy(ctx ml.Context, t2 ml.Tensor) ml.Tensor { | ||||
| 	copy(t2.(*testTensor).data, t.data) | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										156
									
								
								kvcache/encoder.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								kvcache/encoder.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| package kvcache | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"github.com/ollama/ollama/ml" | ||||
| 	"github.com/ollama/ollama/model/input" | ||||
| ) | ||||
|  | ||||
| // Encoder cache stores K and V tensors that are position independent | ||||
| // | ||||
| // The tensors can be of any shape and will be returned as they were stored | ||||
| // The mask is currently always nil | ||||
| // | ||||
| // Not currently safe for multiple sequences | ||||
| type EncoderCache struct { | ||||
| 	// config controls mostly backend-specific optimizations | ||||
| 	config *ml.CacheConfig | ||||
|  | ||||
| 	// ** current forward pass ** | ||||
|  | ||||
| 	// the active layer for Get and Put | ||||
| 	curLayer int | ||||
|  | ||||
| 	// if something is stored during this pass, this | ||||
| 	// will be the position (but there is no guarantee | ||||
| 	// anything will be stored) | ||||
| 	curPos int32 | ||||
|  | ||||
| 	// curReserve indicates that this forward pass is only for | ||||
| 	// memory reservation and we should not update our metadata | ||||
| 	// based on it. | ||||
| 	curReserve bool | ||||
|  | ||||
| 	// ** cache metadata ** | ||||
|  | ||||
| 	// was something stored in the cache? | ||||
| 	encoderCached bool | ||||
|  | ||||
| 	// position of the cached data | ||||
| 	encoderPos int32 | ||||
|  | ||||
| 	// ** cache data storage ** | ||||
| 	backend      ml.Backend | ||||
| 	ctxs         map[int]ml.Context | ||||
| 	keys, values map[int]ml.Tensor | ||||
| } | ||||
|  | ||||
| func NewEncoderCache() *EncoderCache { | ||||
| 	return &EncoderCache{ | ||||
| 		ctxs:   make(map[int]ml.Context), | ||||
| 		keys:   make(map[int]ml.Tensor), | ||||
| 		values: make(map[int]ml.Tensor), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) Init(backend ml.Backend, dtype ml.DType, maxSequences, capacity, maxBatch int) { | ||||
| 	if c.config == nil { | ||||
| 		var config ml.CacheConfig | ||||
| 		if cc, ok := backend.(ml.BackendCacheConfig); ok { | ||||
| 			config = cc.CacheConfig() | ||||
| 		} | ||||
| 		c.config = &config | ||||
| 	} | ||||
|  | ||||
| 	if maxSequences > 1 { | ||||
| 		panic(fmt.Errorf("encoder cache does not support multiple sequences; requested: %v", maxSequences)) | ||||
| 	} | ||||
|  | ||||
| 	if c.config.CachePadding != 0 && c.config.CachePadding != 1 { | ||||
| 		panic(fmt.Errorf("encoder cache is unable to enforce requested CachePadding (%v)", c.config.CachePadding)) | ||||
| 	} | ||||
|  | ||||
| 	c.backend = backend | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) SetConfig(config ml.CacheConfig) { | ||||
| 	if c.config != nil { | ||||
| 		panic("config cannot be changed after being previously set, either by the model or backend") | ||||
| 	} | ||||
|  | ||||
| 	c.config = &config | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) Close() { | ||||
| 	for _, ctx := range c.ctxs { | ||||
| 		ctx.Close() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) StartForward(ctx ml.Context, batch input.Batch, reserve bool) error { | ||||
| 	// We work with the most recent image | ||||
| 	if len(batch.Multimodal) > 0 { | ||||
| 		c.curPos = batch.Positions[batch.Multimodal[len(batch.Multimodal)-1].Index] | ||||
| 	} | ||||
|  | ||||
| 	c.curReserve = reserve | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) SetLayer(layer int) { | ||||
| 	c.curLayer = layer | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) EncoderCached() bool { | ||||
| 	return c.encoderCached | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) Get(ctx ml.Context) (ml.Tensor, ml.Tensor, ml.Tensor) { | ||||
| 	return c.keys[c.curLayer], c.values[c.curLayer], nil | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) Put(ctx ml.Context, key, value ml.Tensor) { | ||||
| 	if !c.curReserve { | ||||
| 		c.encoderPos = c.curPos | ||||
| 		c.encoderCached = true | ||||
| 	} | ||||
|  | ||||
| 	if c.config.PermutedV { | ||||
| 		value = value.Permute(ctx, 1, 2, 0, 3) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.ctxs[c.curLayer]; !ok { | ||||
| 		c.ctxs[c.curLayer] = c.backend.NewContextSize(2).Layer(c.curLayer) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.keys[c.curLayer]; !ok { | ||||
| 		c.keys[c.curLayer] = c.ctxs[c.curLayer].Empty(key.DType(), key.Shape()...) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := c.values[c.curLayer]; !ok { | ||||
| 		c.values[c.curLayer] = c.ctxs[c.curLayer].Empty(value.DType(), value.Shape()...) | ||||
| 	} | ||||
|  | ||||
| 	ctx.Forward( | ||||
| 		key.Copy(ctx, c.keys[c.curLayer]), | ||||
| 		value.Copy(ctx, c.values[c.curLayer]), | ||||
| 	) | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) CopyPrefix(srcSeq, dstSeq int, len int32) { | ||||
| 	panic("encoder cache does not support multiple sequences") | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) CanResume(seq int, pos int32) bool { | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| func (c *EncoderCache) Remove(seq int, beginIndex, endIndex int32) error { | ||||
| 	if c.encoderPos >= beginIndex && c.encoderPos < endIndex { | ||||
| 		c.encoderCached = false | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										110
									
								
								kvcache/wrapper.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								kvcache/wrapper.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| package kvcache | ||||
|  | ||||
| import ( | ||||
| 	"math" | ||||
|  | ||||
| 	"github.com/ollama/ollama/ml" | ||||
| 	"github.com/ollama/ollama/model/input" | ||||
| ) | ||||
|  | ||||
| // Wrapper cache is a container for multiple types of caches, | ||||
| // such as for the encoding and decoding portions of a model. | ||||
| type WrapperCache struct { | ||||
| 	// caches we are wrapping | ||||
| 	caches []Cache | ||||
|  | ||||
| 	// cache to be used for this layer | ||||
| 	curType int | ||||
| } | ||||
|  | ||||
| func NewWrapperCache(caches ...Cache) *WrapperCache { | ||||
| 	return &WrapperCache{ | ||||
| 		caches: caches, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) Init(backend ml.Backend, dtype ml.DType, maxSequences, capacity, maxBatch int) { | ||||
| 	for _, cache := range c.caches { | ||||
| 		cache.Init(backend, dtype, maxSequences, capacity, maxBatch) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) SetConfig(config ml.CacheConfig) { | ||||
| 	for _, cache := range c.caches { | ||||
| 		cache.SetConfig(config) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) Close() { | ||||
| 	for _, cache := range c.caches { | ||||
| 		cache.Close() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) StartForward(ctx ml.Context, batch input.Batch, reserve bool) error { | ||||
| 	for i, cache := range c.caches { | ||||
| 		err := cache.StartForward(ctx, batch, reserve) | ||||
| 		if err != nil { | ||||
| 			// unwind on error - Remove with endIndex set to math.MaxInt32 does not fail | ||||
| 			for j := i - 1; j >= 0; j-- { | ||||
| 				for k := range batch.Positions { | ||||
| 					_ = c.caches[j].Remove(batch.Sequences[k], batch.Positions[k], math.MaxInt32) | ||||
| 				} | ||||
| 			} | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	c.curType = 0 | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) SetLayer(layer int) { | ||||
| 	for _, cache := range c.caches { | ||||
| 		cache.SetLayer(layer) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) SetLayerType(layerType int) { | ||||
| 	c.curType = layerType | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) UnderlyingCache() Cache { | ||||
| 	return c.caches[c.curType] | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) Get(ctx ml.Context) (ml.Tensor, ml.Tensor, ml.Tensor) { | ||||
| 	return c.caches[c.curType].Get(ctx) | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) Put(ctx ml.Context, key, value ml.Tensor) { | ||||
| 	c.caches[c.curType].Put(ctx, key, value) | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) CopyPrefix(srcSeq, dstSeq int, len int32) { | ||||
| 	for _, cache := range c.caches { | ||||
| 		cache.CopyPrefix(srcSeq, dstSeq, len) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) CanResume(seq int, pos int32) bool { | ||||
| 	for _, cache := range c.caches { | ||||
| 		if !cache.CanResume(seq, pos) { | ||||
| 			return false | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| func (c *WrapperCache) Remove(seq int, beginIndex, endIndex int32) error { | ||||
| 	// If the one of these fails, the caller is supposed to retry with endIndex set to math.MaxInt32, which should not fail | ||||
| 	for _, cache := range c.caches { | ||||
| 		err := cache.Remove(seq, beginIndex, endIndex) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										137
									
								
								llama/README.md
									
									
									
									
									
								
							
							
						
						
									
										137
									
								
								llama/README.md
									
									
									
									
									
								
							| @@ -1,157 +1,52 @@ | ||||
| # `llama` | ||||
|  | ||||
| This package integrates the [llama.cpp](https://github.com/ggerganov/llama.cpp) library as a Go package and makes it easy to build it with tags for different CPU and GPU processors. | ||||
|  | ||||
| Supported: | ||||
|  | ||||
| - [x] CPU | ||||
| - [x] avx, avx2 | ||||
| - [x] macOS Metal | ||||
| - [x] Windows CUDA | ||||
| - [x] Windows ROCm | ||||
| - [x] Linux CUDA | ||||
| - [x] Linux ROCm | ||||
| - [x] Llava | ||||
|  | ||||
| Extra build steps are required for CUDA and ROCm on Windows since `nvcc` and `hipcc` both require using msvc as the host compiler. For these shared libraries are created: | ||||
|  | ||||
| - `ggml_cuda.dll` on Windows or `ggml_cuda.so` on Linux | ||||
| - `ggml_hipblas.dll` on Windows or `ggml_hipblas.so` on Linux | ||||
|  | ||||
| > Note: it's important that memory is allocated and freed by the same compiler (e.g. entirely by code compiled with msvc or mingw). Issues from this should be rare, but there are some places where pointers are returned by the CUDA or HIP runtimes and freed elsewhere, causing a a crash. In a future change the same runtime should be used in both cases to avoid crashes. | ||||
|  | ||||
| ## Building | ||||
|  | ||||
| ``` | ||||
| go build . | ||||
| ``` | ||||
|  | ||||
| ### AVX | ||||
|  | ||||
| ```shell | ||||
| go build -tags avx . | ||||
| ``` | ||||
|  | ||||
| ### AVX2 | ||||
|  | ||||
| ```shell | ||||
| # go doesn't recognize `-mfma` as a valid compiler flag | ||||
| # see https://github.com/golang/go/issues/17895 | ||||
| go env -w "CGO_CFLAGS_ALLOW=-mfma|-mf16c" | ||||
| go env -w "CGO_CXXFLAGS_ALLOW=-mfma|-mf16c" | ||||
| go build -tags=avx,avx2 . | ||||
| ``` | ||||
|  | ||||
| ## Linux | ||||
|  | ||||
| ### CUDA | ||||
|  | ||||
| Install the [CUDA toolkit v11.3.1](https://developer.nvidia.com/cuda-11-3-1-download-archive): | ||||
|  | ||||
| ```shell | ||||
| make ggml_cuda.so | ||||
| go build -tags avx,cuda . | ||||
| ``` | ||||
|  | ||||
| ### ROCm | ||||
|  | ||||
| Install [ROCm](https://rocm.docs.amd.com/en/latest/). | ||||
|  | ||||
| ```shell | ||||
| make ggml_hipblas.so | ||||
| go build -tags avx,rocm . | ||||
| ``` | ||||
|  | ||||
| ## Windows | ||||
|  | ||||
| Download [w64devkit](https://github.com/skeeto/w64devkit/releases/latest) for a simple MinGW development environment. | ||||
|  | ||||
| ### CUDA | ||||
|  | ||||
| Install the [CUDA toolkit v11.3.1](https://developer.nvidia.com/cuda-11-3-1-download-archive) then build the cuda code: | ||||
|  | ||||
| ```shell | ||||
| make ggml_cuda.dll | ||||
| go build -tags avx,cuda . | ||||
| ``` | ||||
|  | ||||
| ### ROCm | ||||
|  | ||||
| Install [ROCm](https://rocm.docs.amd.com/en/latest/). | ||||
|  | ||||
| ```shell | ||||
| make ggml_hipblas.dll | ||||
| go build -tags avx,rocm . | ||||
| ``` | ||||
|  | ||||
| ## Building runners | ||||
|  | ||||
| ```shell | ||||
| # build all runners for this platform | ||||
| make -j | ||||
| ``` | ||||
| This package provides Go bindings to [llama.cpp](https://github.com/ggerganov/llama.cpp). | ||||
|  | ||||
| ## Vendoring | ||||
|  | ||||
| Ollama currently vendors [llama.cpp](https://github.com/ggerganov/llama.cpp/) and [ggml](https://github.com/ggerganov/ggml) through a vendoring model. While we generally strive to contribute changes back upstream to avoid drift, we cary a small set of patches which are applied to the tracking commit. A set of make targets are available to aid developers in updating to a newer tracking commit, or to work on changes. | ||||
| Ollama vendors [llama.cpp](https://github.com/ggerganov/llama.cpp/) and [ggml](https://github.com/ggerganov/llama.cpp/tree/master/ggml/src). While we generally strive to contribute changes back upstream to avoid drift, we carry a small set of patches which are applied to the tracking commit. | ||||
|  | ||||
| If you update the vendoring code, start by running the following command to establish the tracking llama.cpp repo in the `./vendor/` directory. | ||||
|  | ||||
| ``` | ||||
| make apply-patches | ||||
| ```shell | ||||
| make -f Makefile.sync apply-patches | ||||
| ``` | ||||
|  | ||||
| ### Updating Base Commit | ||||
|  | ||||
| **Pin to new base commit** | ||||
|  | ||||
| To update to a newer base commit, select the upstream git tag or commit and update `llama/vendoring` | ||||
|  | ||||
| #### Applying patches | ||||
| To change the base commit, update `FETCH_HEAD` in Makefile.sync. | ||||
|  | ||||
| When updating to a newer base commit, the existing patches may not apply cleanly and require manual merge resolution. | ||||
|  | ||||
| Start by applying the patches. If any of the patches have conflicts, the `git am` will stop at the first failure. | ||||
|  | ||||
| ``` | ||||
| make apply-patches | ||||
| ```shell | ||||
| make -f Makefile.sync apply-patches | ||||
| ``` | ||||
|  | ||||
| If you see an error message about a conflict, go into the `./vendor/` directory, and perform merge resolution using your preferred tool to the patch commit which failed. Save the file(s) and continue the patch series with `git am --continue` . If any additional patches fail, follow the same pattern until the full patch series is applied. Once finished, run a final `create-patches` and `sync` target to ensure everything is updated. | ||||
| If there are conflicts, you will see an error message. Resolve the conflicts in `./vendor/`, and continue the patch series with `git am --continue` and rerun `make -f Makefile.sync apply-patches`. Repeat until all patches are successfully applied. | ||||
|  | ||||
| ``` | ||||
| make create-patches sync | ||||
| ``` | ||||
| Once all patches are applied, commit the changes to the tracking repository. | ||||
|  | ||||
| Build and test Ollama, and make any necessary changes to the Go code based on the new base commit. Submit your PR to the Ollama repo. | ||||
| ```shell | ||||
| make -f Makefile.sync format-patches sync | ||||
| ``` | ||||
|  | ||||
| ### Generating Patches | ||||
|  | ||||
| When working on new fixes or features that impact vendored code, use the following model. First get a clean tracking repo with all current patches applied: | ||||
|  | ||||
| ```shell | ||||
| make -f Makefile.sync clean apply-patches | ||||
| ``` | ||||
| make apply-patches | ||||
| ``` | ||||
|  | ||||
| Now edit the upstream native code in the `./vendor/` directory. You do not need to commit every change in order to build, a dirty working tree in the tracking repo is OK while developing. Simply save in your editor, and run the following to refresh the vendored code with your changes, build the backend(s) and build ollama: | ||||
|  | ||||
| ``` | ||||
| make sync | ||||
| make -j 8 | ||||
| go build . | ||||
| ``` | ||||
|  | ||||
| > [!IMPORTANT] | ||||
| > Do **NOT** run `apply-patches` while you're iterating as that will reset the tracking repo. It will detect a dirty tree and abort, but if your tree is clean and you accidentally ran this target, use `git reflog` to recover your commit(s). | ||||
|  | ||||
| Iterate until you're ready to submit PRs. Once your code is ready, commit a change in the `./vendor/` directory, then generate the patches for ollama with | ||||
|  | ||||
| ```shell | ||||
| make -f Makefile.sync format-patches | ||||
| ``` | ||||
| make create-patches | ||||
| ``` | ||||
|  | ||||
| > [!IMPORTANT] | ||||
| > Once you have completed this step, it is safe to run `apply-patches` since your change is preserved in the patches. | ||||
|  | ||||
| In your `./vendor/` directory, create a branch, and cherry-pick the new commit to that branch, then submit a PR upstream to llama.cpp. | ||||
|  | ||||
|   | ||||
							
								
								
									
										34
									
								
								llama/amx.h
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										34
									
								
								llama/amx.h
									
									
									
									
										vendored
									
									
								
							| @@ -1,34 +0,0 @@ | ||||
| /** | ||||
|  * llama.cpp - commit 46e3556e01b824e52395fb050b29804b6cff2a7c - do not edit this file | ||||
|  * | ||||
|  * MIT License | ||||
|  * | ||||
|  * Copyright (c) 2023-2024 The ggml authors | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  */ | ||||
|  | ||||
| #include "ggml-backend.h" | ||||
| #include "ggml-cpu-impl.h" | ||||
|  | ||||
| // GGML internal header | ||||
|  | ||||
| #if defined(__AMX_INT8__) && defined(__AVX512VNNI__) | ||||
| ggml_backend_buffer_type_t ggml_backend_amx_buffer_type(void); | ||||
| #endif | ||||
							
								
								
									
										2
									
								
								llama/build-info.cpp
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								llama/build-info.cpp
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| int LLAMA_BUILD_NUMBER = 0; | ||||
| char const *LLAMA_COMMIT = "ba1cb19cdd0d92e012e0f6e009e0620f854b6afd"; | ||||
| char const *LLAMA_COMMIT = "d7cfe1ffe0f435d0048a6058d529daf76e072d9c"; | ||||
| char const *LLAMA_COMPILER = ""; | ||||
| char const *LLAMA_BUILD_TARGET = ""; | ||||
|   | ||||
							
								
								
									
										4
									
								
								llama/build-info.cpp.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								llama/build-info.cpp.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| int LLAMA_BUILD_NUMBER = 0; | ||||
| char const *LLAMA_COMMIT = "@FETCH_HEAD@"; | ||||
| char const *LLAMA_COMPILER = ""; | ||||
| char const *LLAMA_BUILD_TARGET = ""; | ||||
							
								
								
									
										51
									
								
								llama/ggml-blas.h
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								llama/ggml-blas.h
									
									
									
									
										vendored
									
									
								
							| @@ -1,51 +0,0 @@ | ||||
| /** | ||||
|  * llama.cpp - commit 46e3556e01b824e52395fb050b29804b6cff2a7c - do not edit this file | ||||
|  * | ||||
|  * MIT License | ||||
|  * | ||||
|  * Copyright (c) 2023-2024 The ggml authors | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "ggml.h" | ||||
| #include "ggml-backend.h" | ||||
|  | ||||
|  | ||||
| #ifdef  __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| // backend API | ||||
| GGML_BACKEND_API ggml_backend_t ggml_backend_blas_init(void); | ||||
|  | ||||
| GGML_BACKEND_API bool ggml_backend_is_blas(ggml_backend_t backend); | ||||
|  | ||||
| // number of threads used for conversion to float | ||||
| // for openblas and blis, this will also set the number of threads used for blas operations | ||||
| GGML_BACKEND_API void ggml_backend_blas_set_n_threads(ggml_backend_t backend_blas, int n_threads); | ||||
|  | ||||
| GGML_BACKEND_API ggml_backend_reg_t ggml_backend_blas_reg(void); | ||||
|  | ||||
|  | ||||
| #ifdef  __cplusplus | ||||
| } | ||||
| #endif | ||||
							
								
								
									
										34
									
								
								llama/ggml-cpu-aarch64.h
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										34
									
								
								llama/ggml-cpu-aarch64.h
									
									
									
									
										vendored
									
									
								
							| @@ -1,34 +0,0 @@ | ||||
| /** | ||||
|  * llama.cpp - commit 46e3556e01b824e52395fb050b29804b6cff2a7c - do not edit this file | ||||
|  * | ||||
|  * MIT License | ||||
|  * | ||||
|  * Copyright (c) 2023-2024 The ggml authors | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "ggml-cpu-traits.h" | ||||
| #include "ggml.h" | ||||
|  | ||||
| // GGML internal header | ||||
|  | ||||
| ggml_backend_buffer_type_t ggml_backend_cpu_aarch64_buffer_type(void); | ||||
							
								
								
									
										64
									
								
								llama/ggml-cpu-traits.h
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										64
									
								
								llama/ggml-cpu-traits.h
									
									
									
									
										vendored
									
									
								
							| @@ -1,64 +0,0 @@ | ||||
| /** | ||||
|  * llama.cpp - commit 46e3556e01b824e52395fb050b29804b6cff2a7c - do not edit this file | ||||
|  * | ||||
|  * MIT License | ||||
|  * | ||||
|  * Copyright (c) 2023-2024 The ggml authors | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
| #include "ggml-backend-impl.h" | ||||
| #include "ggml-cpu-impl.h" | ||||
| #include "ggml.h" | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| #    include <vector> | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| // return true if op part of extra "accelerator" | ||||
| bool ggml_cpu_extra_compute_forward(struct ggml_compute_params * params, struct ggml_tensor * op); | ||||
| bool ggml_cpu_extra_work_size(int n_threads, const struct ggml_tensor * op, size_t * size); | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } | ||||
|  | ||||
| namespace ggml::cpu { | ||||
| // register in tensor->extra | ||||
| class tensor_traits { | ||||
|   public: | ||||
|     virtual ~tensor_traits(); | ||||
|     virtual bool work_size(int n_threads, const struct ggml_tensor * op, size_t & size)        = 0; | ||||
|     virtual bool compute_forward(struct ggml_compute_params * params, struct ggml_tensor * op) = 0; | ||||
| }; | ||||
|  | ||||
| class extra_buffer_type { | ||||
|   public: | ||||
|     virtual ~extra_buffer_type(); | ||||
|     virtual bool            supports_op(ggml_backend_dev_t dev, const struct ggml_tensor * op) = 0; | ||||
|     virtual tensor_traits * get_tensor_traits(const struct ggml_tensor * op)                   = 0; | ||||
| }; | ||||
| }  // namespace ggml::cpu | ||||
|  | ||||
| // implemented in ggml-cpu.cpp. | ||||
| std::vector<ggml_backend_buffer_type_t> & ggml_backend_cpu_get_extra_buffers_type(); | ||||
|  | ||||
| #endif | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user