mirror of
https://github.com/ggerganov/whisper.cpp.git
synced 2025-07-01 15:00:31 +02:00
Compare commits
1 Commits
fix_vs_sdl
...
gg/disable
Author | SHA1 | Date | |
---|---|---|---|
ceb77363cd |
@ -12,7 +12,7 @@ FROM ${BASE_CUDA_DEV_CONTAINER} as build
|
|||||||
ARG CUDA_DOCKER_ARCH=all
|
ARG CUDA_DOCKER_ARCH=all
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
apt-get install -y build-essential git cmake libsdl2-dev wget git
|
apt-get install -y build-essential git cmake
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
@ -23,6 +23,6 @@ ENV CUDA_DOCKER_ARCH=${CUDA_DOCKER_ARCH}
|
|||||||
# Enable cuBLAS
|
# Enable cuBLAS
|
||||||
ENV GGML_CUDA=1
|
ENV GGML_CUDA=1
|
||||||
|
|
||||||
RUN make base.en
|
RUN make
|
||||||
|
|
||||||
ENTRYPOINT ["/app/main"]
|
ENTRYPOINT ["/app/main"]
|
||||||
|
@ -17,7 +17,7 @@ ENV CUDA_DOCKER_ARCH=${CUDA_DOCKER_ARCH}
|
|||||||
ENV GGML_CUDA=1
|
ENV GGML_CUDA=1
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
apt-get install -y build-essential libsdl2-dev wget cmake git \
|
apt-get install -y build-essential \
|
||||||
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
||||||
|
|
||||||
# Ref: https://stackoverflow.com/a/53464012
|
# Ref: https://stackoverflow.com/a/53464012
|
||||||
@ -25,7 +25,7 @@ ENV CUDA_MAIN_VERSION=12.3
|
|||||||
ENV LD_LIBRARY_PATH /usr/local/cuda-${CUDA_MAIN_VERSION}/compat:$LD_LIBRARY_PATH
|
ENV LD_LIBRARY_PATH /usr/local/cuda-${CUDA_MAIN_VERSION}/compat:$LD_LIBRARY_PATH
|
||||||
|
|
||||||
COPY .. .
|
COPY .. .
|
||||||
RUN make base.en
|
RUN make
|
||||||
|
|
||||||
FROM ${BASE_CUDA_RUN_CONTAINER} AS runtime
|
FROM ${BASE_CUDA_RUN_CONTAINER} AS runtime
|
||||||
ENV CUDA_MAIN_VERSION=12.3
|
ENV CUDA_MAIN_VERSION=12.3
|
||||||
@ -33,7 +33,7 @@ ENV LD_LIBRARY_PATH /usr/local/cuda-${CUDA_MAIN_VERSION}/compat:$LD_LIBRARY_PATH
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
apt-get install -y curl ffmpeg wget cmake git \
|
apt-get install -y curl ffmpeg \
|
||||||
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
||||||
|
|
||||||
COPY --from=build /app /app
|
COPY --from=build /app /app
|
||||||
|
@ -2,17 +2,17 @@ FROM ubuntu:22.04 AS build
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
apt-get install -y build-essential wget cmake git \
|
apt-get install -y build-essential \
|
||||||
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
||||||
|
|
||||||
COPY .. .
|
COPY .. .
|
||||||
RUN make base.en
|
RUN make
|
||||||
|
|
||||||
FROM ubuntu:22.04 AS runtime
|
FROM ubuntu:22.04 AS runtime
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
apt-get install -y curl ffmpeg libsdl2-dev wget cmake git \
|
apt-get install -y curl ffmpeg \
|
||||||
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
&& rm -rf /var/lib/apt/lists/* /var/cache/apt/archives/*
|
||||||
|
|
||||||
COPY --from=build /app /app
|
COPY --from=build /app /app
|
||||||
|
10
.github/workflows/bindings-go.yml
vendored
10
.github/workflows/bindings-go.yml
vendored
@ -10,13 +10,13 @@ on:
|
|||||||
- whisper.h
|
- whisper.h
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ubuntu-22:
|
ubuntu-latest:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/setup-go@v5
|
- uses: actions/setup-go@v3
|
||||||
with:
|
with:
|
||||||
go-version: '^1.23'
|
go-version: '^1.19'
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v1
|
||||||
- run: |
|
- run: |
|
||||||
cd bindings/go
|
cd bindings/go
|
||||||
make test
|
make test
|
||||||
|
51
.github/workflows/bindings-ruby.yml
vendored
51
.github/workflows/bindings-ruby.yml
vendored
@ -3,53 +3,20 @@ on:
|
|||||||
push:
|
push:
|
||||||
paths:
|
paths:
|
||||||
- bindings/ruby/**
|
- bindings/ruby/**
|
||||||
- src/**/*.c
|
- whisper.h
|
||||||
- src/**/*.cpp
|
|
||||||
- src/**/*.h
|
|
||||||
- src/**/*.m
|
|
||||||
- src/**/*.metal
|
|
||||||
- include/**/*.c
|
|
||||||
- include/**/*.cpp
|
|
||||||
- include/**/*.h
|
|
||||||
- include/**/*.m
|
|
||||||
- include/**/*.metal
|
|
||||||
- ggml/**/*.c
|
|
||||||
- ggml/**/*.cpp
|
|
||||||
- ggml/**/*.h
|
|
||||||
- ggml/**/*.m
|
|
||||||
- ggml/**/*.metal
|
|
||||||
- scripts/get-flags.mk
|
|
||||||
- examples/dr_wav.h
|
|
||||||
pull_request:
|
pull_request:
|
||||||
paths:
|
paths:
|
||||||
- bindings/ruby/**
|
- bindings/ruby/**
|
||||||
- src/**/*.c
|
- whisper.h
|
||||||
- src/**/*.cpp
|
|
||||||
- src/**/*.h
|
|
||||||
- src/**/*.m
|
|
||||||
- src/**/*.metal
|
|
||||||
- include/**/*.c
|
|
||||||
- include/**/*.cpp
|
|
||||||
- include/**/*.h
|
|
||||||
- include/**/*.m
|
|
||||||
- include/**/*.metal
|
|
||||||
- ggml/**/*.c
|
|
||||||
- ggml/**/*.cpp
|
|
||||||
- ggml/**/*.h
|
|
||||||
- ggml/**/*.m
|
|
||||||
- ggml/**/*.metal
|
|
||||||
- scripts/get-flags.mk
|
|
||||||
- examples/dr_wav.h
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ubuntu-22:
|
ubuntu-latest:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
defaults:
|
|
||||||
run:
|
|
||||||
working-directory: bindings/ruby
|
|
||||||
steps:
|
steps:
|
||||||
- uses: ruby/setup-ruby@v1
|
- uses: ruby/setup-ruby@v1
|
||||||
with:
|
with:
|
||||||
ruby-version: '3.1'
|
ruby-version: '3.0'
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v1
|
||||||
- run: rake test
|
- run: |
|
||||||
|
cd bindings/ruby/ext
|
||||||
|
ruby extconf.rb && make
|
||||||
|
476
.github/workflows/build.yml
vendored
476
.github/workflows/build.yml
vendored
@ -1,28 +1,17 @@
|
|||||||
name: CI
|
name: CI
|
||||||
|
on: [push, pull_request]
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
pull_request:
|
|
||||||
types: [opened, synchronize, reopened]
|
|
||||||
|
|
||||||
concurrency:
|
|
||||||
group: ${{ github.workflow }}-${{ github.head_ref && github.ref || github.run_id }}
|
|
||||||
cancel-in-progress: true
|
|
||||||
|
|
||||||
env:
|
env:
|
||||||
ubuntu_image: "ubuntu:22.04"
|
ubuntu_image: "ubuntu:22.04"
|
||||||
VCPKG_BINARY_SOURCES: "clear;x-gha,readwrite"
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ubuntu-22:
|
ubuntu-latest:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
arch: [linux/amd64, linux/ppc64le]
|
arch: [linux/amd64, linux/arm64, linux/arm/v7, linux/ppc64le]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Clone
|
- name: Clone
|
||||||
@ -38,61 +27,9 @@ jobs:
|
|||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
||||||
set -e
|
set -e
|
||||||
apt update
|
apt update
|
||||||
apt install -y build-essential libsdl2-dev cmake git
|
apt install -y build-essential libsdl2-dev
|
||||||
cmake -B build
|
make
|
||||||
cmake --build build --config Release -j $(nproc)'
|
make stream'
|
||||||
|
|
||||||
ubuntu-22-arm64:
|
|
||||||
runs-on: ubuntu-22.04
|
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
arch: [linux/arm64]
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Clone
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
|
|
||||||
- name: Build ${{ matrix.arch }}
|
|
||||||
run: |
|
|
||||||
docker run --platform ${{ matrix.arch }} --rm \
|
|
||||||
-v ${{ github.workspace }}:/workspace \
|
|
||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
|
||||||
set -e
|
|
||||||
apt update
|
|
||||||
apt install -y build-essential libsdl2-dev cmake git
|
|
||||||
cmake -B build -DGGML_NATIVE=OFF -DGGML_CPU_ARM_ARCH=armv8-a
|
|
||||||
cmake --build build --config Release -j $(nproc)'
|
|
||||||
|
|
||||||
ubuntu-22-arm-v7:
|
|
||||||
runs-on: ubuntu-22.04
|
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
arch: [linux/arm/v7]
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Clone
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
|
|
||||||
- name: Build ${{ matrix.arch }}
|
|
||||||
run: |
|
|
||||||
docker run --platform ${{ matrix.arch }} --rm \
|
|
||||||
-v ${{ github.workspace }}:/workspace \
|
|
||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
|
||||||
set -e
|
|
||||||
apt update
|
|
||||||
apt install -y build-essential libsdl2-dev cmake git
|
|
||||||
cmake -B build -DGGML_NATIVE=OFF -DGGML_CPU_ARM_ARCH=armv7-a+fp
|
|
||||||
cmake --build build --config Release -j $(nproc)'
|
|
||||||
|
|
||||||
macOS-latest:
|
macOS-latest:
|
||||||
runs-on: macOS-latest
|
runs-on: macOS-latest
|
||||||
@ -104,39 +41,39 @@ jobs:
|
|||||||
- name: Dependencies
|
- name: Dependencies
|
||||||
run: |
|
run: |
|
||||||
brew update
|
brew update
|
||||||
brew install sdl2 cmake
|
brew install sdl2
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
cmake -B build
|
make
|
||||||
cmake --build build --config Release
|
make stream
|
||||||
|
|
||||||
# freeBSD-latest:
|
freeBSD-latest:
|
||||||
# runs-on: macos-12
|
runs-on: macos-12
|
||||||
#
|
|
||||||
# steps:
|
|
||||||
# - name: Clone
|
|
||||||
# uses: actions/checkout@v4
|
|
||||||
#
|
|
||||||
# - name: Build
|
|
||||||
# uses: cross-platform-actions/action@v0.24.0
|
|
||||||
# with:
|
|
||||||
# operating_system: freebsd
|
|
||||||
# version: '13.3'
|
|
||||||
# run: |
|
|
||||||
# sudo pkg update
|
|
||||||
# sudo pkg install -y gmake sdl2 cmake
|
|
||||||
# cmake -B build
|
|
||||||
# cmake --build build --config Release
|
|
||||||
|
|
||||||
ubuntu-22-gcc:
|
steps:
|
||||||
runs-on: ubuntu-22.04
|
- name: Clone
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
uses: cross-platform-actions/action@v0.24.0
|
||||||
|
with:
|
||||||
|
operating_system: freebsd
|
||||||
|
version: '13.2'
|
||||||
|
run: |
|
||||||
|
sudo pkg update
|
||||||
|
sudo pkg install -y gmake sdl2
|
||||||
|
gmake
|
||||||
|
gmake stream
|
||||||
|
|
||||||
|
ubuntu-latest-gcc:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
build: [Debug, Release]
|
build: [Debug, Release]
|
||||||
arch: [linux/amd64, linux/ppc64le]
|
arch: [linux/amd64, linux/arm64, linux/arm/v7, linux/ppc64le]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Clone
|
- name: Clone
|
||||||
@ -152,69 +89,13 @@ jobs:
|
|||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
||||||
set -e
|
set -e
|
||||||
apt update
|
apt update
|
||||||
apt install -y build-essential cmake libsdl2-dev git
|
apt install -y build-essential cmake libsdl2-dev
|
||||||
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
||||||
make
|
make
|
||||||
ctest -L gh --output-on-failure'
|
ctest -L gh --output-on-failure'
|
||||||
|
|
||||||
ubuntu-22-gcc-arm64:
|
ubuntu-latest-clang:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
build: [Debug, Release]
|
|
||||||
arch: [linux/arm64]
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Clone
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
|
|
||||||
- name: Build ${{ matrix.arch }}
|
|
||||||
run: |
|
|
||||||
docker run --platform ${{ matrix.arch }} --rm \
|
|
||||||
-v ${{ github.workspace }}:/workspace \
|
|
||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
|
||||||
set -e
|
|
||||||
apt update
|
|
||||||
apt install -y build-essential cmake libsdl2-dev git
|
|
||||||
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }} -DGGML_NATIVE=OFF -DGGML_CPU_ARM_ARCH=armv8-a
|
|
||||||
make
|
|
||||||
ctest -L gh --output-on-failure'
|
|
||||||
|
|
||||||
ubuntu-22-gcc-arm-v7:
|
|
||||||
runs-on: ubuntu-22.04
|
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
build: [Debug, Release]
|
|
||||||
arch: [linux/arm/v7]
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Clone
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
|
|
||||||
- name: Build ${{ matrix.arch }}
|
|
||||||
run: |
|
|
||||||
docker run --platform ${{ matrix.arch }} --rm \
|
|
||||||
-v ${{ github.workspace }}:/workspace \
|
|
||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
|
||||||
set -e
|
|
||||||
apt update
|
|
||||||
apt install -y build-essential cmake libsdl2-dev git
|
|
||||||
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }} -DGGML_NATIVE=OFF -DGGML_CPU_ARM_ARCH=armv7-a+fp
|
|
||||||
make
|
|
||||||
ctest -L gh --output-on-failure'
|
|
||||||
|
|
||||||
ubuntu-22-clang:
|
|
||||||
runs-on: ubuntu-22.04
|
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
@ -239,13 +120,13 @@ jobs:
|
|||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
||||||
set -e
|
set -e
|
||||||
apt update
|
apt update
|
||||||
apt install -y clang build-essential cmake libsdl2-dev git
|
apt install -y clang build-essential cmake libsdl2-dev
|
||||||
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }} -DCMAKE_CXX_COMPILER=clang++ -DCMAKE_C_COMPILER=clang
|
cmake . -DWHISPER_SDL2=ON -DCMAKE_BUILD_TYPE=${{ matrix.build }} -DCMAKE_CXX_COMPILER=clang++ -DCMAKE_C_COMPILER=clang
|
||||||
make
|
make
|
||||||
ctest -L gh --output-on-failure'
|
ctest -L gh --output-on-failure'
|
||||||
|
|
||||||
ubuntu-22-gcc-sanitized:
|
ubuntu-latest-gcc-sanitized:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
@ -267,7 +148,7 @@ jobs:
|
|||||||
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
-w /workspace ${{ env.ubuntu_image }} /bin/sh -c '
|
||||||
set -e
|
set -e
|
||||||
apt update
|
apt update
|
||||||
apt install -y build-essential cmake git
|
apt install -y build-essential cmake
|
||||||
cmake . -DCMAKE_BUILD_TYPE=Debug -DWHISPER_SANITIZE_${{ matrix.sanitizer }}=ON
|
cmake . -DCMAKE_BUILD_TYPE=Debug -DWHISPER_SANITIZE_${{ matrix.sanitizer }}=ON
|
||||||
make
|
make
|
||||||
ctest -L gh --output-on-failure'
|
ctest -L gh --output-on-failure'
|
||||||
@ -302,12 +183,12 @@ jobs:
|
|||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
sudo apt update
|
sudo apt update
|
||||||
sudo apt install intel-oneapi-compiler-dpcpp-cpp git
|
sudo apt install intel-oneapi-compiler-dpcpp-cpp
|
||||||
|
|
||||||
- name: install oneAPI MKL library
|
- name: install oneAPI MKL library
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
sudo apt install intel-oneapi-mkl-devel git
|
sudo apt install intel-oneapi-mkl-devel
|
||||||
|
|
||||||
- name: Clone
|
- name: Clone
|
||||||
id: checkout
|
id: checkout
|
||||||
@ -352,7 +233,7 @@ jobs:
|
|||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
sudo apt update
|
sudo apt update
|
||||||
sudo apt install intel-oneapi-compiler-dpcpp-cpp git
|
sudo apt install intel-oneapi-compiler-dpcpp-cpp
|
||||||
|
|
||||||
- name: install oneAPI MKL library
|
- name: install oneAPI MKL library
|
||||||
shell: bash
|
shell: bash
|
||||||
@ -393,16 +274,30 @@ jobs:
|
|||||||
msystem: ${{matrix.sys}}
|
msystem: ${{matrix.sys}}
|
||||||
install: >-
|
install: >-
|
||||||
base-devel
|
base-devel
|
||||||
git
|
|
||||||
mingw-w64-${{matrix.env}}-toolchain
|
mingw-w64-${{matrix.env}}-toolchain
|
||||||
mingw-w64-${{matrix.env}}-cmake
|
mingw-w64-${{matrix.env}}-cmake
|
||||||
mingw-w64-${{matrix.env}}-SDL2
|
mingw-w64-${{matrix.env}}-SDL2
|
||||||
mingw-w64-${{matrix.env}}-openblas
|
mingw-w64-${{matrix.env}}-openblas
|
||||||
|
|
||||||
|
- name: Build using make
|
||||||
|
shell: msys2 {0}
|
||||||
|
run: |
|
||||||
|
make -j $(nproc)
|
||||||
|
|
||||||
|
- name: Clean after building using make
|
||||||
|
shell: msys2 {0}
|
||||||
|
run: |
|
||||||
|
make clean
|
||||||
|
|
||||||
|
- name: Build using make w/ OpenBLAS
|
||||||
|
shell: msys2 {0}
|
||||||
|
run: |
|
||||||
|
make GGML_OPENBLAS=1 -j $(nproc)
|
||||||
|
|
||||||
- name: Build using CMake
|
- name: Build using CMake
|
||||||
shell: msys2 {0}
|
shell: msys2 {0}
|
||||||
run: |
|
run: |
|
||||||
cmake -B build -DWHISPER_SDL2=ON
|
cmake -B build
|
||||||
cmake --build build --config ${{ matrix.build }} -j $(nproc)
|
cmake --build build --config ${{ matrix.build }} -j $(nproc)
|
||||||
|
|
||||||
- name: Clean after building using CMake
|
- name: Clean after building using CMake
|
||||||
@ -413,7 +308,7 @@ jobs:
|
|||||||
- name: Build using CMake w/ OpenBLAS
|
- name: Build using CMake w/ OpenBLAS
|
||||||
shell: msys2 {0}
|
shell: msys2 {0}
|
||||||
run: |
|
run: |
|
||||||
cmake -B build -DGGML_BLAS=ON -DGGML_BLAS_VENDOR=OpenBLAS
|
cmake -B build -DGGML_OPENBLAS=ON
|
||||||
cmake --build build --config ${{ matrix.build }} -j $(nproc)
|
cmake --build build --config ${{ matrix.build }} -j $(nproc)
|
||||||
|
|
||||||
windows:
|
windows:
|
||||||
@ -487,8 +382,10 @@ jobs:
|
|||||||
sdl2: [ON]
|
sdl2: [ON]
|
||||||
include:
|
include:
|
||||||
- arch: Win32
|
- arch: Win32
|
||||||
|
obzip: https://github.com/OpenMathLib/OpenBLAS/releases/download/v0.3.25/OpenBLAS-0.3.25-x86.zip
|
||||||
s2arc: x86
|
s2arc: x86
|
||||||
- arch: x64
|
- arch: x64
|
||||||
|
obzip: https://github.com/OpenMathLib/OpenBLAS/releases/download/v0.3.25/OpenBLAS-0.3.25-x64.zip
|
||||||
s2arc: x64
|
s2arc: x64
|
||||||
- sdl2: ON
|
- sdl2: ON
|
||||||
s2ver: 2.28.5
|
s2ver: 2.28.5
|
||||||
@ -497,21 +394,17 @@ jobs:
|
|||||||
- name: Clone
|
- name: Clone
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Export GitHub Actions cache environment variables
|
|
||||||
uses: actions/github-script@v7
|
|
||||||
with:
|
|
||||||
script: |
|
|
||||||
core.exportVariable('ACTIONS_CACHE_URL', process.env.ACTIONS_CACHE_URL || '');
|
|
||||||
core.exportVariable('ACTIONS_RUNTIME_TOKEN', process.env.ACTIONS_RUNTIME_TOKEN || '');
|
|
||||||
|
|
||||||
- name: Add msbuild to PATH
|
- name: Add msbuild to PATH
|
||||||
uses: microsoft/setup-msbuild@v2
|
uses: microsoft/setup-msbuild@v2
|
||||||
|
|
||||||
- name: Install OpenBLAS and pkgconfiglite
|
- name: Fetch OpenBLAS
|
||||||
if: matrix.blas == 'ON'
|
if: matrix.blas == 'ON'
|
||||||
run: |
|
run: |
|
||||||
vcpkg install --triplet=${{ matrix.s2arc }}-windows openblas
|
C:/msys64/usr/bin/wget.exe -qO blas.zip ${{ matrix.obzip }}
|
||||||
choco install pkgconfiglite
|
7z x blas.zip -oblas -y
|
||||||
|
copy blas/include/cblas.h .
|
||||||
|
copy blas/include/openblas_config.h .
|
||||||
|
echo "OPENBLAS_PATH=$env:GITHUB_WORKSPACE/blas" >> $env:GITHUB_ENV
|
||||||
|
|
||||||
- name: Fetch SDL2 and set SDL2_DIR
|
- name: Fetch SDL2 and set SDL2_DIR
|
||||||
if: matrix.sdl2 == 'ON'
|
if: matrix.sdl2 == 'ON'
|
||||||
@ -523,10 +416,9 @@ jobs:
|
|||||||
- name: Configure
|
- name: Configure
|
||||||
run: >
|
run: >
|
||||||
cmake -S . -B ./build -A ${{ matrix.arch }}
|
cmake -S . -B ./build -A ${{ matrix.arch }}
|
||||||
-DCMAKE_TOOLCHAIN_FILE="$env:VCPKG_INSTALLATION_ROOT/scripts/buildsystems/vcpkg.cmake"
|
|
||||||
-DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
-DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
||||||
-DGGML_BLAS=${{ matrix.blas }}
|
-DGGML_OPENBLAS=${{ matrix.blas }}
|
||||||
-DGGML_BLAS_VENDOR=OpenBLAS
|
-DCMAKE_LIBRARY_PATH="$env:OPENBLAS_PATH/lib"
|
||||||
-DWHISPER_SDL2=${{ matrix.sdl2 }}
|
-DWHISPER_SDL2=${{ matrix.sdl2 }}
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
@ -534,9 +426,9 @@ jobs:
|
|||||||
cd ./build
|
cd ./build
|
||||||
msbuild ALL_BUILD.vcxproj -t:build -p:configuration=${{ matrix.build }} -p:platform=${{ matrix.arch }}
|
msbuild ALL_BUILD.vcxproj -t:build -p:configuration=${{ matrix.build }} -p:platform=${{ matrix.arch }}
|
||||||
|
|
||||||
- name: Copy openblas.dll
|
- name: Copy libopenblas.dll
|
||||||
if: matrix.blas == 'ON'
|
if: matrix.blas == 'ON'
|
||||||
run: copy "C:/vcpkg/packages/openblas_${{ matrix.s2arc }}-windows/bin/openblas.dll" build/bin/${{ matrix.build }}
|
run: copy "$env:OPENBLAS_PATH/bin/libopenblas.dll" build/bin/${{ matrix.build }}
|
||||||
|
|
||||||
- name: Copy SDL2.dll
|
- name: Copy SDL2.dll
|
||||||
if: matrix.sdl2 == 'ON'
|
if: matrix.sdl2 == 'ON'
|
||||||
@ -551,6 +443,7 @@ jobs:
|
|||||||
|
|
||||||
windows-cublas:
|
windows-cublas:
|
||||||
runs-on: windows-2019
|
runs-on: windows-2019
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
build: [Release]
|
build: [Release]
|
||||||
@ -560,10 +453,12 @@ jobs:
|
|||||||
cuda-toolkit: [12.2.0, 11.8.0]
|
cuda-toolkit: [12.2.0, 11.8.0]
|
||||||
include:
|
include:
|
||||||
- arch: x64
|
- arch: x64
|
||||||
sdl2: ON
|
s2arc: x64
|
||||||
sdl2_ver: 2.28.5
|
- sdl2: ON
|
||||||
|
s2ver: 2.28.5
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Clone repository
|
- name: Clone
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Add msbuild to PATH
|
- name: Add msbuild to PATH
|
||||||
@ -575,50 +470,45 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
cuda: '${{ matrix.cuda-toolkit }}'
|
cuda: '${{ matrix.cuda-toolkit }}'
|
||||||
|
|
||||||
- name: Install 7-Zip
|
|
||||||
run: choco install 7zip -y
|
|
||||||
|
|
||||||
- name: Fetch SDL2 and set SDL2_DIR
|
- name: Fetch SDL2 and set SDL2_DIR
|
||||||
if: matrix.sdl2 == 'ON'
|
if: matrix.sdl2 == 'ON'
|
||||||
run: |
|
run: |
|
||||||
Invoke-WebRequest -Uri https://github.com/libsdl-org/SDL/releases/download/release-${{ matrix.sdl2_ver }}/SDL2-devel-${{ matrix.sdl2_ver }}-VC.zip -OutFile sdl2.zip
|
C:/msys64/usr/bin/wget.exe -qO sdl2.zip https://github.com/libsdl-org/SDL/releases/download/release-${{ matrix.s2ver }}/SDL2-devel-${{ matrix.s2ver }}-VC.zip
|
||||||
7z x sdl2.zip
|
7z x sdl2.zip
|
||||||
echo "SDL2_DIR=${{ github.workspace }}\SDL2-${{ matrix.sdl2_ver }}\cmake" | Out-File -FilePath $env:GITHUB_ENV -Append
|
echo "SDL2_DIR=$env:GITHUB_WORKSPACE/SDL2-${{ matrix.s2ver }}/cmake" >> $env:GITHUB_ENV
|
||||||
echo "${{ github.workspace }}\SDL2-${{ matrix.sdl2_ver }}\cmake" > SDL2_PATH.txt
|
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure
|
||||||
shell: cmd
|
run: >
|
||||||
run: |
|
cmake -S . -B ./build -A ${{ matrix.arch }}
|
||||||
cmake -S . -B ./build -A ${{ matrix.arch }} ^
|
-DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
||||||
-DCMAKE_BUILD_TYPE=${{ matrix.build }} ^
|
-DGGML_CUDA=${{ matrix.cublas }}
|
||||||
-DGGML_CUDA=${{ matrix.cublas }} ^
|
-DWHISPER_SDL2=${{ matrix.sdl2 }}
|
||||||
-DCMAKE_CUDA_ARCHITECTURES=all ^
|
|
||||||
-DWHISPER_SDL2=${{ matrix.sdl2 }} ^
|
|
||||||
-DSDL2_DIR="%SDL2_DIR%"
|
|
||||||
|
|
||||||
- name: Build Project
|
- name: Build ${{ matrix.cuda-toolkit }}
|
||||||
shell: cmd
|
|
||||||
run: |
|
run: |
|
||||||
cd ./build
|
cd ./build
|
||||||
cmake --build . --config ${{ matrix.build }}
|
cmake --build . --config ${{ matrix.build }}
|
||||||
|
|
||||||
- name: Copy CUDA DLLs
|
- name: Copy CUDA DLLs
|
||||||
run: |
|
run: >
|
||||||
Get-ChildItem "${{ steps.cuda-toolkit.outputs.CUDA_PATH }}/bin/" -Filter "*.dll" |
|
Copy-Item -PassThru
|
||||||
Copy-Item -Destination "build/bin/${{ matrix.build }}"
|
-Path "${{ steps.cuda-toolkit.outputs.CUDA_PATH }}/bin/*.dll"
|
||||||
|
-Include cudart64_*,cublas64_*,cublasLt64_*
|
||||||
|
-Destination build/bin/${{ matrix.build }}
|
||||||
|
|
||||||
- name: Copy SDL2.dll
|
- name: Copy SDL2.dll
|
||||||
if: matrix.sdl2 == 'ON'
|
if: matrix.sdl2 == 'ON'
|
||||||
run: copy "$env:SDL2_DIR/../lib/${{ matrix.arch }}/SDL2.dll" build/bin/${{ matrix.build }}
|
run: copy "$env:SDL2_DIR/../lib/${{ matrix.s2arc }}/SDL2.dll" build/bin/${{ matrix.build }}
|
||||||
|
|
||||||
- name: Upload binaries
|
- name: Upload binaries
|
||||||
|
if: matrix.sdl2 == 'ON'
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: whisper-cublas-${{ matrix.cuda-toolkit }}-bin-${{ matrix.arch }}
|
name: whisper-cublas-${{ matrix.cuda-toolkit }}-bin-${{ matrix.arch }}
|
||||||
path: build/bin/${{ matrix.build }}
|
path: build/bin/${{ matrix.build }}
|
||||||
|
|
||||||
emscripten:
|
emscripten:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
@ -639,7 +529,7 @@ jobs:
|
|||||||
emcmake cmake . -DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
emcmake cmake . -DCMAKE_BUILD_TYPE=${{ matrix.build }}
|
||||||
make
|
make
|
||||||
|
|
||||||
ios-xcode-build:
|
ios:
|
||||||
runs-on: macos-latest
|
runs-on: macos-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
@ -647,7 +537,7 @@ jobs:
|
|||||||
build: [Release]
|
build: [Release]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Clone
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Configure
|
- name: Configure
|
||||||
@ -655,37 +545,14 @@ jobs:
|
|||||||
cp models/for-tests-ggml-base.en.bin models/ggml-base.en.bin
|
cp models/for-tests-ggml-base.en.bin models/ggml-base.en.bin
|
||||||
mkdir models/ggml-base.en-encoder.mlmodelc
|
mkdir models/ggml-base.en-encoder.mlmodelc
|
||||||
|
|
||||||
- name: Build
|
|
||||||
id: cmake_build
|
|
||||||
run: |
|
|
||||||
sysctl -a
|
|
||||||
mkdir build
|
|
||||||
cd build
|
|
||||||
cmake -G Xcode .. \
|
|
||||||
-DGGML_METAL_USE_BF16=ON \
|
|
||||||
-DGGML_METAL_EMBED_LIBRARY=ON \
|
|
||||||
-DWHISPER_BUILD_EXAMPLES=OFF \
|
|
||||||
-DWHISPER_BUILD_TESTS=OFF \
|
|
||||||
-DWHISPER_BUILD_SERVER=OFF \
|
|
||||||
-DCMAKE_SYSTEM_NAME=iOS \
|
|
||||||
-DCMAKE_OSX_DEPLOYMENT_TARGET=14.0 \
|
|
||||||
-DCMAKE_XCODE_ATTRIBUTE_DEVELOPMENT_TEAM=ggml
|
|
||||||
cmake --build . --config Release -j $(sysctl -n hw.logicalcpu) -- CODE_SIGNING_ALLOWED=NO
|
|
||||||
sudo cmake --install . --config Release
|
|
||||||
|
|
||||||
- name: xcodebuild for swift package
|
|
||||||
id: xcodebuild
|
|
||||||
run: |
|
|
||||||
xcodebuild -scheme whisper-Package -destination 'generic/platform=iOS'
|
|
||||||
|
|
||||||
- name: Build objc example
|
- name: Build objc example
|
||||||
run: xcodebuild -project examples/whisper.objc/whisper.objc.xcodeproj -scheme whisper.objc -configuration ${{ matrix.build }} -sdk iphoneos CODE_SIGN_IDENTITY="" CODE_SIGNING_REQUIRED=NO build
|
run: xcodebuild -project examples/whisper.objc/whisper.objc.xcodeproj -scheme whisper.objc -configuration ${{ matrix.build }} -sdk iphonesimulator build
|
||||||
|
|
||||||
- name: Build swiftui example
|
- name: Build swiftui example
|
||||||
run: xcodebuild -project examples/whisper.swiftui/whisper.swiftui.xcodeproj -scheme WhisperCppDemo -configuration ${{ matrix.build }} -sdk iphoneos CODE_SIGNING_REQUIRED=NO CODE_SIGN_IDENTITY= -destination 'generic/platform=iOS' build
|
run: xcodebuild -project examples/whisper.swiftui/whisper.swiftui.xcodeproj -scheme WhisperCppDemo -configuration ${{ matrix.build }} -sdk iphonesimulator build
|
||||||
|
|
||||||
android:
|
android:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Clone
|
- name: Clone
|
||||||
@ -693,6 +560,12 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
path: whisper
|
path: whisper
|
||||||
|
|
||||||
|
- name: Clone
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
repository: ggerganov/ggml
|
||||||
|
path: ggml
|
||||||
|
|
||||||
- name: Install Java
|
- name: Install Java
|
||||||
uses: actions/setup-java@v4
|
uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
@ -711,80 +584,78 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
export PATH_TO_GGML=$PWD/ggml
|
export PATH_TO_GGML=$PWD/ggml
|
||||||
cd whisper/examples/whisper.android
|
cd whisper/examples/whisper.android
|
||||||
./gradlew assembleRelease --no-daemon
|
./gradlew assembleRelease --no-daemon -PGGML_HOME=$PATH_TO_GGML
|
||||||
|
|
||||||
# TODO: disable because of following fail: https://github.com/ggerganov/whisper.cpp/actions/runs/11019444420/job/30627193602
|
android_java:
|
||||||
# android_java:
|
runs-on: ubuntu-latest
|
||||||
# runs-on: ubuntu-22.04
|
|
||||||
#
|
|
||||||
# steps:
|
|
||||||
# - name: Clone
|
|
||||||
# uses: actions/checkout@v4
|
|
||||||
#
|
|
||||||
# - name: set up JDK 11
|
|
||||||
# uses: actions/setup-java@v4
|
|
||||||
# with:
|
|
||||||
# java-version: '11'
|
|
||||||
# distribution: 'temurin'
|
|
||||||
# cache: gradle
|
|
||||||
#
|
|
||||||
# - name: Setup Android SDK
|
|
||||||
# uses: android-actions/setup-android@v3
|
|
||||||
# with:
|
|
||||||
# cmdline-tools-version: 9.0
|
|
||||||
#
|
|
||||||
# - name: Build
|
|
||||||
# run: |
|
|
||||||
# cd examples/whisper.android.java
|
|
||||||
# chmod +x ./gradlew
|
|
||||||
# ./gradlew assembleRelease
|
|
||||||
|
|
||||||
# TODO: disabled because of following fail: https://github.com/ggerganov/whisper.cpp/actions/runs/9686220096/job/26735899598
|
steps:
|
||||||
# java:
|
- name: Clone
|
||||||
# needs: [ 'windows' ]
|
uses: actions/checkout@v4
|
||||||
# runs-on: windows-latest
|
|
||||||
# steps:
|
- name: set up JDK 11
|
||||||
# - uses: actions/checkout@v4
|
uses: actions/setup-java@v4
|
||||||
#
|
with:
|
||||||
# - name: Install Java
|
java-version: '11'
|
||||||
# uses: actions/setup-java@v4
|
distribution: 'temurin'
|
||||||
# with:
|
cache: gradle
|
||||||
# distribution: zulu
|
|
||||||
# java-version: 20
|
- name: Setup Android SDK
|
||||||
#
|
uses: android-actions/setup-android@v3
|
||||||
# - name: Download Windows lib
|
with:
|
||||||
# uses: actions/download-artifact@v4
|
cmdline-tools-version: 9.0
|
||||||
# with:
|
|
||||||
# name: win32-x86-64_whisper.dll
|
- name: Build
|
||||||
# path: bindings/java/build/generated/resources/main/win32-x86-64
|
run: |
|
||||||
#
|
cd examples/whisper.android.java
|
||||||
# - name: Build
|
chmod +x ./gradlew
|
||||||
# run: |
|
./gradlew assembleRelease
|
||||||
# models\download-ggml-model.cmd tiny.en
|
|
||||||
# cd bindings/java
|
java:
|
||||||
# chmod +x ./gradlew
|
needs: [ 'windows' ]
|
||||||
# ./gradlew build
|
runs-on: windows-latest
|
||||||
#
|
steps:
|
||||||
# - name: Upload jar
|
- uses: actions/checkout@v4
|
||||||
# uses: actions/upload-artifact@v4
|
|
||||||
# with:
|
- name: Install Java
|
||||||
# name: whispercpp.jar
|
uses: actions/setup-java@v4
|
||||||
# path: bindings/java/build/libs/whispercpp-*.jar
|
with:
|
||||||
#
|
distribution: zulu
|
||||||
# - name: Publish package
|
java-version: 20
|
||||||
# if: ${{ github.ref == 'refs/heads/master' }}
|
|
||||||
# uses: gradle/gradle-build-action@v2.4.2
|
- name: Download Windows lib
|
||||||
# with:
|
uses: actions/download-artifact@v4
|
||||||
# arguments: publish
|
with:
|
||||||
# build-root-directory: bindings/java
|
name: win32-x86-64_whisper.dll
|
||||||
# env:
|
path: bindings/java/build/generated/resources/main/win32-x86-64
|
||||||
# MAVEN_USERNAME: ${{ secrets.JIRA_USER }}
|
|
||||||
# MAVEN_PASSWORD: ${{ secrets.JIRA_PASS }}
|
- name: Build
|
||||||
# PGP_SECRET: ${{ secrets.GPG_PRIVATE_KEY }}
|
run: |
|
||||||
# PGP_PASSPHRASE: ${{ secrets.GPG_PASSPHRASE }}
|
models\download-ggml-model.cmd tiny.en
|
||||||
|
cd bindings/java
|
||||||
|
chmod +x ./gradlew
|
||||||
|
./gradlew build
|
||||||
|
|
||||||
|
- name: Upload jar
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: whispercpp.jar
|
||||||
|
path: bindings/java/build/libs/whispercpp-*.jar
|
||||||
|
|
||||||
|
- name: Publish package
|
||||||
|
if: ${{ github.ref == 'refs/heads/master' }}
|
||||||
|
uses: gradle/gradle-build-action@v2.4.2
|
||||||
|
with:
|
||||||
|
arguments: publish
|
||||||
|
build-root-directory: bindings/java
|
||||||
|
env:
|
||||||
|
MAVEN_USERNAME: ${{ secrets.JIRA_USER }}
|
||||||
|
MAVEN_PASSWORD: ${{ secrets.JIRA_PASS }}
|
||||||
|
PGP_SECRET: ${{ secrets.GPG_PRIVATE_KEY }}
|
||||||
|
PGP_PASSPHRASE: ${{ secrets.GPG_PASSPHRASE }}
|
||||||
|
|
||||||
quantize:
|
quantize:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Clone
|
- name: Clone
|
||||||
@ -793,6 +664,5 @@ jobs:
|
|||||||
- name: Test quantize
|
- name: Test quantize
|
||||||
run: |
|
run: |
|
||||||
./models/download-ggml-model.sh tiny.en
|
./models/download-ggml-model.sh tiny.en
|
||||||
cmake -B build
|
make quantize
|
||||||
cmake --build build --config Release
|
./quantize models/ggml-tiny.en.bin models/ggml-tiny.en-q4_0.bin q4_0
|
||||||
./build/bin/quantize models/ggml-tiny.en.bin models/ggml-tiny.en-q4_0.bin q4_0
|
|
||||||
|
12
.github/workflows/docker.yml
vendored
12
.github/workflows/docker.yml
vendored
@ -11,16 +11,14 @@ jobs:
|
|||||||
name: Push Docker image to Docker Hub
|
name: Push Docker image to Docker Hub
|
||||||
if: github.event.pull_request.draft == false
|
if: github.event.pull_request.draft == false
|
||||||
|
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
env:
|
env:
|
||||||
COMMIT_SHA: ${{ github.sha }}
|
COMMIT_SHA: ${{ github.sha }}
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
config:
|
config:
|
||||||
- { tag: "main", dockerfile: ".devops/main.Dockerfile", platform: "linux/amd64" }
|
- { tag: "main", dockerfile: ".devops/main.Dockerfile", platform: "linux/amd64,linux/arm64" }
|
||||||
#TODO: the cuda image keeps failing - disable for now
|
- { tag: "main-cuda", dockerfile: ".devops/main-cuda.Dockerfile", platform: "linux/amd64" }
|
||||||
# https://github.com/ggerganov/whisper.cpp/actions/runs/11019444428/job/30602020339
|
|
||||||
#- { tag: "main-cuda", dockerfile: ".devops/main-cuda.Dockerfile", platform: "linux/amd64" }
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out the repo
|
- name: Check out the repo
|
||||||
@ -45,7 +43,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
push: true
|
push: true
|
||||||
platforms: ${{ matrix.config.platform }}
|
platforms: ${{ matrix.config.platforms }}
|
||||||
tags: "ghcr.io/${{ github.repository }}:${{ matrix.config.tag }}-${{ env.COMMIT_SHA }}"
|
tags: "ghcr.io/${{ github.repository }}:${{ matrix.config.tag }}-${{ env.COMMIT_SHA }}"
|
||||||
file: ${{ matrix.config.dockerfile }}
|
file: ${{ matrix.config.dockerfile }}
|
||||||
|
|
||||||
@ -54,6 +52,6 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
push: ${{ github.event_name == 'push' }}
|
push: ${{ github.event_name == 'push' }}
|
||||||
platforms: ${{ matrix.config.platform }}
|
platforms: ${{ matrix.config.platforms }}
|
||||||
tags: "ghcr.io/${{ github.repository }}:${{ matrix.config.tag }}"
|
tags: "ghcr.io/${{ github.repository }}:${{ matrix.config.tag }}"
|
||||||
file: ${{ matrix.config.dockerfile }}
|
file: ${{ matrix.config.dockerfile }}
|
||||||
|
6
.github/workflows/examples.yml
vendored
6
.github/workflows/examples.yml
vendored
@ -10,8 +10,8 @@ on:
|
|||||||
- whisper.h
|
- whisper.h
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
addon_node-ubuntu-22:
|
addon_node-ubuntu-latest:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
node-version: [ 16.x, 18.x ]
|
node-version: [ 16.x, 18.x ]
|
||||||
@ -22,7 +22,7 @@ jobs:
|
|||||||
- name: Dependencies
|
- name: Dependencies
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install build-essential git
|
sudo apt-get install build-essential
|
||||||
sudo apt-get install cmake
|
sudo apt-get install cmake
|
||||||
sudo apt-get install libsdl2-dev
|
sudo apt-get install libsdl2-dev
|
||||||
|
|
||||||
|
6
.gitignore
vendored
6
.gitignore
vendored
@ -1,16 +1,13 @@
|
|||||||
*.o
|
*.o
|
||||||
*.a
|
*.a
|
||||||
*.d
|
|
||||||
.cache/
|
.cache/
|
||||||
.coreml/
|
.coreml/
|
||||||
.test/
|
.test/
|
||||||
.venv/
|
|
||||||
.vs/
|
.vs/
|
||||||
.vscode/
|
.vscode/
|
||||||
.DS_Store
|
.DS_Store
|
||||||
.vimspector.json
|
.vimspector.json
|
||||||
/CMakeSettings.json
|
/CMakeSettings.json
|
||||||
/talk-llama.dSYM/
|
|
||||||
|
|
||||||
build/
|
build/
|
||||||
build-*/
|
build-*/
|
||||||
@ -20,9 +17,6 @@ build-*/
|
|||||||
.swiftpm
|
.swiftpm
|
||||||
*.metallib
|
*.metallib
|
||||||
|
|
||||||
ggml-metal-embed.metal
|
|
||||||
ggml-metal-embed.metal.tmp
|
|
||||||
|
|
||||||
/main
|
/main
|
||||||
/stream
|
/stream
|
||||||
/command
|
/command
|
||||||
|
211
AUTHORS
211
AUTHORS
@ -1,51 +1,34 @@
|
|||||||
# date: Tue Feb 4 13:03:35 EET 2025
|
# date: Tue Apr 9 20:27:03 EEST 2024
|
||||||
# this file is auto-generated by scripts/gen-authors.sh
|
# this file is auto-generated by scripts/gen-authors.sh
|
||||||
|
|
||||||
0/0 <zero@imaskeleton.me>
|
0/0 <zero@imaskeleton.me>
|
||||||
0cc4m <picard12@live.de>
|
0cc4m <picard12@live.de>
|
||||||
0xsourcecode <134374803+0xsourcecode@users.noreply.github.com>
|
0xsourcecode <134374803+0xsourcecode@users.noreply.github.com>
|
||||||
65a <10104049+65a@users.noreply.github.com>
|
|
||||||
AIWintermuteAI <32562299+AIWintermuteAI@users.noreply.github.com>
|
|
||||||
AT <manyoso@users.noreply.github.com>
|
AT <manyoso@users.noreply.github.com>
|
||||||
Aarni Koskela <akx@iki.fi>
|
Aarni Koskela <akx@iki.fi>
|
||||||
Aaron Pham <29749331+aarnphm@users.noreply.github.com>
|
Aaron Pham <29749331+aarnphm@users.noreply.github.com>
|
||||||
Aaron Taylor <aaron@exphat.com>
|
Aaron Taylor <aaron@exphat.com>
|
||||||
Abhilash Majumder <30946547+abhilash1910@users.noreply.github.com>
|
Abhilash Majumder <30946547+abhilash1910@users.noreply.github.com>
|
||||||
Abitofevrything <54505189+abitofevrything@users.noreply.github.com>
|
Abitofevrything <54505189+abitofevrything@users.noreply.github.com>
|
||||||
Adam Jones <domdomegg+git@gmail.com>
|
|
||||||
Adrien Gallouët <adrien@gallouet.fr>
|
|
||||||
Adrien Gallouët <angt@huggingface.co>
|
|
||||||
AfryMask <AfryMask@163.com>
|
AfryMask <AfryMask@163.com>
|
||||||
Ahmad Bilal <ahmad.bilal@empglabs.com>
|
Ahmad Bilal <ahmad.bilal@empglabs.com>
|
||||||
Ahmad Tameem <113388789+Tameem-10xE@users.noreply.github.com>
|
|
||||||
AidanBeltonS <87009434+AidanBeltonS@users.noreply.github.com>
|
AidanBeltonS <87009434+AidanBeltonS@users.noreply.github.com>
|
||||||
AidanBeltonS <aidan.belton@codeplay.com>
|
|
||||||
Akarshan Biswas <akarshan.biswas@gmail.com>
|
|
||||||
Akarshan Biswas <akarshanbiswas@fedoraproject.org>
|
|
||||||
Akash Mahajan <akash7190@gmail.com>
|
Akash Mahajan <akash7190@gmail.com>
|
||||||
Akash Mahajan <akashmjn@stanford.edu>
|
Akash Mahajan <akashmjn@stanford.edu>
|
||||||
Al Hoang <3811822-hoanga@users.noreply.gitlab.com>
|
Al Hoang <3811822-hoanga@users.noreply.gitlab.com>
|
||||||
Alan <unknown>
|
Alan <unknown>
|
||||||
Albert Jin <albert.jin@gmail.com>
|
|
||||||
Alberto Cabrera Pérez <alberto.cabrera@codeplay.com>
|
|
||||||
Alberto Cabrera Pérez <alberto.cabrera@intel.com>
|
|
||||||
Aleksander Andrzejewski <18704749+aleksanderandrzejewski@users.noreply.github.com>
|
Aleksander Andrzejewski <18704749+aleksanderandrzejewski@users.noreply.github.com>
|
||||||
Alex Azarov <alex@azarov.by>
|
Alex Azarov <alex@azarov.by>
|
||||||
Alex Bacart <13940752+alex-bacart@users.noreply.github.com>
|
Alex Bacart <13940752+alex-bacart@users.noreply.github.com>
|
||||||
Alex Evgrashin <aevgrashin@yandex.ru>
|
Alex Evgrashin <aevgrashin@yandex.ru>
|
||||||
Alex O'Connell <35843486+acon96@users.noreply.github.com>
|
|
||||||
Alexandr Graschenkov <alexandr.graschenkov91@gmail.com>
|
Alexandr Graschenkov <alexandr.graschenkov91@gmail.com>
|
||||||
Alexandru Mariuti <alex@mariuti.com>
|
Alexandru Mariuti <alex@mariuti.com>
|
||||||
Alexey Kharlamov <alexey@kharlamov.biz>
|
Alexey Kharlamov <alexey@kharlamov.biz>
|
||||||
Alfredo Montesinos <alfredo.montesinos@g.austincc.edu>
|
Alfredo Montesinos <alfredo.montesinos@g.austincc.edu>
|
||||||
Ali Alameh <ali.alameh@isae.edu.lb>
|
Ali Alameh <ali.alameh@isae.edu.lb>
|
||||||
Alter <0x7c48@gmail.com>
|
|
||||||
Ananta Bastola <anantarajbastola@gmail.com>
|
Ananta Bastola <anantarajbastola@gmail.com>
|
||||||
Andreas Kieslinger <47689530+aendk@users.noreply.github.com>
|
|
||||||
Andreas Lubbe <git@lubbe.org>
|
|
||||||
Andreu Huguet <andreuhuguet@gmail.com>
|
Andreu Huguet <andreuhuguet@gmail.com>
|
||||||
Andrew Huynh <a5thuynh@gmail.com>
|
Andrew Huynh <a5thuynh@gmail.com>
|
||||||
Andrew Minh Nguyen <40281306+amqdn@users.noreply.github.com>
|
|
||||||
Andrew S <andrews54757@gmail.com>
|
Andrew S <andrews54757@gmail.com>
|
||||||
Andy Maloney <asmaloney@gmail.com>
|
Andy Maloney <asmaloney@gmail.com>
|
||||||
Anton Kostin <masguit42@users.noreply.github.com>
|
Anton Kostin <masguit42@users.noreply.github.com>
|
||||||
@ -57,11 +40,8 @@ AustinMroz <austinmroz@utexas.edu>
|
|||||||
Avik Sengupta <avik@sengupta.net>
|
Avik Sengupta <avik@sengupta.net>
|
||||||
Bader-eddine Ouaich <49657842+baderouaich@users.noreply.github.com>
|
Bader-eddine Ouaich <49657842+baderouaich@users.noreply.github.com>
|
||||||
Baffin Lee <baffinlee@gmail.com>
|
Baffin Lee <baffinlee@gmail.com>
|
||||||
Ben Ashbaugh <ben.ashbaugh@intel.com>
|
|
||||||
Ben Nortier <bjnortier@gmail.com>
|
Ben Nortier <bjnortier@gmail.com>
|
||||||
Benjamin Heiniger <benjamin.heiniger@bluewin.ch>
|
Benjamin Heiniger <benjamin.heiniger@bluewin.ch>
|
||||||
Bernhard M. Wiedemann <githubbmwprimary@lsmod.de>
|
|
||||||
Binozo <70137898+Binozo@users.noreply.github.com>
|
|
||||||
Bo-Yi Wu <appleboy.tw@gmail.com>
|
Bo-Yi Wu <appleboy.tw@gmail.com>
|
||||||
Boris Bliznioukov <blib@mail.com>
|
Boris Bliznioukov <blib@mail.com>
|
||||||
Borislav Stanimirov <b.stanimirov@abv.bg>
|
Borislav Stanimirov <b.stanimirov@abv.bg>
|
||||||
@ -69,86 +49,47 @@ Brad Murray <59848399+bradmurray-dt@users.noreply.github.com>
|
|||||||
Brian Murray <brian@bmurray.ca>
|
Brian Murray <brian@bmurray.ca>
|
||||||
CRD716 <crd716@gmail.com>
|
CRD716 <crd716@gmail.com>
|
||||||
Canis Lupus <Canis-UK@users.noreply.github.com>
|
Canis Lupus <Canis-UK@users.noreply.github.com>
|
||||||
Carlos Zoido <mrgalleta@gmail.com>
|
|
||||||
Carolinabanana <140120812+Carolinabanana@users.noreply.github.com>
|
Carolinabanana <140120812+Carolinabanana@users.noreply.github.com>
|
||||||
CarterLi999 <664681047@qq.com>
|
|
||||||
ChangSeok Oh <shivamidow@users.noreply.github.com>
|
ChangSeok Oh <shivamidow@users.noreply.github.com>
|
||||||
Changyeon Kim <cyzero.kim@samsung.com>
|
|
||||||
Chaoqun <27287694+OpenWaygate@users.noreply.github.com>
|
Chaoqun <27287694+OpenWaygate@users.noreply.github.com>
|
||||||
Charles Xu <63788048+chaxu01@users.noreply.github.com>
|
|
||||||
Charles Xu <charles.xu@arm.com>
|
|
||||||
Chen Xi <xi2.chen@intel.com>
|
|
||||||
Chen Xi <xixichen08@foxmail.com>
|
|
||||||
Chenguang Li <87689256+noemotiovon@users.noreply.github.com>
|
|
||||||
Chia-Hsiang Cheng <88014292+garychia@users.noreply.github.com>
|
Chia-Hsiang Cheng <88014292+garychia@users.noreply.github.com>
|
||||||
Chidi Williams <williamschidi1@gmail.com>
|
Chidi Williams <williamschidi1@gmail.com>
|
||||||
Chris Elrod <elrodc@gmail.com>
|
|
||||||
Christian <12550267+iceychris@users.noreply.github.com>
|
Christian <12550267+iceychris@users.noreply.github.com>
|
||||||
Christian Kastner <ckk@kvr.at>
|
|
||||||
Clifford Heath <clifford.heath@gmail.com>
|
Clifford Heath <clifford.heath@gmail.com>
|
||||||
Clint Herron <hanclinto@gmail.com>
|
|
||||||
Colin <github@whoisc.cc>
|
Colin <github@whoisc.cc>
|
||||||
Conrad Kramer <conrad@conradkramer.com>
|
|
||||||
Corey Earwood <iamcgn+github@gmail.com>
|
|
||||||
CrispStrobe <154636388+CrispStrobe@users.noreply.github.com>
|
|
||||||
DAN™ <dranger003@gmail.com>
|
|
||||||
DGdev91 <DGdev91@users.noreply.github.com>
|
DGdev91 <DGdev91@users.noreply.github.com>
|
||||||
Damian Czaja <trojan295@protonmail.com>
|
Damian Czaja <trojan295@protonmail.com>
|
||||||
Dan Johansson <164997844+eddnjjn@users.noreply.github.com>
|
|
||||||
Dan Johansson <dan.johansson@arm.com>
|
|
||||||
Daniel Bevenius <daniel.bevenius@gmail.com>
|
Daniel Bevenius <daniel.bevenius@gmail.com>
|
||||||
Daniel Valdivia <18384552+dvaldivia@users.noreply.github.com>
|
|
||||||
Daniel Ziegenberg <daniel@ziegenberg.at>
|
|
||||||
Daniele <57776841+daniandtheweb@users.noreply.github.com>
|
|
||||||
Dave <dave-fl@users.noreply.github.com>
|
|
||||||
Dave Airlie <airlied@gmail.com>
|
|
||||||
Dave Airlie <airlied@redhat.com>
|
|
||||||
Daven Sanassy <daven@vochlea.co.uk>
|
|
||||||
David <dnhkng@gmail.com>
|
David <dnhkng@gmail.com>
|
||||||
David Thorpe <djt@mutablelogic.com>
|
David Thorpe <djt@mutablelogic.com>
|
||||||
DavidKorczynski <david@adalogics.com>
|
|
||||||
Davidson Francis <davidsondfgl@gmail.com>
|
Davidson Francis <davidsondfgl@gmail.com>
|
||||||
Dener Stassun <denerstassun@gmail.com>
|
Dener Stassun <denerstassun@gmail.com>
|
||||||
Dibakar Gope <dibakar.gope@arm.com>
|
|
||||||
Didzis Gosko <didzis@users.noreply.github.com>
|
Didzis Gosko <didzis@users.noreply.github.com>
|
||||||
Diego Devesa <slarengh@gmail.com>
|
|
||||||
Digipom <admin@digipom.com>
|
Digipom <admin@digipom.com>
|
||||||
Dimo <dimo@ieee.org>
|
Dimo <dimo@ieee.org>
|
||||||
Djip007 <3705339+Djip007@users.noreply.github.com>
|
|
||||||
Djip007 <djip.perois@free.fr>
|
|
||||||
Dody Suria Wijaya <dodysw@gmail.com>
|
Dody Suria Wijaya <dodysw@gmail.com>
|
||||||
Dou Xinpeng <15529241576@163.com>
|
|
||||||
Dou Xinpeng <81913537+Dou-Git@users.noreply.github.com>
|
|
||||||
Dr. Tom Murphy VII Ph.D <499244+tom7@users.noreply.github.com>
|
Dr. Tom Murphy VII Ph.D <499244+tom7@users.noreply.github.com>
|
||||||
Duncan McConnell <ddmcconnell4@gmail.com>
|
Duncan McConnell <ddmcconnell4@gmail.com>
|
||||||
Egor Egorov <me@egorfine.com>
|
Egor Egorov <me@egorfine.com>
|
||||||
Elkana Bardugo <ttv200@gmail.com>
|
Elkana Bardugo <ttv200@gmail.com>
|
||||||
Emmanuel Schmidbauer <eschmidbauer@gmail.com>
|
Emmanuel Schmidbauer <eschmidbauer@gmail.com>
|
||||||
Engininja2 <139037756+Engininja2@users.noreply.github.com>
|
Engininja2 <139037756+Engininja2@users.noreply.github.com>
|
||||||
Eric Curtin <ericcurtin17@gmail.com>
|
|
||||||
Eric Swanson <eswanson@alloscomp.com>
|
Eric Swanson <eswanson@alloscomp.com>
|
||||||
Eric Tendian <erictendian@gmail.com>
|
Eric Tendian <erictendian@gmail.com>
|
||||||
Eric Zhang <34133756+EZForever@users.noreply.github.com>
|
|
||||||
Erik Scholz <Green-Sky@users.noreply.github.com>
|
Erik Scholz <Green-Sky@users.noreply.github.com>
|
||||||
Evan Jones <evan.q.jones@gmail.com>
|
Evan Jones <evan.q.jones@gmail.com>
|
||||||
Evan Martin <evan.martin@gmail.com>
|
Evan Martin <evan.martin@gmail.com>
|
||||||
Eve <139727413+netrunnereve@users.noreply.github.com>
|
Eve <139727413+netrunnereve@users.noreply.github.com>
|
||||||
Evgeny Kuznetsov <evgeny@kuznetsov.md>
|
Evgeny Kuznetsov <evgeny@kuznetsov.md>
|
||||||
F1L1P <78918286+F1L1Pv2@users.noreply.github.com>
|
F1L1P <78918286+F1L1Pv2@users.noreply.github.com>
|
||||||
Faisal Zaghloul <quic_fzaghlou@quicinc.com>
|
|
||||||
Fangjun Kuang <csukuangfj@gmail.com>
|
Fangjun Kuang <csukuangfj@gmail.com>
|
||||||
Felix <stenbackfelix@gmail.com>
|
Felix <stenbackfelix@gmail.com>
|
||||||
Finn Voorhees <finnvoorhees@gmail.com>
|
Finn Voorhees <finnvoorhees@gmail.com>
|
||||||
FirstTimeEZ <179362031+FirstTimeEZ@users.noreply.github.com>
|
|
||||||
FlippFuzz <41221030+FlippFuzz@users.noreply.github.com>
|
FlippFuzz <41221030+FlippFuzz@users.noreply.github.com>
|
||||||
Frankie Robertson <frankier@users.noreply.github.com>
|
|
||||||
Gang Chen <goncha@gmail.com>
|
Gang Chen <goncha@gmail.com>
|
||||||
Gavin Cai <gavin1818@hotmail.com>
|
Gavin Cai <gavin1818@hotmail.com>
|
||||||
George Hindle <george@georgehindle.com>
|
George Hindle <george@georgehindle.com>
|
||||||
Georgi Gerganov <ggerganov@gmail.com>
|
Georgi Gerganov <ggerganov@gmail.com>
|
||||||
Gilad S <7817232+giladgd@users.noreply.github.com>
|
|
||||||
Gilad S <giladgd@users.noreply.github.com>
|
|
||||||
Gilad S. <7817232+giladgd@users.noreply.github.com>
|
|
||||||
GitAritron <103900385+GitAritron@users.noreply.github.com>
|
GitAritron <103900385+GitAritron@users.noreply.github.com>
|
||||||
GiviMAD <GiviMAD@users.noreply.github.com>
|
GiviMAD <GiviMAD@users.noreply.github.com>
|
||||||
Gleicon Moraes <gleicon@gmail.com>
|
Gleicon Moraes <gleicon@gmail.com>
|
||||||
@ -157,66 +98,41 @@ Guillaume Wenzek <gwenzek@users.noreply.github.com>
|
|||||||
HY. Kelvin Lee <34256578+hykelvinlee42@users.noreply.github.com>
|
HY. Kelvin Lee <34256578+hykelvinlee42@users.noreply.github.com>
|
||||||
Halalaluyafail3 <55773281+Halalaluyafail3@users.noreply.github.com>
|
Halalaluyafail3 <55773281+Halalaluyafail3@users.noreply.github.com>
|
||||||
Hang <bebound@gmail.com>
|
Hang <bebound@gmail.com>
|
||||||
Haus1 <haus.xda@gmail.com>
|
|
||||||
Herman Semenov <GermanAizek@yandex.ru>
|
Herman Semenov <GermanAizek@yandex.ru>
|
||||||
HimariO <dsfhe49854@gmail.com>
|
|
||||||
Hong Bo PENG <penghb@cn.ibm.com>
|
|
||||||
Hrishikesh Barman <geekodour@users.noreply.github.com>
|
Hrishikesh Barman <geekodour@users.noreply.github.com>
|
||||||
Hugo <hugo@whynothugo.nl>
|
|
||||||
Ian Bicking <ian@ianbicking.org>
|
Ian Bicking <ian@ianbicking.org>
|
||||||
Ian Bull <irbull@eclipsesource.com>
|
Ian Bull <irbull@eclipsesource.com>
|
||||||
Ihar Hrachyshka <ihrachys@redhat.com>
|
|
||||||
Ikko Ashimine <eltociear@gmail.com>
|
Ikko Ashimine <eltociear@gmail.com>
|
||||||
Ikko Eltociear Ashimine <eltociear@gmail.com>
|
|
||||||
InconsolableCellist <23345188+InconsolableCellist@users.noreply.github.com>
|
InconsolableCellist <23345188+InconsolableCellist@users.noreply.github.com>
|
||||||
Ismatulla Mansurov <47342870+sapoepsilon@users.noreply.github.com>
|
Ismatulla Mansurov <47342870+sapoepsilon@users.noreply.github.com>
|
||||||
Ivan <nekotekina@gmail.com>
|
|
||||||
Ivan Filipov <159561759+vanaka11@users.noreply.github.com>
|
|
||||||
Ivan Gorin <ivangorin21@gmail.com>
|
Ivan Gorin <ivangorin21@gmail.com>
|
||||||
Ivo von Putzer Reibegg <ivo.putzer@gmail.com>
|
|
||||||
JJ <103335846+computerscienceiscool@users.noreply.github.com>
|
JJ <103335846+computerscienceiscool@users.noreply.github.com>
|
||||||
Jack Mousseau <jmousseau@users.noreply.github.com>
|
Jack Mousseau <jmousseau@users.noreply.github.com>
|
||||||
JacobLinCool <jacoblincool@gmail.com>
|
JacobLinCool <jacoblincool@gmail.com>
|
||||||
Jakub Ráček <blizzcz@gmail.com>
|
Jakub Ráček <blizzcz@gmail.com>
|
||||||
Jared Van Bortel <jared@nomic.ai>
|
Jared Van Bortel <jared@nomic.ai>
|
||||||
Jay Binks <jaybinks@gmail.com>
|
Jay Binks <jaybinks@gmail.com>
|
||||||
Jayant <jayantyadav202@gmail.com>
|
|
||||||
Jeff Bolz <jbolz@nvidia.com>
|
|
||||||
Jeroen Mostert <jeroen.mostert@cm.com>
|
|
||||||
Jhen-Jie Hong <developer@jhen.me>
|
Jhen-Jie Hong <developer@jhen.me>
|
||||||
Jhen-Jie Hong <iainst0409@gmail.com>
|
Jhen-Jie Hong <iainst0409@gmail.com>
|
||||||
JidongZhang-THU <1119708529@qq.com>
|
JidongZhang-THU <1119708529@qq.com>
|
||||||
Jo Liss <joliss42@gmail.com>
|
Jo Liss <joliss42@gmail.com>
|
||||||
Joe Todd <joe.todd@codeplay.com>
|
|
||||||
Johan <jr.raffin@gmail.com>
|
Johan <jr.raffin@gmail.com>
|
||||||
Johannes Gäßler <johannesg@5d6.de>
|
Johannes Gäßler <johannesg@5d6.de>
|
||||||
John Balis <phobossystems@gmail.com>
|
John Balis <phobossystems@gmail.com>
|
||||||
JohnnyB <jboero@users.noreply.github.com>
|
|
||||||
Jonathan Soo <jcsoo@agora.com>
|
Jonathan Soo <jcsoo@agora.com>
|
||||||
Jonno <1160532+razodactyl@users.noreply.github.com>
|
Jonno <1160532+razodactyl@users.noreply.github.com>
|
||||||
Joonas Pihlajamaa <joonas.pihlajamaa@iki.fi>
|
Joonas Pihlajamaa <joonas.pihlajamaa@iki.fi>
|
||||||
Jose <34888496+Jerry-Master@users.noreply.github.com>
|
Jose <34888496+Jerry-Master@users.noreply.github.com>
|
||||||
Josh Bleecher Snyder <josharian@gmail.com>
|
Josh Bleecher Snyder <josharian@gmail.com>
|
||||||
Josscii <jossciiweiyi@gmail.com>
|
|
||||||
Judd <foldl@users.noreply.github.com>
|
Judd <foldl@users.noreply.github.com>
|
||||||
Jumper775 <78500318+jumpers775@users.noreply.github.com>
|
Jumper775 <78500318+jumpers775@users.noreply.github.com>
|
||||||
Jun Hee Yoo <contact.jhyoo@gmail.com>
|
|
||||||
Junil Kim <logyourself@gmail.com>
|
|
||||||
Justina Cho <justcho5@gmail.com>
|
|
||||||
Justine Tunney <jtunney@gmail.com>
|
Justine Tunney <jtunney@gmail.com>
|
||||||
Justine Tunney <jtunney@mozilla.com>
|
|
||||||
KITAITI Makoto <KitaitiMakoto@gmail.com>
|
|
||||||
KP Kaiser <kirk@zothcorp.com>
|
KP Kaiser <kirk@zothcorp.com>
|
||||||
Kamilake <exjang0@gmail.com>
|
Kamilake <exjang0@gmail.com>
|
||||||
Karol Kontny <82021046+kkontny@users.noreply.github.com>
|
|
||||||
Karthick <j.karthic2004@gmail.com>
|
|
||||||
Kartik Saranathan <278928+Kartiku@users.noreply.github.com>
|
Kartik Saranathan <278928+Kartiku@users.noreply.github.com>
|
||||||
Kasumi <90275229+kasumi-1@users.noreply.github.com>
|
Kasumi <90275229+kasumi-1@users.noreply.github.com>
|
||||||
Kawrakow <48489457+ikawrakow@users.noreply.github.com>
|
Kawrakow <48489457+ikawrakow@users.noreply.github.com>
|
||||||
Kendrick Taylor <kendrick@circuitsix.com>
|
|
||||||
Kevin Brothaler <admin@digipom.com>
|
Kevin Brothaler <admin@digipom.com>
|
||||||
Kevin Gibbons <bakkot@gmail.com>
|
|
||||||
Konosuke Sakai <konosuke@konosuke.work>
|
|
||||||
Konstantin Zhuravlyov <konstantin.zhuravlyov@amd.com>
|
Konstantin Zhuravlyov <konstantin.zhuravlyov@amd.com>
|
||||||
Kreijstal <rainb@tfwno.gf>
|
Kreijstal <rainb@tfwno.gf>
|
||||||
Kylin <56434533+KyL0N@users.noreply.github.com>
|
Kylin <56434533+KyL0N@users.noreply.github.com>
|
||||||
@ -231,110 +147,56 @@ Luis Herrera <herrera-luis@users.noreply.github.com>
|
|||||||
Lukas Rist <glaslos@gmail.com>
|
Lukas Rist <glaslos@gmail.com>
|
||||||
M. A. Ali <73258591+MightyStud@users.noreply.github.com>
|
M. A. Ali <73258591+MightyStud@users.noreply.github.com>
|
||||||
M. Eren Akbiyik <erenakbiyik@gmail.com>
|
M. Eren Akbiyik <erenakbiyik@gmail.com>
|
||||||
Ma Mingfei <mingfei.ma@intel.com>
|
|
||||||
Maciek <maciek.mab122@gmail.com>
|
Maciek <maciek.mab122@gmail.com>
|
||||||
Mahesh Madhav <67384846+heshpdx@users.noreply.github.com>
|
|
||||||
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
|
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
|
||||||
Mark Karpelès <MagicalTux@users.noreply.github.com>
|
|
||||||
Mark Zhuang <zhuangqiubin@gmail.com>
|
|
||||||
Markus Tavenrath <mtavenrath@users.noreply.github.com>
|
|
||||||
Martin Delille <martin@delille.org>
|
|
||||||
Martin Warnaar <martinwarnaar@gmail.com>
|
Martin Warnaar <martinwarnaar@gmail.com>
|
||||||
Masaya, Kato <62578291+msy-kato@users.noreply.github.com>
|
|
||||||
Matheus de Sousa <23645013+keyehzy@users.noreply.github.com>
|
Matheus de Sousa <23645013+keyehzy@users.noreply.github.com>
|
||||||
Mathieu Baudier <mbaudier@argeo.org>
|
|
||||||
Mathijs de Bruin <mathijs@mathijsfietst.nl>
|
Mathijs de Bruin <mathijs@mathijsfietst.nl>
|
||||||
Matija Pevec <mightymatth@users.noreply.github.com>
|
Matija Pevec <mightymatth@users.noreply.github.com>
|
||||||
Matt Stephenson <mstephenson6@users.noreply.github.com>
|
|
||||||
Max Krasnyansky <max.krasnyansky@gmail.com>
|
|
||||||
Max Krasnyansky <quic_maxk@quicinc.com>
|
|
||||||
Maximiliano Levi <8160966+maxilevi@users.noreply.github.com>
|
Maximiliano Levi <8160966+maxilevi@users.noreply.github.com>
|
||||||
Meng, Hengyu <hengyu.meng@intel.com>
|
Meng, Hengyu <hengyu.meng@intel.com>
|
||||||
Mengqing Cao <cmq0113@163.com>
|
|
||||||
Michael Podvitskiy <podvitskiymichael@gmail.com>
|
Michael Podvitskiy <podvitskiymichael@gmail.com>
|
||||||
Michael Rienstra <mrienstra@gmail.com>
|
Michael Rienstra <mrienstra@gmail.com>
|
||||||
Mikhail Grigorev <sleuthhound@gmail.com>
|
Mikhail Grigorev <sleuthhound@gmail.com>
|
||||||
Mohammadreza Hendiani <hendiani.mohammadreza@gmail.com>
|
Mohammadreza Hendiani <hendiani.mohammadreza@gmail.com>
|
||||||
Mohit Agarwal <mohit@sdf.org>
|
Mohit Agarwal <mohit@sdf.org>
|
||||||
Molly Sophia <mollysophia379@gmail.com>
|
|
||||||
Murilo Santana <mvrilo@gmail.com>
|
Murilo Santana <mvrilo@gmail.com>
|
||||||
NETZkultur GmbH <mulholland@netzkultur.de>
|
|
||||||
Natsu <chino@hotococoa.moe>
|
|
||||||
Neil Chudleigh <nchudleigh@users.noreply.github.com>
|
Neil Chudleigh <nchudleigh@users.noreply.github.com>
|
||||||
Neo Zhang <14088817+arthw@users.noreply.github.com>
|
|
||||||
Neo Zhang Jianyu <jianyu.zhang@intel.com>
|
Neo Zhang Jianyu <jianyu.zhang@intel.com>
|
||||||
Neuman Vong <neuman.vong@gmail.com>
|
Neuman Vong <neuman.vong@gmail.com>
|
||||||
Nicholai Tukanov <nicholaitukanov@gmail.com>
|
|
||||||
Nicholas Albion <nalbion@yahoo.com>
|
Nicholas Albion <nalbion@yahoo.com>
|
||||||
Nico Bosshard <nico@bosshome.ch>
|
|
||||||
Nicolò Scipione <nicolo.scipione@codeplay.com>
|
|
||||||
Niels Mayer <Niels.Mayer@gmail.com>
|
Niels Mayer <Niels.Mayer@gmail.com>
|
||||||
Nikita Sarychev <42014488+sARY77@users.noreply.github.com>
|
|
||||||
Nikolaj Olsson <nikse.dk@gmail.com>
|
|
||||||
Okabintaro <103938900+Okabintaro@users.noreply.github.com>
|
Okabintaro <103938900+Okabintaro@users.noreply.github.com>
|
||||||
Oleg Sidorov <me@whitebox.io>
|
Oleg Sidorov <me@whitebox.io>
|
||||||
Oleg Sidorov <oleg@sidorov.nl>
|
Oleg Sidorov <oleg@sidorov.nl>
|
||||||
Olivier Chafik <ochafik@users.noreply.github.com>
|
|
||||||
Ondrej Kokes <ondrej.kokes@gmail.com>
|
Ondrej Kokes <ondrej.kokes@gmail.com>
|
||||||
Ouadie EL FAROUKI <ouadie.elfarouki@codeplay.com>
|
Ouadie EL FAROUKI <ouadie.elfarouki@codeplay.com>
|
||||||
PAB <pierreantoine.bannier@gmail.com>
|
|
||||||
Paul Tsochantaris <ptsochantaris@icloud.com>
|
Paul Tsochantaris <ptsochantaris@icloud.com>
|
||||||
Pedro Probst <pprobst@insiberia.net>
|
|
||||||
Peng <hzp1024@qq.com>
|
|
||||||
Peter <peter277@users.noreply.github.com>
|
|
||||||
Philipp Zabel <philipp.zabel@gmail.com>
|
Philipp Zabel <philipp.zabel@gmail.com>
|
||||||
Philippe Normand <phil@base-art.net>
|
Philippe Normand <phil@base-art.net>
|
||||||
Philippe Normand <philn@igalia.com>
|
|
||||||
Plamen Minev <pacominev@gmail.com>
|
|
||||||
Prashant Vithule <119530321+Vithulep@users.noreply.github.com>
|
|
||||||
Przemysław Pawełczyk <przemoc@gmail.com>
|
Przemysław Pawełczyk <przemoc@gmail.com>
|
||||||
Qianhe Chen <54462604+chenqianhe@users.noreply.github.com>
|
Qianhe Chen <54462604+chenqianhe@users.noreply.github.com>
|
||||||
R0CKSTAR <xiaodong.ye@mthreads.com>
|
|
||||||
R0CKSTAR <yeahdongcn@gmail.com>
|
|
||||||
Radoslav Gerganov <rgerganov@gmail.com>
|
|
||||||
Radosław Gryta <radek.gryta@gmail.com>
|
Radosław Gryta <radek.gryta@gmail.com>
|
||||||
Rahul Vadhyar <107788610+RahulVadhyar@users.noreply.github.com>
|
|
||||||
Raiya Araki <83504221+rai62@users.noreply.github.com>
|
|
||||||
Reinforce-II <fate@eastal.com>
|
Reinforce-II <fate@eastal.com>
|
||||||
Reinis Muiznieks <muiznieks.reinis@gmail.com>
|
Reinis Muiznieks <muiznieks.reinis@gmail.com>
|
||||||
RelatedTitle <r3latedtitle@gmail.com>
|
RelatedTitle <r3latedtitle@gmail.com>
|
||||||
Rémy Oudompheng <oudomphe@phare.normalesup.org>
|
|
||||||
RhinoDevel <RhinoDevel@users.noreply.github.com>
|
RhinoDevel <RhinoDevel@users.noreply.github.com>
|
||||||
Rich Jones <miserlou@gmail.com>
|
Rich Jones <miserlou@gmail.com>
|
||||||
Robert Ormandi <52251610+ormandi@users.noreply.github.com>
|
|
||||||
Robin <robin.xw@hotmail.com>
|
Robin <robin.xw@hotmail.com>
|
||||||
Roddur Dasgupta <roddurd@gmail.com>
|
Roddur Dasgupta <roddurd@gmail.com>
|
||||||
Roland Rabien <figbug@gmail.com>
|
Roland Rabien <figbug@gmail.com>
|
||||||
Romain Biessy <romain.biessy@codeplay.com>
|
|
||||||
Ronsor <ronsor@ronsor.pw>
|
|
||||||
Rotem Dan <rotemdan@gmail.com>
|
Rotem Dan <rotemdan@gmail.com>
|
||||||
Ryan Hitchman <hitchmanr@gmail.com>
|
Ryan Hitchman <hitchmanr@gmail.com>
|
||||||
Ryan Metcalfe <107415876+RyanMetcalfeInt8@users.noreply.github.com>
|
Ryan Metcalfe <107415876+RyanMetcalfeInt8@users.noreply.github.com>
|
||||||
RyanChang <ftes90015@gmail.com>
|
RyanChang <ftes90015@gmail.com>
|
||||||
SRHMorris <69468379+SRHMorris@users.noreply.github.com>
|
|
||||||
SXX <sxx1136965276@gmail.com>
|
|
||||||
Sacha Arbonel <sacha.arbonel@hotmail.fr>
|
|
||||||
Salman Faroz <stsfaroz@gmail.com>
|
|
||||||
Salvatore Mesoraca <s.mesoraca16@gmail.com>
|
|
||||||
Sam <49637763+Onlyartist9@users.noreply.github.com>
|
Sam <49637763+Onlyartist9@users.noreply.github.com>
|
||||||
Sam Pullara <spullara@gmail.com>
|
Sam Pullara <spullara@gmail.com>
|
||||||
Samuel Durante <44513615+samueldurantes@users.noreply.github.com>
|
|
||||||
Sanchit Gandhi <93869735+sanchit-gandhi@users.noreply.github.com>
|
Sanchit Gandhi <93869735+sanchit-gandhi@users.noreply.github.com>
|
||||||
Sandro Hanea <40202887+sandrohanea@users.noreply.github.com>
|
|
||||||
Sergio López <slp@redhat.com>
|
|
||||||
Sergio López <slp@sinrega.org>
|
Sergio López <slp@sinrega.org>
|
||||||
Shanshan Shen <467638484@qq.com>
|
|
||||||
Shijie <821898965@qq.com>
|
|
||||||
Shupei Fan <dymarkfan@outlook.com>
|
|
||||||
Siddharth Ramakrishnan <srr2141@columbia.edu>
|
Siddharth Ramakrishnan <srr2141@columbia.edu>
|
||||||
Sigbjørn Skjæret <sigbjorn.skjaeret@scala.com>
|
|
||||||
Simon Moisselin <simon.moisstoll@gmail.com>
|
Simon Moisselin <simon.moisstoll@gmail.com>
|
||||||
Sindre Sorhus <sindresorhus@gmail.com>
|
Sindre Sorhus <sindresorhus@gmail.com>
|
||||||
Slava Primenko <primenko.s@gmail.com>
|
Slava Primenko <primenko.s@gmail.com>
|
||||||
Srihari-mcw <96763064+Srihari-mcw@users.noreply.github.com>
|
|
||||||
Stavros Panakakis <53979866+Stavrospanakakis@users.noreply.github.com>
|
|
||||||
Stefan Sydow <s.sydow@heinlein-video.de>
|
|
||||||
Stefan Sydow <stefan@sydow.email>
|
|
||||||
Syahmi Azhar <prsyahmi@gmail.com>
|
Syahmi Azhar <prsyahmi@gmail.com>
|
||||||
Syed Jafri <syedjafri97@gmail.com>
|
Syed Jafri <syedjafri97@gmail.com>
|
||||||
Sơn Phan Trung <phantrungson17@gmail.com>
|
Sơn Phan Trung <phantrungson17@gmail.com>
|
||||||
@ -343,63 +205,37 @@ Takeshi Inoue <inoue.takeshi@gmail.com>
|
|||||||
Tamotsu Takahashi <ttakah+github@gmail.com>
|
Tamotsu Takahashi <ttakah+github@gmail.com>
|
||||||
Taras Glek <taras@thegp.com>
|
Taras Glek <taras@thegp.com>
|
||||||
Tauseef Mohiuddin <35351464+tauseefmohammed2@users.noreply.github.com>
|
Tauseef Mohiuddin <35351464+tauseefmohammed2@users.noreply.github.com>
|
||||||
Thamster <Thamster@users.noreply.github.com>
|
|
||||||
Thijs Raymakers <thijs@raymakers.nl>
|
Thijs Raymakers <thijs@raymakers.nl>
|
||||||
Thomas Fitzsimmons <fitzsim@fitzsim.org>
|
Thomas Fitzsimmons <fitzsim@fitzsim.org>
|
||||||
Tiago Fassoni <tiagofassoni@users.noreply.github.com>
|
Tiago Fassoni <tiagofassoni@users.noreply.github.com>
|
||||||
Tienshiao Ma <tienshiao@tienshiao.org>
|
Tienshiao Ma <tienshiao@tienshiao.org>
|
||||||
Tim Miller <drasticactions@users.noreply.github.com>
|
|
||||||
Timothy Cronin <40186632+4imothy@users.noreply.github.com>
|
Timothy Cronin <40186632+4imothy@users.noreply.github.com>
|
||||||
Tobrun <tobrun.van.nuland@gmail.com>
|
Tobrun <tobrun.van.nuland@gmail.com>
|
||||||
Todd <taf2@users.noreply.github.com>
|
Todd <taf2@users.noreply.github.com>
|
||||||
Toliver <teejae@gmail.com>
|
|
||||||
Tong Li <31761981+litongjava@users.noreply.github.com>
|
Tong Li <31761981+litongjava@users.noreply.github.com>
|
||||||
Tony Wasserka <4840017+neobrain@users.noreply.github.com>
|
|
||||||
Topping1 <78745143+Topping1@users.noreply.github.com>
|
Topping1 <78745143+Topping1@users.noreply.github.com>
|
||||||
Travis Cline <travis.cline@gmail.com>
|
Travis Cline <travis.cline@gmail.com>
|
||||||
UEXTM.com <84163508+uextm@users.noreply.github.com>
|
UEXTM.com <84163508+uextm@users.noreply.github.com>
|
||||||
UsernamesLame <156965854+UsernamesLame@users.noreply.github.com>
|
|
||||||
Vadim Peretokin <vperetokin@hey.com>
|
Vadim Peretokin <vperetokin@hey.com>
|
||||||
Valentin Gosu <1454649+valenting@users.noreply.github.com>
|
Valentin Gosu <1454649+valenting@users.noreply.github.com>
|
||||||
Vin Misra <vinith@alum.mit.edu>
|
|
||||||
Vulcan <93451215+trholding@users.noreply.github.com>
|
Vulcan <93451215+trholding@users.noreply.github.com>
|
||||||
WhiteOlivierus <36532695+WhiteOlivierus@users.noreply.github.com>
|
WhiteOlivierus <36532695+WhiteOlivierus@users.noreply.github.com>
|
||||||
William Tambellini <william.tambellini@gmail.com>
|
|
||||||
William Tambellini <wtambellini@sdl.com>
|
|
||||||
Wilson Silva <wilson.dsigns@gmail.com>
|
|
||||||
Xiang (Kevin) Li <kevinli020508@gmail.com>
|
Xiang (Kevin) Li <kevinli020508@gmail.com>
|
||||||
Xiao-Yong Jin <jinxiaoyong@gmail.com>
|
Xiao-Yong Jin <jinxiaoyong@gmail.com>
|
||||||
XiaotaoChen <chenxiaotao1234@gmail.com>
|
XiaotaoChen <chenxiaotao1234@gmail.com>
|
||||||
Xingchen Song(宋星辰) <xingchensong1996@163.com>
|
|
||||||
Xinpeng Dou <81913537+Dou-Git@users.noreply.github.com>
|
|
||||||
Xuan Son Nguyen <thichthat@gmail.com>
|
|
||||||
Yajing Tang <phillis@google.com>
|
Yajing Tang <phillis@google.com>
|
||||||
Yang Shen <aplshenyang@gmail.com>
|
Yang Shen <aplshenyang@gmail.com>
|
||||||
Yunès <jean.baptiste.yunes@free.fr>
|
Yunès <jean.baptiste.yunes@free.fr>
|
||||||
Yuri Khrustalev <ykhrustalev@users.noreply.github.com>
|
|
||||||
Yusuf Redžić <48274562+redzic@users.noreply.github.com>
|
|
||||||
ZaBlazzingZephyrus <119159668+blazingzephyr@users.noreply.github.com>
|
ZaBlazzingZephyrus <119159668+blazingzephyr@users.noreply.github.com>
|
||||||
Zhenwei Jin <109658203+kylo5aby@users.noreply.github.com>
|
|
||||||
Zhiyuan Li <lizhiyuan@uniartisan.com>
|
|
||||||
Zhiyuan Li <uniartisan2017@gmail.com>
|
|
||||||
Zigfrid Zvezdin <ziggerZZ@gmail.com>
|
Zigfrid Zvezdin <ziggerZZ@gmail.com>
|
||||||
Zollner <24618122+Zolliner@users.noreply.github.com>
|
Zollner <24618122+Zolliner@users.noreply.github.com>
|
||||||
a3sh <38979186+A3shTnT@users.noreply.github.com>
|
|
||||||
ag2s20150909 <19373730+ag2s20150909@users.noreply.github.com>
|
|
||||||
agray3 <agray3@users.noreply.github.com>
|
|
||||||
ai-at-home <149282006+ai-at-home@users.noreply.github.com>
|
ai-at-home <149282006+ai-at-home@users.noreply.github.com>
|
||||||
aldorof <aldorof@users.noreply.github.com>
|
|
||||||
alonfaraj <alonfaraj@gmail.com>
|
alonfaraj <alonfaraj@gmail.com>
|
||||||
amd-dwang <dong.wang@amd.com>
|
|
||||||
amritahs-ibm <amritahs@linux.vnet.ibm.com>
|
|
||||||
andypayne <apayne@gmail.com>
|
andypayne <apayne@gmail.com>
|
||||||
ardfork <134447697+ardfork@users.noreply.github.com>
|
ardfork <134447697+ardfork@users.noreply.github.com>
|
||||||
arizhih <40765267+arizhih@users.noreply.github.com>
|
|
||||||
automaticcat <daogiatuank54@gmail.com>
|
automaticcat <daogiatuank54@gmail.com>
|
||||||
bandoti <141645996+bandoti@users.noreply.github.com>
|
|
||||||
be-next <jerome.ramette@gmail.com>
|
be-next <jerome.ramette@gmail.com>
|
||||||
bert hubert <bert@hubertnet.nl>
|
bert hubert <bert@hubertnet.nl>
|
||||||
billyct <billy_allen@126.com>
|
|
||||||
bmwl <brian.marshall@tolko.com>
|
bmwl <brian.marshall@tolko.com>
|
||||||
bobqianic <129547291+bobqianic@users.noreply.github.com>
|
bobqianic <129547291+bobqianic@users.noreply.github.com>
|
||||||
bocytko <bocytko+github@gmail.com>
|
bocytko <bocytko+github@gmail.com>
|
||||||
@ -412,9 +248,7 @@ byte-6174 <88070277+byte-6174@users.noreply.github.com>
|
|||||||
cdosoftei <ciprian.dosoftei@gmail.com>
|
cdosoftei <ciprian.dosoftei@gmail.com>
|
||||||
clach04 <Chris.Clark@actian.com>
|
clach04 <Chris.Clark@actian.com>
|
||||||
compilade <113953597+compilade@users.noreply.github.com>
|
compilade <113953597+compilade@users.noreply.github.com>
|
||||||
compilade <git@compilade.net>
|
|
||||||
conradg <conradjgodfrey@gmail.com>
|
conradg <conradjgodfrey@gmail.com>
|
||||||
crummyh <elijah@crums.us>
|
|
||||||
ddpasa <112642920+ddpasa@users.noreply.github.com>
|
ddpasa <112642920+ddpasa@users.noreply.github.com>
|
||||||
denersc <denerstassun@gmail.com>
|
denersc <denerstassun@gmail.com>
|
||||||
dscripka <dscripka@users.noreply.github.com>
|
dscripka <dscripka@users.noreply.github.com>
|
||||||
@ -422,55 +256,28 @@ duthils <duthils@duthils.net>
|
|||||||
ecneladis <ecneladis@users.noreply.github.com>
|
ecneladis <ecneladis@users.noreply.github.com>
|
||||||
faker <nspyia2002@gmail.com>
|
faker <nspyia2002@gmail.com>
|
||||||
fitzsim <fitzsim@fitzsim.org>
|
fitzsim <fitzsim@fitzsim.org>
|
||||||
fj-y-saito <85871716+fj-y-saito@users.noreply.github.com>
|
|
||||||
fraxy-v <65565042+fraxy-v@users.noreply.github.com>
|
fraxy-v <65565042+fraxy-v@users.noreply.github.com>
|
||||||
genevera (she/her) <genevera@users.noreply.github.com>
|
genevera (she/her) <genevera@users.noreply.github.com>
|
||||||
geniusnut <geniusnut@gmail.com>
|
geniusnut <geniusnut@gmail.com>
|
||||||
gilbertgong <gilbert.gong@gmail.com>
|
|
||||||
gn64 <yukikaze.jp@gmail.com>
|
|
||||||
goldwaving <77494627+goldwaving@users.noreply.github.com>
|
|
||||||
greeshmay <greeshmay@gmail.com>
|
greeshmay <greeshmay@gmail.com>
|
||||||
haopeng <657407891@qq.com>
|
|
||||||
hipudding <huafengchun@gmail.com>
|
|
||||||
hsinhoyeh <yhh92u@gmail.com>
|
|
||||||
hydai <z54981220@gmail.com>
|
hydai <z54981220@gmail.com>
|
||||||
iamthad <thadeus.j.fleming@gmail.com>
|
iamthad <thadeus.j.fleming@gmail.com>
|
||||||
issixx <46835150+issixx@users.noreply.github.com>
|
|
||||||
james wolf <contractorwolf@hotmail.com>
|
james wolf <contractorwolf@hotmail.com>
|
||||||
jdomke <28772296+jdomke@users.noreply.github.com>
|
|
||||||
jettoblack <jettoblack@gmail.com>
|
|
||||||
jiez <373447296@qq.com>
|
|
||||||
joecryptotoo <80373433+joecryptotoo@users.noreply.github.com>
|
joecryptotoo <80373433+joecryptotoo@users.noreply.github.com>
|
||||||
jorismertz <35079666+jorismertz@users.noreply.github.com>
|
jorismertz <35079666+jorismertz@users.noreply.github.com>
|
||||||
junchao-loongson <68935141+junchao-loongson@users.noreply.github.com>
|
|
||||||
junkfood <69683722+JunkFood02@users.noreply.github.com>
|
junkfood <69683722+JunkFood02@users.noreply.github.com>
|
||||||
jwijffels <jwijffels@bnosac.be>
|
jwijffels <jwijffels@bnosac.be>
|
||||||
k.h.lai <adrian.k.h.lai@outlook.com>
|
|
||||||
kamranjon <kamranjon@gmail.com>
|
kamranjon <kamranjon@gmail.com>
|
||||||
katsu560 <katsu560oo-@docomo.ne.jp>
|
katsu560 <katsu560oo-@docomo.ne.jp>
|
||||||
kennethge <57784063+kenneth-ge@users.noreply.github.com>
|
kennethge <57784063+kenneth-ge@users.noreply.github.com>
|
||||||
keyehzy <msamuel@aluno.puc-rio.br>
|
keyehzy <msamuel@aluno.puc-rio.br>
|
||||||
kunnis <kunnis@users.noreply.github.com>
|
|
||||||
l3utterfly <gc.pthzfoldr@gmail.com>
|
|
||||||
leejet <leejet714@gmail.com>
|
leejet <leejet714@gmail.com>
|
||||||
leo-pony <nengjunma@outlook.com>
|
|
||||||
lhez <quic_lih@quicinc.com>
|
|
||||||
litong <31761981+litongjava@users.noreply.github.com>
|
litong <31761981+litongjava@users.noreply.github.com>
|
||||||
liuwei-git <14815172+liuwei-git@users.noreply.github.com>
|
|
||||||
lnyan <lkwq007@gmail.com>
|
lnyan <lkwq007@gmail.com>
|
||||||
luoyu-intel <yu.luo@intel.com>
|
|
||||||
m.bell <m.bell@techsmith.com>
|
m.bell <m.bell@techsmith.com>
|
||||||
mahorozte <41834471+mahorozte@users.noreply.github.com>
|
|
||||||
mashizora <30516315+mashizora@users.noreply.github.com>
|
|
||||||
matt23654 <matthew.webber@protonmail.com>
|
|
||||||
matteo <matteogeniaccio@yahoo.it>
|
|
||||||
mgrachten <maarten@grachten.eu>
|
|
||||||
mkiol <mkiol@users.noreply.github.com>
|
mkiol <mkiol@users.noreply.github.com>
|
||||||
mky_coder <47767389+mkycoder@users.noreply.github.com>
|
|
||||||
novag <7754358+novag@users.noreply.github.com>
|
novag <7754358+novag@users.noreply.github.com>
|
||||||
pajowu <pajowu@pajowu.de>
|
pajowu <pajowu@pajowu.de>
|
||||||
pengxin99 <pengxin.yuan@intel.com>
|
|
||||||
petterreinholdtsen <pere-github@hungry.com>
|
|
||||||
polarmoon <90010972+polarmoon@users.noreply.github.com>
|
polarmoon <90010972+polarmoon@users.noreply.github.com>
|
||||||
rlapray <lapray.romain@gmail.com>
|
rlapray <lapray.romain@gmail.com>
|
||||||
sandrohanea <40202887+sandrohanea@users.noreply.github.com>
|
sandrohanea <40202887+sandrohanea@users.noreply.github.com>
|
||||||
@ -480,31 +287,15 @@ shikokuchuo <53399081+shikokuchuo@users.noreply.github.com>
|
|||||||
slaren <slarengh@gmail.com>
|
slaren <slarengh@gmail.com>
|
||||||
slashlib <slashlib@users.noreply.github.com>
|
slashlib <slashlib@users.noreply.github.com>
|
||||||
snadampal <87143774+snadampal@users.noreply.github.com>
|
snadampal <87143774+snadampal@users.noreply.github.com>
|
||||||
someone13574 <81528246+someone13574@users.noreply.github.com>
|
|
||||||
st-gr <38470677+st-gr@users.noreply.github.com>
|
st-gr <38470677+st-gr@users.noreply.github.com>
|
||||||
stduhpf <stephduh@live.fr>
|
|
||||||
stormofice <58337328+stormofice@users.noreply.github.com>
|
|
||||||
texmex76 <40733439+texmex76@users.noreply.github.com>
|
texmex76 <40733439+texmex76@users.noreply.github.com>
|
||||||
thefinaldegree <thefinaldegree@gmail.com>
|
thefinaldegree <thefinaldegree@gmail.com>
|
||||||
thewh1teagle <61390950+thewh1teagle@users.noreply.github.com>
|
|
||||||
toboil-features <160222185+toboil-features@users.noreply.github.com>
|
|
||||||
trixirt <trix@redhat.com>
|
trixirt <trix@redhat.com>
|
||||||
ulatekh <ulatekh@yahoo.com>
|
ulatekh <ulatekh@yahoo.com>
|
||||||
undef <undefdev@gmail.com>
|
undef <undefdev@gmail.com>
|
||||||
uvos <devnull@uvos.xyz>
|
|
||||||
uvos <philipp@uvos.xyz>
|
|
||||||
valVk <valVk@users.noreply.github.com>
|
|
||||||
venkr <venkateshrameshkumar+1@gmail.com>
|
venkr <venkateshrameshkumar+1@gmail.com>
|
||||||
vicalloy <zbirder@gmail.com>
|
vicalloy <zbirder@gmail.com>
|
||||||
wangshuai09 <391746016@qq.com>
|
|
||||||
woachk <24752637+woachk@users.noreply.github.com>
|
|
||||||
xctan <axunlei@gmail.com>
|
|
||||||
xdrudis <xavierdrudis@yahoo.es>
|
xdrudis <xavierdrudis@yahoo.es>
|
||||||
yuri@FreeBSD <yuri@FreeBSD>
|
|
||||||
zhangjixiong <code.zjx@gmail.com>
|
|
||||||
zhentaoyu <zhentao.yu@intel.com>
|
|
||||||
zhouwg <6889919+zhouwg@users.noreply.github.com>
|
zhouwg <6889919+zhouwg@users.noreply.github.com>
|
||||||
zhouwg <zhouwg2000@gmail.com>
|
|
||||||
谢乃闻 <sienaiwun@users.noreply.github.com>
|
|
||||||
布客飞龙 <562826179@qq.com>
|
布客飞龙 <562826179@qq.com>
|
||||||
Артём Земляк <azemlyak@smart-consulting.ru>
|
Артём Земляк <azemlyak@smart-consulting.ru>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
cmake_minimum_required(VERSION 3.5) # for add_link_options and implicit target directories.
|
cmake_minimum_required(VERSION 3.5) # for add_link_options and implicit target directories.
|
||||||
project("whisper.cpp" C CXX)
|
project("whisper.cpp" C CXX)
|
||||||
project("whisper.cpp" VERSION 1.7.4)
|
project("whisper.cpp" VERSION 1.6.2)
|
||||||
include(CheckIncludeFileCXX)
|
include(CheckIncludeFileCXX)
|
||||||
|
|
||||||
set(SOVERSION 1)
|
set(SOVERSION 1)
|
||||||
@ -120,10 +120,7 @@ whisper_option_depr(WARNING WHISPER_SYCL_F16 GGML_SYCL_F16)
|
|||||||
# build the library
|
# build the library
|
||||||
#
|
#
|
||||||
|
|
||||||
if (NOT TARGET ggml)
|
add_subdirectory(ggml)
|
||||||
add_subdirectory(ggml)
|
|
||||||
# ... otherwise assume ggml is added by a parent CMakeLists.txt
|
|
||||||
endif()
|
|
||||||
add_subdirectory(src)
|
add_subdirectory(src)
|
||||||
|
|
||||||
#
|
#
|
||||||
@ -164,6 +161,18 @@ install(FILES ${CMAKE_CURRENT_BINARY_DIR}/whisper-config.cmake
|
|||||||
${CMAKE_CURRENT_BINARY_DIR}/whisper-version.cmake
|
${CMAKE_CURRENT_BINARY_DIR}/whisper-version.cmake
|
||||||
DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/whisper)
|
DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/whisper)
|
||||||
|
|
||||||
|
install(
|
||||||
|
FILES convert-hf-to-gguf.py
|
||||||
|
PERMISSIONS
|
||||||
|
OWNER_READ
|
||||||
|
OWNER_WRITE
|
||||||
|
OWNER_EXECUTE
|
||||||
|
GROUP_READ
|
||||||
|
GROUP_EXECUTE
|
||||||
|
WORLD_READ
|
||||||
|
WORLD_EXECUTE
|
||||||
|
DESTINATION ${CMAKE_INSTALL_BINDIR})
|
||||||
|
|
||||||
configure_file(cmake/whisper.pc.in
|
configure_file(cmake/whisper.pc.in
|
||||||
"${CMAKE_CURRENT_BINARY_DIR}/whisper.pc"
|
"${CMAKE_CURRENT_BINARY_DIR}/whisper.pc"
|
||||||
@ONLY)
|
@ONLY)
|
||||||
|
@ -14,6 +14,46 @@ let package = Package(
|
|||||||
.library(name: "whisper", targets: ["whisper"]),
|
.library(name: "whisper", targets: ["whisper"]),
|
||||||
],
|
],
|
||||||
targets: [
|
targets: [
|
||||||
.systemLibrary(name: "whisper", pkgConfig: "whisper"),
|
.target(
|
||||||
]
|
name: "whisper",
|
||||||
|
path: ".",
|
||||||
|
exclude: [
|
||||||
|
"bindings",
|
||||||
|
"cmake",
|
||||||
|
"coreml",
|
||||||
|
"examples",
|
||||||
|
"extra",
|
||||||
|
"models",
|
||||||
|
"samples",
|
||||||
|
"tests",
|
||||||
|
"CMakeLists.txt",
|
||||||
|
"Makefile"
|
||||||
|
],
|
||||||
|
sources: [
|
||||||
|
"ggml/src/ggml.c",
|
||||||
|
"src/whisper.cpp",
|
||||||
|
"ggml/src/ggml-alloc.c",
|
||||||
|
"ggml/src/ggml-backend.c",
|
||||||
|
"ggml/src/ggml-quants.c",
|
||||||
|
"ggml/src/ggml-metal.m"
|
||||||
|
],
|
||||||
|
resources: [.process("ggml-metal.metal")],
|
||||||
|
publicHeadersPath: "spm-headers",
|
||||||
|
cSettings: [
|
||||||
|
.unsafeFlags(["-Wno-shorten-64-to-32", "-O3", "-DNDEBUG"]),
|
||||||
|
.define("GGML_USE_ACCELERATE"),
|
||||||
|
.unsafeFlags(["-fno-objc-arc"]),
|
||||||
|
.define("GGML_USE_METAL")
|
||||||
|
// NOTE: NEW_LAPACK will required iOS version 16.4+
|
||||||
|
// We should consider add this in the future when we drop support for iOS 14
|
||||||
|
// (ref: ref: https://developer.apple.com/documentation/accelerate/1513264-cblas_sgemm?language=objc)
|
||||||
|
// .define("ACCELERATE_NEW_LAPACK"),
|
||||||
|
// .define("ACCELERATE_LAPACK_ILP64")
|
||||||
|
],
|
||||||
|
linkerSettings: [
|
||||||
|
.linkedFramework("Accelerate")
|
||||||
|
]
|
||||||
|
)
|
||||||
|
],
|
||||||
|
cxxLanguageStandard: .cxx11
|
||||||
)
|
)
|
||||||
|
399
README.md
399
README.md
@ -7,26 +7,21 @@
|
|||||||
[](https://conan.io/center/whisper-cpp)
|
[](https://conan.io/center/whisper-cpp)
|
||||||
[](https://www.npmjs.com/package/whisper.cpp/)
|
[](https://www.npmjs.com/package/whisper.cpp/)
|
||||||
|
|
||||||
> [!NOTE]
|
Stable: [v1.6.2](https://github.com/ggerganov/whisper.cpp/releases/tag/v1.6.0) / [Roadmap | F.A.Q.](https://github.com/ggerganov/whisper.cpp/discussions/126)
|
||||||
> New maintenance roadmap: https://github.com/ggerganov/whisper.cpp/discussions/2788
|
|
||||||
|
|
||||||
Stable: [v1.7.4](https://github.com/ggerganov/whisper.cpp/releases/tag/v1.7.4) / [Roadmap | F.A.Q.](https://github.com/ggerganov/whisper.cpp/discussions/126)
|
|
||||||
|
|
||||||
High-performance inference of [OpenAI's Whisper](https://github.com/openai/whisper) automatic speech recognition (ASR) model:
|
High-performance inference of [OpenAI's Whisper](https://github.com/openai/whisper) automatic speech recognition (ASR) model:
|
||||||
|
|
||||||
- Plain C/C++ implementation without dependencies
|
- Plain C/C++ implementation without dependencies
|
||||||
- Apple Silicon first-class citizen - optimized via ARM NEON, Accelerate framework, Metal and [Core ML](#core-ml-support)
|
- Apple Silicon first-class citizen - optimized via ARM NEON, Accelerate framework, Metal and [Core ML](https://github.com/ggerganov/whisper.cpp#core-ml-support)
|
||||||
- AVX intrinsics support for x86 architectures
|
- AVX intrinsics support for x86 architectures
|
||||||
- [VSX intrinsics support for POWER architectures](#power-vsx-intrinsics)
|
- VSX intrinsics support for POWER architectures
|
||||||
- Mixed F16 / F32 precision
|
- Mixed F16 / F32 precision
|
||||||
- [Integer quantization support](#quantization)
|
- [4-bit and 5-bit integer quantization support](https://github.com/ggerganov/whisper.cpp#quantization)
|
||||||
- Zero memory allocations at runtime
|
- Zero memory allocations at runtime
|
||||||
- [Vulkan support](#vulkan-gpu-support)
|
|
||||||
- Support for CPU-only inference
|
- Support for CPU-only inference
|
||||||
- [Efficient GPU support for NVIDIA](#nvidia-gpu-support)
|
- [Efficient GPU support for NVIDIA](https://github.com/ggerganov/whisper.cpp#nvidia-gpu-support-via-cublas)
|
||||||
- [OpenVINO Support](#openvino-support)
|
- [OpenVINO Support](https://github.com/ggerganov/whisper.cpp#openvino-support)
|
||||||
- [Ascend NPU Support](#ascend-npu-support)
|
- [C-style API](https://github.com/ggerganov/whisper.cpp/blob/master/whisper.h)
|
||||||
- [C-style API](https://github.com/ggerganov/whisper.cpp/blob/master/include/whisper.h)
|
|
||||||
|
|
||||||
Supported platforms:
|
Supported platforms:
|
||||||
|
|
||||||
@ -38,9 +33,9 @@ Supported platforms:
|
|||||||
- [x] [WebAssembly](examples/whisper.wasm)
|
- [x] [WebAssembly](examples/whisper.wasm)
|
||||||
- [x] Windows ([MSVC](https://github.com/ggerganov/whisper.cpp/blob/master/.github/workflows/build.yml#L117-L144) and [MinGW](https://github.com/ggerganov/whisper.cpp/issues/168)]
|
- [x] Windows ([MSVC](https://github.com/ggerganov/whisper.cpp/blob/master/.github/workflows/build.yml#L117-L144) and [MinGW](https://github.com/ggerganov/whisper.cpp/issues/168)]
|
||||||
- [x] [Raspberry Pi](https://github.com/ggerganov/whisper.cpp/discussions/166)
|
- [x] [Raspberry Pi](https://github.com/ggerganov/whisper.cpp/discussions/166)
|
||||||
- [x] [Docker](https://github.com/ggerganov/whisper.cpp/pkgs/container/whisper.cpp)
|
- [x] [docker](https://github.com/ggerganov/whisper.cpp/pkgs/container/whisper.cpp)
|
||||||
|
|
||||||
The entire high-level implementation of the model is contained in [whisper.h](include/whisper.h) and [whisper.cpp](src/whisper.cpp).
|
The entire high-level implementation of the model is contained in [whisper.h](whisper.h) and [whisper.cpp](whisper.cpp).
|
||||||
The rest of the code is part of the [`ggml`](https://github.com/ggerganov/ggml) machine learning library.
|
The rest of the code is part of the [`ggml`](https://github.com/ggerganov/ggml) machine learning library.
|
||||||
|
|
||||||
Having such a lightweight implementation of the model allows to easily integrate it in different platforms and applications.
|
Having such a lightweight implementation of the model allows to easily integrate it in different platforms and applications.
|
||||||
@ -56,6 +51,18 @@ On Apple Silicon, the inference runs fully on the GPU via Metal:
|
|||||||
|
|
||||||
https://github.com/ggerganov/whisper.cpp/assets/1991296/c82e8f86-60dc-49f2-b048-d2fdbd6b5225
|
https://github.com/ggerganov/whisper.cpp/assets/1991296/c82e8f86-60dc-49f2-b048-d2fdbd6b5225
|
||||||
|
|
||||||
|
Or you can even run it straight in the browser: [talk.wasm](examples/talk.wasm)
|
||||||
|
|
||||||
|
## Implementation details
|
||||||
|
|
||||||
|
- The core tensor operations are implemented in C ([ggml.h](ggml.h) / [ggml.c](ggml.c))
|
||||||
|
- The transformer model and the high-level C-style API are implemented in C++ ([whisper.h](whisper.h) / [whisper.cpp](whisper.cpp))
|
||||||
|
- Sample usage is demonstrated in [main.cpp](examples/main)
|
||||||
|
- Sample real-time audio transcription from the microphone is demonstrated in [stream.cpp](examples/stream)
|
||||||
|
- Various other examples are available in the [examples](examples) folder
|
||||||
|
|
||||||
|
The tensor operators are optimized heavily for Apple silicon CPUs. Depending on the computation size, Arm Neon SIMD intrinsics or CBLAS Accelerate framework routines are used. The latter are especially effective for bigger sizes since the Accelerate framework utilizes the special-purpose AMX coprocessor available in modern Apple products.
|
||||||
|
|
||||||
## Quick start
|
## Quick start
|
||||||
|
|
||||||
First clone the repository:
|
First clone the repository:
|
||||||
@ -64,38 +71,140 @@ First clone the repository:
|
|||||||
git clone https://github.com/ggerganov/whisper.cpp.git
|
git clone https://github.com/ggerganov/whisper.cpp.git
|
||||||
```
|
```
|
||||||
|
|
||||||
Navigate into the directory:
|
|
||||||
|
|
||||||
```
|
|
||||||
cd whisper.cpp
|
|
||||||
```
|
|
||||||
|
|
||||||
Then, download one of the Whisper [models](models/README.md) converted in [`ggml` format](#ggml-format). For example:
|
Then, download one of the Whisper [models](models/README.md) converted in [`ggml` format](#ggml-format). For example:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sh ./models/download-ggml-model.sh base.en
|
bash ./models/download-ggml-model.sh base.en
|
||||||
```
|
```
|
||||||
|
|
||||||
Now build the [whisper-cli](examples/cli) example and transcribe an audio file like this:
|
Now build the [main](examples/main) example and transcribe an audio file like this:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# build the project
|
# build the main example
|
||||||
cmake -B build
|
make
|
||||||
cmake --build build --config Release
|
|
||||||
|
|
||||||
# transcribe an audio file
|
# transcribe an audio file
|
||||||
./build/bin/whisper-cli -f samples/jfk.wav
|
./main -f samples/jfk.wav
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
For a quick demo, simply run `make base.en`.
|
For a quick demo, simply run `make base.en`:
|
||||||
|
|
||||||
|
```text
|
||||||
|
$ make base.en
|
||||||
|
|
||||||
|
cc -I. -O3 -std=c11 -pthread -DGGML_USE_ACCELERATE -c ggml.c -o ggml.o
|
||||||
|
c++ -I. -I./examples -O3 -std=c++11 -pthread -c whisper.cpp -o whisper.o
|
||||||
|
c++ -I. -I./examples -O3 -std=c++11 -pthread examples/main/main.cpp whisper.o ggml.o -o main -framework Accelerate
|
||||||
|
./main -h
|
||||||
|
|
||||||
|
usage: ./main [options] file0.wav file1.wav ...
|
||||||
|
|
||||||
|
options:
|
||||||
|
-h, --help [default] show this help message and exit
|
||||||
|
-t N, --threads N [4 ] number of threads to use during computation
|
||||||
|
-p N, --processors N [1 ] number of processors to use during computation
|
||||||
|
-ot N, --offset-t N [0 ] time offset in milliseconds
|
||||||
|
-on N, --offset-n N [0 ] segment index offset
|
||||||
|
-d N, --duration N [0 ] duration of audio to process in milliseconds
|
||||||
|
-mc N, --max-context N [-1 ] maximum number of text context tokens to store
|
||||||
|
-ml N, --max-len N [0 ] maximum segment length in characters
|
||||||
|
-sow, --split-on-word [false ] split on word rather than on token
|
||||||
|
-bo N, --best-of N [5 ] number of best candidates to keep
|
||||||
|
-bs N, --beam-size N [5 ] beam size for beam search
|
||||||
|
-wt N, --word-thold N [0.01 ] word timestamp probability threshold
|
||||||
|
-et N, --entropy-thold N [2.40 ] entropy threshold for decoder fail
|
||||||
|
-lpt N, --logprob-thold N [-1.00 ] log probability threshold for decoder fail
|
||||||
|
-debug, --debug-mode [false ] enable debug mode (eg. dump log_mel)
|
||||||
|
-tr, --translate [false ] translate from source language to english
|
||||||
|
-di, --diarize [false ] stereo audio diarization
|
||||||
|
-tdrz, --tinydiarize [false ] enable tinydiarize (requires a tdrz model)
|
||||||
|
-nf, --no-fallback [false ] do not use temperature fallback while decoding
|
||||||
|
-otxt, --output-txt [false ] output result in a text file
|
||||||
|
-ovtt, --output-vtt [false ] output result in a vtt file
|
||||||
|
-osrt, --output-srt [false ] output result in a srt file
|
||||||
|
-olrc, --output-lrc [false ] output result in a lrc file
|
||||||
|
-owts, --output-words [false ] output script for generating karaoke video
|
||||||
|
-fp, --font-path [/System/Library/Fonts/Supplemental/Courier New Bold.ttf] path to a monospace font for karaoke video
|
||||||
|
-ocsv, --output-csv [false ] output result in a CSV file
|
||||||
|
-oj, --output-json [false ] output result in a JSON file
|
||||||
|
-ojf, --output-json-full [false ] include more information in the JSON file
|
||||||
|
-of FNAME, --output-file FNAME [ ] output file path (without file extension)
|
||||||
|
-ps, --print-special [false ] print special tokens
|
||||||
|
-pc, --print-colors [false ] print colors
|
||||||
|
-pp, --print-progress [false ] print progress
|
||||||
|
-nt, --no-timestamps [false ] do not print timestamps
|
||||||
|
-l LANG, --language LANG [en ] spoken language ('auto' for auto-detect)
|
||||||
|
-dl, --detect-language [false ] exit after automatically detecting language
|
||||||
|
--prompt PROMPT [ ] initial prompt
|
||||||
|
-m FNAME, --model FNAME [models/ggml-base.en.bin] model path
|
||||||
|
-f FNAME, --file FNAME [ ] input WAV file path
|
||||||
|
-oved D, --ov-e-device DNAME [CPU ] the OpenVINO device used for encode inference
|
||||||
|
-ls, --log-score [false ] log best decoder scores of tokens
|
||||||
|
-ng, --no-gpu [false ] disable GPU
|
||||||
|
|
||||||
|
|
||||||
|
bash ./models/download-ggml-model.sh base.en
|
||||||
|
Downloading ggml model base.en ...
|
||||||
|
ggml-base.en.bin 100%[========================>] 141.11M 6.34MB/s in 24s
|
||||||
|
Done! Model 'base.en' saved in 'models/ggml-base.en.bin'
|
||||||
|
You can now use it like this:
|
||||||
|
|
||||||
|
$ ./main -m models/ggml-base.en.bin -f samples/jfk.wav
|
||||||
|
|
||||||
|
|
||||||
|
===============================================
|
||||||
|
Running base.en on all samples in ./samples ...
|
||||||
|
===============================================
|
||||||
|
|
||||||
|
----------------------------------------------
|
||||||
|
[+] Running base.en on samples/jfk.wav ... (run 'ffplay samples/jfk.wav' to listen)
|
||||||
|
----------------------------------------------
|
||||||
|
|
||||||
|
whisper_init_from_file: loading model from 'models/ggml-base.en.bin'
|
||||||
|
whisper_model_load: loading model
|
||||||
|
whisper_model_load: n_vocab = 51864
|
||||||
|
whisper_model_load: n_audio_ctx = 1500
|
||||||
|
whisper_model_load: n_audio_state = 512
|
||||||
|
whisper_model_load: n_audio_head = 8
|
||||||
|
whisper_model_load: n_audio_layer = 6
|
||||||
|
whisper_model_load: n_text_ctx = 448
|
||||||
|
whisper_model_load: n_text_state = 512
|
||||||
|
whisper_model_load: n_text_head = 8
|
||||||
|
whisper_model_load: n_text_layer = 6
|
||||||
|
whisper_model_load: n_mels = 80
|
||||||
|
whisper_model_load: f16 = 1
|
||||||
|
whisper_model_load: type = 2
|
||||||
|
whisper_model_load: mem required = 215.00 MB (+ 6.00 MB per decoder)
|
||||||
|
whisper_model_load: kv self size = 5.25 MB
|
||||||
|
whisper_model_load: kv cross size = 17.58 MB
|
||||||
|
whisper_model_load: adding 1607 extra tokens
|
||||||
|
whisper_model_load: model ctx = 140.60 MB
|
||||||
|
whisper_model_load: model size = 140.54 MB
|
||||||
|
|
||||||
|
system_info: n_threads = 4 / 10 | AVX = 0 | AVX2 = 0 | AVX512 = 0 | FMA = 0 | NEON = 1 | ARM_FMA = 1 | F16C = 0 | FP16_VA = 1 | WASM_SIMD = 0 | BLAS = 1 | SSE3 = 0 | VSX = 0 |
|
||||||
|
|
||||||
|
main: processing 'samples/jfk.wav' (176000 samples, 11.0 sec), 4 threads, 1 processors, lang = en, task = transcribe, timestamps = 1 ...
|
||||||
|
|
||||||
|
|
||||||
|
[00:00:00.000 --> 00:00:11.000] And so my fellow Americans, ask not what your country can do for you, ask what you can do for your country.
|
||||||
|
|
||||||
|
|
||||||
|
whisper_print_timings: fallbacks = 0 p / 0 h
|
||||||
|
whisper_print_timings: load time = 113.81 ms
|
||||||
|
whisper_print_timings: mel time = 15.40 ms
|
||||||
|
whisper_print_timings: sample time = 11.58 ms / 27 runs ( 0.43 ms per run)
|
||||||
|
whisper_print_timings: encode time = 266.60 ms / 1 runs ( 266.60 ms per run)
|
||||||
|
whisper_print_timings: decode time = 66.11 ms / 27 runs ( 2.45 ms per run)
|
||||||
|
whisper_print_timings: total time = 476.31 ms
|
||||||
|
```
|
||||||
|
|
||||||
The command downloads the `base.en` model converted to custom `ggml` format and runs the inference on all `.wav` samples in the folder `samples`.
|
The command downloads the `base.en` model converted to custom `ggml` format and runs the inference on all `.wav` samples in the folder `samples`.
|
||||||
|
|
||||||
For detailed usage instructions, run: `./build/bin/whisper-cli -h`
|
For detailed usage instructions, run: `./main -h`
|
||||||
|
|
||||||
Note that the [whisper-cli](examples/cli) example currently runs only with 16-bit WAV files, so make sure to convert your input before running the tool.
|
Note that the [main](examples/main) example currently runs only with 16-bit WAV files, so make sure to convert your input before running the tool.
|
||||||
For example, you can use `ffmpeg` like this:
|
For example, you can use `ffmpeg` like this:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
@ -107,7 +216,7 @@ ffmpeg -i input.mp3 -ar 16000 -ac 1 -c:a pcm_s16le output.wav
|
|||||||
If you want some extra audio samples to play with, simply run:
|
If you want some extra audio samples to play with, simply run:
|
||||||
|
|
||||||
```
|
```
|
||||||
make -j samples
|
make samples
|
||||||
```
|
```
|
||||||
|
|
||||||
This will download a few more audio files from Wikipedia and convert them to 16-bit WAV format via `ffmpeg`.
|
This will download a few more audio files from Wikipedia and convert them to 16-bit WAV format via `ffmpeg`.
|
||||||
@ -115,18 +224,17 @@ This will download a few more audio files from Wikipedia and convert them to 16-
|
|||||||
You can download and run the other models as follows:
|
You can download and run the other models as follows:
|
||||||
|
|
||||||
```
|
```
|
||||||
make -j tiny.en
|
make tiny.en
|
||||||
make -j tiny
|
make tiny
|
||||||
make -j base.en
|
make base.en
|
||||||
make -j base
|
make base
|
||||||
make -j small.en
|
make small.en
|
||||||
make -j small
|
make small
|
||||||
make -j medium.en
|
make medium.en
|
||||||
make -j medium
|
make medium
|
||||||
make -j large-v1
|
make large-v1
|
||||||
make -j large-v2
|
make large-v2
|
||||||
make -j large-v3
|
make large-v3
|
||||||
make -j large-v3-turbo
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Memory usage
|
## Memory usage
|
||||||
@ -139,20 +247,6 @@ make -j large-v3-turbo
|
|||||||
| medium | 1.5 GiB | ~2.1 GB |
|
| medium | 1.5 GiB | ~2.1 GB |
|
||||||
| large | 2.9 GiB | ~3.9 GB |
|
| large | 2.9 GiB | ~3.9 GB |
|
||||||
|
|
||||||
## POWER VSX Intrinsics
|
|
||||||
|
|
||||||
`whisper.cpp` supports POWER architectures and includes code which
|
|
||||||
significantly speeds operation on Linux running on POWER9/10, making it
|
|
||||||
capable of faster-than-realtime transcription on underclocked Raptor
|
|
||||||
Talos II. Ensure you have a BLAS package installed, and replace the
|
|
||||||
standard cmake setup with:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# build with GGML_BLAS defined
|
|
||||||
cmake -B build -DGGML_BLAS=1
|
|
||||||
cmake --build build --config Release
|
|
||||||
./build/bin/whisper-cli [ .. etc .. ]
|
|
||||||
|
|
||||||
## Quantization
|
## Quantization
|
||||||
|
|
||||||
`whisper.cpp` supports integer quantization of the Whisper `ggml` models.
|
`whisper.cpp` supports integer quantization of the Whisper `ggml` models.
|
||||||
@ -162,12 +256,11 @@ Here are the steps for creating and using a quantized model:
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
# quantize a model with Q5_0 method
|
# quantize a model with Q5_0 method
|
||||||
cmake -B build
|
make quantize
|
||||||
cmake --build build --config Release
|
./quantize models/ggml-base.en.bin models/ggml-base.en-q5_0.bin q5_0
|
||||||
./build/bin/quantize models/ggml-base.en.bin models/ggml-base.en-q5_0.bin q5_0
|
|
||||||
|
|
||||||
# run the examples as usual, specifying the quantized model file
|
# run the examples as usual, specifying the quantized model file
|
||||||
./build/bin/whisper-cli -m models/ggml-base.en-q5_0.bin ./samples/gb0.wav
|
./main -m models/ggml-base.en-q5_0.bin ./samples/gb0.wav
|
||||||
```
|
```
|
||||||
|
|
||||||
## Core ML support
|
## Core ML support
|
||||||
@ -201,6 +294,10 @@ speed-up - more than x3 faster compared with CPU-only execution. Here are the in
|
|||||||
- Build `whisper.cpp` with Core ML support:
|
- Build `whisper.cpp` with Core ML support:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
# using Makefile
|
||||||
|
make clean
|
||||||
|
WHISPER_COREML=1 make -j
|
||||||
|
|
||||||
# using CMake
|
# using CMake
|
||||||
cmake -B build -DWHISPER_COREML=1
|
cmake -B build -DWHISPER_COREML=1
|
||||||
cmake --build build -j --config Release
|
cmake --build build -j --config Release
|
||||||
@ -209,7 +306,7 @@ speed-up - more than x3 faster compared with CPU-only execution. Here are the in
|
|||||||
- Run the examples as usual. For example:
|
- Run the examples as usual. For example:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
$ ./build/bin/whisper-cli -m models/ggml-base.en.bin -f samples/jfk.wav
|
$ ./main -m models/ggml-base.en.bin -f samples/jfk.wav
|
||||||
|
|
||||||
...
|
...
|
||||||
|
|
||||||
@ -293,7 +390,7 @@ This can result in significant speedup in encoder performance. Here are the inst
|
|||||||
- Run the examples as usual. For example:
|
- Run the examples as usual. For example:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
$ ./build/bin/whisper-cli -m models/ggml-base.en.bin -f samples/jfk.wav
|
$ ./main -m models/ggml-base.en.bin -f samples/jfk.wav
|
||||||
|
|
||||||
...
|
...
|
||||||
|
|
||||||
@ -310,7 +407,7 @@ This can result in significant speedup in encoder performance. Here are the inst
|
|||||||
The first time run on an OpenVINO device is slow, since the OpenVINO framework will compile the IR (Intermediate Representation) model to a device-specific 'blob'. This device-specific blob will get
|
The first time run on an OpenVINO device is slow, since the OpenVINO framework will compile the IR (Intermediate Representation) model to a device-specific 'blob'. This device-specific blob will get
|
||||||
cached for the next run.
|
cached for the next run.
|
||||||
|
|
||||||
For more information about the OpenVINO implementation please refer to PR [#1037](https://github.com/ggerganov/whisper.cpp/pull/1037).
|
For more information about the Core ML implementation please refer to PR [#1037](https://github.com/ggerganov/whisper.cpp/pull/1037).
|
||||||
|
|
||||||
## NVIDIA GPU support
|
## NVIDIA GPU support
|
||||||
|
|
||||||
@ -320,18 +417,8 @@ First, make sure you have installed `cuda`: https://developer.nvidia.com/cuda-do
|
|||||||
Now build `whisper.cpp` with CUDA support:
|
Now build `whisper.cpp` with CUDA support:
|
||||||
|
|
||||||
```
|
```
|
||||||
cmake -B build -DGGML_CUDA=1
|
make clean
|
||||||
cmake --build build -j --config Release
|
GGML_CUDA=1 make -j
|
||||||
```
|
|
||||||
|
|
||||||
## Vulkan GPU support
|
|
||||||
Cross-vendor solution which allows you to accelerate workload on your GPU.
|
|
||||||
First, make sure your graphics card driver provides support for Vulkan API.
|
|
||||||
|
|
||||||
Now build `whisper.cpp` with Vulkan support:
|
|
||||||
```
|
|
||||||
cmake -B build -DGGML_VULKAN=1
|
|
||||||
cmake --build build -j --config Release
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## BLAS CPU support via OpenBLAS
|
## BLAS CPU support via OpenBLAS
|
||||||
@ -342,41 +429,25 @@ First, make sure you have installed `openblas`: https://www.openblas.net/
|
|||||||
Now build `whisper.cpp` with OpenBLAS support:
|
Now build `whisper.cpp` with OpenBLAS support:
|
||||||
|
|
||||||
```
|
```
|
||||||
cmake -B build -DGGML_BLAS=1
|
make clean
|
||||||
cmake --build build -j --config Release
|
GGML_OPENBLAS=1 make -j
|
||||||
```
|
```
|
||||||
|
|
||||||
## Ascend NPU support
|
## BLAS CPU support via Intel MKL
|
||||||
|
|
||||||
Ascend NPU provides inference acceleration via [`CANN`](https://www.hiascend.com/en/software/cann) and AI cores.
|
Encoder processing can be accelerated on the CPU via the BLAS compatible interface of Intel's Math Kernel Library.
|
||||||
|
First, make sure you have installed Intel's MKL runtime and development packages: https://www.intel.com/content/www/us/en/developer/tools/oneapi/onemkl-download.html
|
||||||
|
|
||||||
First, check if your Ascend NPU device is supported:
|
Now build `whisper.cpp` with Intel MKL BLAS support:
|
||||||
|
|
||||||
**Verified devices**
|
|
||||||
| Ascend NPU | Status |
|
|
||||||
|:-----------------------------:|:-------:|
|
|
||||||
| Atlas 300T A2 | Support |
|
|
||||||
|
|
||||||
Then, make sure you have installed [`CANN toolkit`](https://www.hiascend.com/en/software/cann/community) . The lasted version of CANN is recommanded.
|
|
||||||
|
|
||||||
Now build `whisper.cpp` with CANN support:
|
|
||||||
|
|
||||||
```
|
```
|
||||||
cmake -B build -DGGML_CANN=1
|
source /opt/intel/oneapi/setvars.sh
|
||||||
cmake --build build -j --config Release
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake -DWHISPER_MKL=ON ..
|
||||||
|
WHISPER_MKL=1 make -j
|
||||||
```
|
```
|
||||||
|
|
||||||
Run the inference examples as usual, for example:
|
|
||||||
|
|
||||||
```
|
|
||||||
./build/bin/whisper-cli -f samples/jfk.wav -m models/ggml-base.en.bin -t 8
|
|
||||||
```
|
|
||||||
|
|
||||||
*Notes:*
|
|
||||||
|
|
||||||
- If you have trouble with Ascend NPU device, please create a issue with **[CANN]** prefix/tag.
|
|
||||||
- If you run successfully with your Ascend NPU device, please help update the table `Verified devices`.
|
|
||||||
|
|
||||||
## Docker
|
## Docker
|
||||||
|
|
||||||
### Prerequisites
|
### Prerequisites
|
||||||
@ -423,6 +494,89 @@ For detailed instructions on how to use Conan, please refer to the [Conan docume
|
|||||||
|
|
||||||
- Inference only
|
- Inference only
|
||||||
|
|
||||||
|
## Another example
|
||||||
|
|
||||||
|
Here is another example of transcribing a [3:24 min speech](https://upload.wikimedia.org/wikipedia/commons/1/1f/George_W_Bush_Columbia_FINAL.ogg)
|
||||||
|
in about half a minute on a MacBook M1 Pro, using `medium.en` model:
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Expand to see the result</summary>
|
||||||
|
|
||||||
|
```text
|
||||||
|
$ ./main -m models/ggml-medium.en.bin -f samples/gb1.wav -t 8
|
||||||
|
|
||||||
|
whisper_init_from_file: loading model from 'models/ggml-medium.en.bin'
|
||||||
|
whisper_model_load: loading model
|
||||||
|
whisper_model_load: n_vocab = 51864
|
||||||
|
whisper_model_load: n_audio_ctx = 1500
|
||||||
|
whisper_model_load: n_audio_state = 1024
|
||||||
|
whisper_model_load: n_audio_head = 16
|
||||||
|
whisper_model_load: n_audio_layer = 24
|
||||||
|
whisper_model_load: n_text_ctx = 448
|
||||||
|
whisper_model_load: n_text_state = 1024
|
||||||
|
whisper_model_load: n_text_head = 16
|
||||||
|
whisper_model_load: n_text_layer = 24
|
||||||
|
whisper_model_load: n_mels = 80
|
||||||
|
whisper_model_load: f16 = 1
|
||||||
|
whisper_model_load: type = 4
|
||||||
|
whisper_model_load: mem required = 1720.00 MB (+ 43.00 MB per decoder)
|
||||||
|
whisper_model_load: kv self size = 42.00 MB
|
||||||
|
whisper_model_load: kv cross size = 140.62 MB
|
||||||
|
whisper_model_load: adding 1607 extra tokens
|
||||||
|
whisper_model_load: model ctx = 1462.35 MB
|
||||||
|
whisper_model_load: model size = 1462.12 MB
|
||||||
|
|
||||||
|
system_info: n_threads = 8 / 10 | AVX = 0 | AVX2 = 0 | AVX512 = 0 | FMA = 0 | NEON = 1 | ARM_FMA = 1 | F16C = 0 | FP16_VA = 1 | WASM_SIMD = 0 | BLAS = 1 | SSE3 = 0 | VSX = 0 |
|
||||||
|
|
||||||
|
main: processing 'samples/gb1.wav' (3179750 samples, 198.7 sec), 8 threads, 1 processors, lang = en, task = transcribe, timestamps = 1 ...
|
||||||
|
|
||||||
|
|
||||||
|
[00:00:00.000 --> 00:00:08.000] My fellow Americans, this day has brought terrible news and great sadness to our country.
|
||||||
|
[00:00:08.000 --> 00:00:17.000] At nine o'clock this morning, Mission Control in Houston lost contact with our Space Shuttle Columbia.
|
||||||
|
[00:00:17.000 --> 00:00:23.000] A short time later, debris was seen falling from the skies above Texas.
|
||||||
|
[00:00:23.000 --> 00:00:29.000] The Columbia's lost. There are no survivors.
|
||||||
|
[00:00:29.000 --> 00:00:32.000] On board was a crew of seven.
|
||||||
|
[00:00:32.000 --> 00:00:39.000] Colonel Rick Husband, Lieutenant Colonel Michael Anderson, Commander Laurel Clark,
|
||||||
|
[00:00:39.000 --> 00:00:48.000] Captain David Brown, Commander William McCool, Dr. Kultna Shavla, and Ilan Ramon,
|
||||||
|
[00:00:48.000 --> 00:00:52.000] a colonel in the Israeli Air Force.
|
||||||
|
[00:00:52.000 --> 00:00:58.000] These men and women assumed great risk in the service to all humanity.
|
||||||
|
[00:00:58.000 --> 00:01:03.000] In an age when space flight has come to seem almost routine,
|
||||||
|
[00:01:03.000 --> 00:01:07.000] it is easy to overlook the dangers of travel by rocket
|
||||||
|
[00:01:07.000 --> 00:01:12.000] and the difficulties of navigating the fierce outer atmosphere of the Earth.
|
||||||
|
[00:01:12.000 --> 00:01:18.000] These astronauts knew the dangers, and they faced them willingly,
|
||||||
|
[00:01:18.000 --> 00:01:23.000] knowing they had a high and noble purpose in life.
|
||||||
|
[00:01:23.000 --> 00:01:31.000] Because of their courage and daring and idealism, we will miss them all the more.
|
||||||
|
[00:01:31.000 --> 00:01:36.000] All Americans today are thinking as well of the families of these men and women
|
||||||
|
[00:01:36.000 --> 00:01:40.000] who have been given this sudden shock and grief.
|
||||||
|
[00:01:40.000 --> 00:01:45.000] You're not alone. Our entire nation grieves with you,
|
||||||
|
[00:01:45.000 --> 00:01:52.000] and those you love will always have the respect and gratitude of this country.
|
||||||
|
[00:01:52.000 --> 00:01:56.000] The cause in which they died will continue.
|
||||||
|
[00:01:56.000 --> 00:02:04.000] Mankind is led into the darkness beyond our world by the inspiration of discovery
|
||||||
|
[00:02:04.000 --> 00:02:11.000] and the longing to understand. Our journey into space will go on.
|
||||||
|
[00:02:11.000 --> 00:02:16.000] In the skies today, we saw destruction and tragedy.
|
||||||
|
[00:02:16.000 --> 00:02:22.000] Yet farther than we can see, there is comfort and hope.
|
||||||
|
[00:02:22.000 --> 00:02:29.000] In the words of the prophet Isaiah, "Lift your eyes and look to the heavens
|
||||||
|
[00:02:29.000 --> 00:02:35.000] who created all these. He who brings out the starry hosts one by one
|
||||||
|
[00:02:35.000 --> 00:02:39.000] and calls them each by name."
|
||||||
|
[00:02:39.000 --> 00:02:46.000] Because of His great power and mighty strength, not one of them is missing.
|
||||||
|
[00:02:46.000 --> 00:02:55.000] The same Creator who names the stars also knows the names of the seven souls we mourn today.
|
||||||
|
[00:02:55.000 --> 00:03:01.000] The crew of the shuttle Columbia did not return safely to earth,
|
||||||
|
[00:03:01.000 --> 00:03:05.000] yet we can pray that all are safely home.
|
||||||
|
[00:03:05.000 --> 00:03:13.000] May God bless the grieving families, and may God continue to bless America.
|
||||||
|
[00:03:13.000 --> 00:03:19.000] [Silence]
|
||||||
|
|
||||||
|
|
||||||
|
whisper_print_timings: fallbacks = 1 p / 0 h
|
||||||
|
whisper_print_timings: load time = 569.03 ms
|
||||||
|
whisper_print_timings: mel time = 146.85 ms
|
||||||
|
whisper_print_timings: sample time = 238.66 ms / 553 runs ( 0.43 ms per run)
|
||||||
|
whisper_print_timings: encode time = 18665.10 ms / 9 runs ( 2073.90 ms per run)
|
||||||
|
whisper_print_timings: decode time = 13090.93 ms / 549 runs ( 23.85 ms per run)
|
||||||
|
whisper_print_timings: total time = 32733.52 ms
|
||||||
|
```
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
## Real-time audio input example
|
## Real-time audio input example
|
||||||
|
|
||||||
This is a naive example of performing real-time inference on audio from your microphone.
|
This is a naive example of performing real-time inference on audio from your microphone.
|
||||||
@ -430,9 +584,8 @@ The [stream](examples/stream) tool samples the audio every half a second and run
|
|||||||
More info is available in [issue #10](https://github.com/ggerganov/whisper.cpp/issues/10).
|
More info is available in [issue #10](https://github.com/ggerganov/whisper.cpp/issues/10).
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cmake -B build -DWHISPER_SDL2=ON
|
make stream
|
||||||
cmake --build build --config Release
|
./stream -m ./models/ggml-base.en.bin -t 8 --step 500 --length 5000
|
||||||
./build/bin/whisper-stream -m ./models/ggml-base.en.bin -t 8 --step 500 --length 5000
|
|
||||||
```
|
```
|
||||||
|
|
||||||
https://user-images.githubusercontent.com/1991296/194935793-76afede7-cfa8-48d8-a80f-28ba83be7d09.mp4
|
https://user-images.githubusercontent.com/1991296/194935793-76afede7-cfa8-48d8-a80f-28ba83be7d09.mp4
|
||||||
@ -443,7 +596,7 @@ Adding the `--print-colors` argument will print the transcribed text using an ex
|
|||||||
to highlight words with high or low confidence:
|
to highlight words with high or low confidence:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./build/bin/whisper-cli -m models/ggml-base.en.bin -f samples/gb0.wav --print-colors
|
./main -m models/ggml-base.en.bin -f samples/gb0.wav --print-colors
|
||||||
```
|
```
|
||||||
|
|
||||||
<img width="965" alt="image" src="https://user-images.githubusercontent.com/1991296/197356445-311c8643-9397-4e5e-b46e-0b4b4daa2530.png">
|
<img width="965" alt="image" src="https://user-images.githubusercontent.com/1991296/197356445-311c8643-9397-4e5e-b46e-0b4b4daa2530.png">
|
||||||
@ -453,7 +606,7 @@ to highlight words with high or low confidence:
|
|||||||
For example, to limit the line length to a maximum of 16 characters, simply add `-ml 16`:
|
For example, to limit the line length to a maximum of 16 characters, simply add `-ml 16`:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
$ ./build/bin/whisper-cli -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -ml 16
|
$ ./main -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -ml 16
|
||||||
|
|
||||||
whisper_model_load: loading model from './models/ggml-base.en.bin'
|
whisper_model_load: loading model from './models/ggml-base.en.bin'
|
||||||
...
|
...
|
||||||
@ -477,7 +630,7 @@ main: processing './samples/jfk.wav' (176000 samples, 11.0 sec), 4 threads, 1 pr
|
|||||||
The `--max-len` argument can be used to obtain word-level timestamps. Simply use `-ml 1`:
|
The `--max-len` argument can be used to obtain word-level timestamps. Simply use `-ml 1`:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
$ ./build/bin/whisper-cli -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -ml 1
|
$ ./main -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -ml 1
|
||||||
|
|
||||||
whisper_model_load: loading model from './models/ggml-base.en.bin'
|
whisper_model_load: loading model from './models/ggml-base.en.bin'
|
||||||
...
|
...
|
||||||
@ -524,7 +677,7 @@ Sample usage:
|
|||||||
./models/download-ggml-model.sh small.en-tdrz
|
./models/download-ggml-model.sh small.en-tdrz
|
||||||
|
|
||||||
# run as usual, adding the "-tdrz" command-line argument
|
# run as usual, adding the "-tdrz" command-line argument
|
||||||
./build/bin/whisper-cli -f ./samples/a13.wav -m ./models/ggml-small.en-tdrz.bin -tdrz
|
./main -f ./samples/a13.wav -m ./models/ggml-small.en-tdrz.bin -tdrz
|
||||||
...
|
...
|
||||||
main: processing './samples/a13.wav' (480000 samples, 30.0 sec), 4 threads, 1 processors, lang = en, task = transcribe, tdrz = 1, timestamps = 1 ...
|
main: processing './samples/a13.wav' (480000 samples, 30.0 sec), 4 threads, 1 processors, lang = en, task = transcribe, tdrz = 1, timestamps = 1 ...
|
||||||
...
|
...
|
||||||
@ -541,14 +694,14 @@ main: processing './samples/a13.wav' (480000 samples, 30.0 sec), 4 threads, 1 pr
|
|||||||
|
|
||||||
## Karaoke-style movie generation (experimental)
|
## Karaoke-style movie generation (experimental)
|
||||||
|
|
||||||
The [whisper-cli](examples/cli) example provides support for output of karaoke-style movies, where the
|
The [main](examples/main) example provides support for output of karaoke-style movies, where the
|
||||||
currently pronounced word is highlighted. Use the `-wts` argument and run the generated bash script.
|
currently pronounced word is highlighted. Use the `-wts` argument and run the generated bash script.
|
||||||
This requires to have `ffmpeg` installed.
|
This requires to have `ffmpeg` installed.
|
||||||
|
|
||||||
Here are a few _"typical"_ examples:
|
Here are a few _"typical"_ examples:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./build/bin/whisper-cli -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -owts
|
./main -m ./models/ggml-base.en.bin -f ./samples/jfk.wav -owts
|
||||||
source ./samples/jfk.wav.wts
|
source ./samples/jfk.wav.wts
|
||||||
ffplay ./samples/jfk.wav.mp4
|
ffplay ./samples/jfk.wav.mp4
|
||||||
```
|
```
|
||||||
@ -558,7 +711,7 @@ https://user-images.githubusercontent.com/1991296/199337465-dbee4b5e-9aeb-48a3-b
|
|||||||
---
|
---
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./build/bin/whisper-cli -m ./models/ggml-base.en.bin -f ./samples/mm0.wav -owts
|
./main -m ./models/ggml-base.en.bin -f ./samples/mm0.wav -owts
|
||||||
source ./samples/mm0.wav.wts
|
source ./samples/mm0.wav.wts
|
||||||
ffplay ./samples/mm0.wav.mp4
|
ffplay ./samples/mm0.wav.mp4
|
||||||
```
|
```
|
||||||
@ -568,7 +721,7 @@ https://user-images.githubusercontent.com/1991296/199337504-cc8fd233-0cb7-4920-9
|
|||||||
---
|
---
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./build/bin/whisper-cli -m ./models/ggml-base.en.bin -f ./samples/gb0.wav -owts
|
./main -m ./models/ggml-base.en.bin -f ./samples/gb0.wav -owts
|
||||||
source ./samples/gb0.wav.wts
|
source ./samples/gb0.wav.wts
|
||||||
ffplay ./samples/gb0.wav.mp4
|
ffplay ./samples/gb0.wav.mp4
|
||||||
```
|
```
|
||||||
@ -593,12 +746,12 @@ https://user-images.githubusercontent.com/1991296/223206245-2d36d903-cf8e-4f09-8
|
|||||||
## Benchmarks
|
## Benchmarks
|
||||||
|
|
||||||
In order to have an objective comparison of the performance of the inference across different system configurations,
|
In order to have an objective comparison of the performance of the inference across different system configurations,
|
||||||
use the [whisper-bench](examples/bench) tool. The tool simply runs the Encoder part of the model and prints how much time it
|
use the [bench](examples/bench) tool. The tool simply runs the Encoder part of the model and prints how much time it
|
||||||
took to execute it. The results are summarized in the following Github issue:
|
took to execute it. The results are summarized in the following Github issue:
|
||||||
|
|
||||||
[Benchmark results](https://github.com/ggerganov/whisper.cpp/issues/89)
|
[Benchmark results](https://github.com/ggerganov/whisper.cpp/issues/89)
|
||||||
|
|
||||||
Additionally a script to run whisper.cpp with different models and audio files is provided [bench.py](scripts/bench.py).
|
Additionally a script to run whisper.cpp with different models and audio files is provided [bench.py](bench.py).
|
||||||
|
|
||||||
You can run it with the following command, by default it will run against any standard model in the models folder.
|
You can run it with the following command, by default it will run against any standard model in the models folder.
|
||||||
|
|
||||||
@ -645,7 +798,6 @@ For more details, see the conversion script [models/convert-pt-to-ggml.py](model
|
|||||||
- [stlukey/whispercpp.py](https://github.com/stlukey/whispercpp.py) (Cython)
|
- [stlukey/whispercpp.py](https://github.com/stlukey/whispercpp.py) (Cython)
|
||||||
- [AIWintermuteAI/whispercpp](https://github.com/AIWintermuteAI/whispercpp) (Updated fork of aarnphm/whispercpp)
|
- [AIWintermuteAI/whispercpp](https://github.com/AIWintermuteAI/whispercpp) (Updated fork of aarnphm/whispercpp)
|
||||||
- [aarnphm/whispercpp](https://github.com/aarnphm/whispercpp) (Pybind11)
|
- [aarnphm/whispercpp](https://github.com/aarnphm/whispercpp) (Pybind11)
|
||||||
- [abdeladim-s/pywhispercpp](https://github.com/abdeladim-s/pywhispercpp) (Pybind11)
|
|
||||||
- [x] R: [bnosac/audio.whisper](https://github.com/bnosac/audio.whisper)
|
- [x] R: [bnosac/audio.whisper](https://github.com/bnosac/audio.whisper)
|
||||||
- [x] Unity: [macoron/whisper.unity](https://github.com/Macoron/whisper.unity)
|
- [x] Unity: [macoron/whisper.unity](https://github.com/Macoron/whisper.unity)
|
||||||
|
|
||||||
@ -656,12 +808,13 @@ Some of the examples are even ported to run in the browser using WebAssembly. Ch
|
|||||||
|
|
||||||
| Example | Web | Description |
|
| Example | Web | Description |
|
||||||
| --------------------------------------------------- | ------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------- |
|
| --------------------------------------------------- | ------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------- |
|
||||||
| [whisper-cli](examples/cli) | [whisper.wasm](examples/whisper.wasm) | Tool for translating and transcribing audio using Whisper |
|
| [main](examples/main) | [whisper.wasm](examples/whisper.wasm) | Tool for translating and transcribing audio using Whisper |
|
||||||
| [whisper-bench](examples/bench) | [bench.wasm](examples/bench.wasm) | Benchmark the performance of Whisper on your machine |
|
| [bench](examples/bench) | [bench.wasm](examples/bench.wasm) | Benchmark the performance of Whisper on your machine |
|
||||||
| [whisper-stream](examples/stream) | [stream.wasm](examples/stream.wasm) | Real-time transcription of raw microphone capture |
|
| [stream](examples/stream) | [stream.wasm](examples/stream.wasm) | Real-time transcription of raw microphone capture |
|
||||||
| [whisper-command](examples/command) | [command.wasm](examples/command.wasm) | Basic voice assistant example for receiving voice commands from the mic |
|
| [command](examples/command) | [command.wasm](examples/command.wasm) | Basic voice assistant example for receiving voice commands from the mic |
|
||||||
| [whisper-server](examples/server) | | HTTP transcription server with OAI-like API |
|
| [wchess](examples/wchess) | [wchess.wasm](examples/wchess) | Voice-controlled chess |
|
||||||
| [whisper-talk-llama](examples/talk-llama) | | Talk with a LLaMA bot |
|
| [talk](examples/talk) | [talk.wasm](examples/talk.wasm) | Talk with a GPT-2 bot |
|
||||||
|
| [talk-llama](examples/talk-llama) | | Talk with a LLaMA bot |
|
||||||
| [whisper.objc](examples/whisper.objc) | | iOS mobile application using whisper.cpp |
|
| [whisper.objc](examples/whisper.objc) | | iOS mobile application using whisper.cpp |
|
||||||
| [whisper.swiftui](examples/whisper.swiftui) | | SwiftUI iOS / macOS application using whisper.cpp |
|
| [whisper.swiftui](examples/whisper.swiftui) | | SwiftUI iOS / macOS application using whisper.cpp |
|
||||||
| [whisper.android](examples/whisper.android) | | Android mobile application using whisper.cpp |
|
| [whisper.android](examples/whisper.android) | | Android mobile application using whisper.cpp |
|
||||||
@ -669,7 +822,7 @@ Some of the examples are even ported to run in the browser using WebAssembly. Ch
|
|||||||
| [generate-karaoke.sh](examples/generate-karaoke.sh) | | Helper script to easily [generate a karaoke video](https://youtu.be/uj7hVta4blM) of raw audio capture |
|
| [generate-karaoke.sh](examples/generate-karaoke.sh) | | Helper script to easily [generate a karaoke video](https://youtu.be/uj7hVta4blM) of raw audio capture |
|
||||||
| [livestream.sh](examples/livestream.sh) | | [Livestream audio transcription](https://github.com/ggerganov/whisper.cpp/issues/185) |
|
| [livestream.sh](examples/livestream.sh) | | [Livestream audio transcription](https://github.com/ggerganov/whisper.cpp/issues/185) |
|
||||||
| [yt-wsp.sh](examples/yt-wsp.sh) | | Download + transcribe and/or translate any VOD [(original)](https://gist.github.com/DaniruKun/96f763ec1a037cc92fe1a059b643b818) |
|
| [yt-wsp.sh](examples/yt-wsp.sh) | | Download + transcribe and/or translate any VOD [(original)](https://gist.github.com/DaniruKun/96f763ec1a037cc92fe1a059b643b818) |
|
||||||
| [wchess](examples/wchess) | [wchess.wasm](examples/wchess) | Voice-controlled chess |
|
| [server](examples/server) | | HTTP transcription server with OAI-like API |
|
||||||
|
|
||||||
## [Discussions](https://github.com/ggerganov/whisper.cpp/discussions)
|
## [Discussions](https://github.com/ggerganov/whisper.cpp/discussions)
|
||||||
|
|
||||||
|
@ -1,5 +0,0 @@
|
|||||||
module whisper [system] {
|
|
||||||
header "whisper.h"
|
|
||||||
link "whisper"
|
|
||||||
export *
|
|
||||||
}
|
|
@ -1,4 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <whisper.h>
|
|
||||||
|
|
@ -14,14 +14,9 @@ GGML_METAL_PATH_RESOURCES := $(abspath ../..)
|
|||||||
BUILD_DIR := build
|
BUILD_DIR := build
|
||||||
MODELS_DIR := models
|
MODELS_DIR := models
|
||||||
EXAMPLES_DIR := $(wildcard examples/*)
|
EXAMPLES_DIR := $(wildcard examples/*)
|
||||||
INCLUDE_PATH := $(abspath ../../include):$(abspath ../../ggml/include)
|
INCLUDE_PATH := $(abspath ../..)
|
||||||
LIBRARY_PATH := $(abspath ../..)
|
LIBRARY_PATH := $(abspath ../..)
|
||||||
|
|
||||||
ifeq ($(GGML_CUDA),1)
|
|
||||||
LIBRARY_PATH := $(LIBRARY_PATH):$(CUDA_PATH)/targets/$(UNAME_M)-linux/lib/
|
|
||||||
BUILD_FLAGS := -ldflags "-extldflags '-lcudart -lcuda -lcublas'"
|
|
||||||
endif
|
|
||||||
|
|
||||||
ifeq ($(UNAME_S),Darwin)
|
ifeq ($(UNAME_S),Darwin)
|
||||||
EXT_LDFLAGS := -framework Foundation -framework Metal -framework MetalKit
|
EXT_LDFLAGS := -framework Foundation -framework Metal -framework MetalKit
|
||||||
endif
|
endif
|
||||||
|
@ -62,12 +62,6 @@ This will compile a static `libwhisper.a` in a `build` folder, download a model
|
|||||||
make examples
|
make examples
|
||||||
```
|
```
|
||||||
|
|
||||||
To build using cuda support add `GGML_CUDA=1`:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
GGML_CUDA=1 make examples
|
|
||||||
```
|
|
||||||
|
|
||||||
The examples are placed in the `build` directory. Once built, you can download all the models with the following command:
|
The examples are placed in the `build` directory. Once built, you can download all the models with the following command:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
@ -24,7 +24,7 @@ const (
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
// The models which will be downloaded, if no model is specified as an argument
|
// The models which will be downloaded, if no model is specified as an argument
|
||||||
modelNames = []string{"ggml-tiny.en", "ggml-tiny", "ggml-base.en", "ggml-base", "ggml-small.en", "ggml-small", "ggml-medium.en", "ggml-medium", "ggml-large-v1", "ggml-large-v2", "ggml-large-v3", "large-v3-turbo"}
|
modelNames = []string{"ggml-tiny.en", "ggml-tiny", "ggml-base.en", "ggml-base", "ggml-small.en", "ggml-small", "ggml-medium.en", "ggml-medium", "ggml-large-v1", "ggml-large-v2", "ggml-large-v3"}
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
module github.com/ggerganov/whisper.cpp/bindings/go
|
module github.com/ggerganov/whisper.cpp/bindings/go
|
||||||
|
|
||||||
go 1.23
|
go 1.19
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/go-audio/wav v1.1.0
|
github.com/go-audio/wav v1.1.0
|
||||||
github.com/stretchr/testify v1.9.0
|
github.com/stretchr/testify v1.8.1
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
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/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/go-audio/audio v1.0.0 h1:zS9vebldgbQqktK4H0lUqWrG8P0NxCJVqcj7ZpNnwd4=
|
github.com/go-audio/audio v1.0.0 h1:zS9vebldgbQqktK4H0lUqWrG8P0NxCJVqcj7ZpNnwd4=
|
||||||
@ -8,9 +9,15 @@ github.com/go-audio/wav v1.1.0 h1:jQgLtbqBzY7G+BM8fXF7AHUk1uHUviWS4X39d5rsL2g=
|
|||||||
github.com/go-audio/wav v1.1.0/go.mod h1:mpe9qfwbScEbkd8uybLuIpTgHyrISw/OTuvjUW2iGtE=
|
github.com/go-audio/wav v1.1.0/go.mod h1:mpe9qfwbScEbkd8uybLuIpTgHyrISw/OTuvjUW2iGtE=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||||
|
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
||||||
|
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
|
github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk=
|
||||||
|
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
@ -119,28 +119,6 @@ func (p *Params) SetAudioCtx(n int) {
|
|||||||
p.audio_ctx = C.int(n)
|
p.audio_ctx = C.int(n)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Params) SetMaxContext(n int) {
|
|
||||||
p.n_max_text_ctx = C.int(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *Params) SetBeamSize(n int) {
|
|
||||||
p.beam_search.beam_size = C.int(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *Params) SetEntropyThold(t float32) {
|
|
||||||
p.entropy_thold = C.float(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *Params) SetTemperature(t float32) {
|
|
||||||
p.temperature = C.float(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sets the fallback temperature incrementation
|
|
||||||
// Pass -1.0 to disable this feature
|
|
||||||
func (p *Params) SetTemperatureFallback(t float32) {
|
|
||||||
p.temperature_inc = C.float(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set initial prompt
|
// Set initial prompt
|
||||||
func (p *Params) SetInitialPrompt(prompt string) {
|
func (p *Params) SetInitialPrompt(prompt string) {
|
||||||
p.initial_prompt = C.CString(prompt)
|
p.initial_prompt = C.CString(prompt)
|
||||||
@ -171,10 +149,6 @@ func (p *Params) String() string {
|
|||||||
str += fmt.Sprintf(" duration_ms=%d", p.duration_ms)
|
str += fmt.Sprintf(" duration_ms=%d", p.duration_ms)
|
||||||
str += fmt.Sprintf(" audio_ctx=%d", p.audio_ctx)
|
str += fmt.Sprintf(" audio_ctx=%d", p.audio_ctx)
|
||||||
str += fmt.Sprintf(" initial_prompt=%s", C.GoString(p.initial_prompt))
|
str += fmt.Sprintf(" initial_prompt=%s", C.GoString(p.initial_prompt))
|
||||||
str += fmt.Sprintf(" entropy_thold=%f", p.entropy_thold)
|
|
||||||
str += fmt.Sprintf(" temperature=%f", p.temperature)
|
|
||||||
str += fmt.Sprintf(" temperature_inc=%f", p.temperature_inc)
|
|
||||||
str += fmt.Sprintf(" beam_size=%d", p.beam_search.beam_size)
|
|
||||||
if p.translate {
|
if p.translate {
|
||||||
str += " translate"
|
str += " translate"
|
||||||
}
|
}
|
||||||
|
@ -125,32 +125,6 @@ func (context *context) SetAudioCtx(n uint) {
|
|||||||
context.params.SetAudioCtx(int(n))
|
context.params.SetAudioCtx(int(n))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set maximum number of text context tokens to store
|
|
||||||
func (context *context) SetMaxContext(n int) {
|
|
||||||
context.params.SetMaxContext(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set Beam Size
|
|
||||||
func (context *context) SetBeamSize(n int) {
|
|
||||||
context.params.SetBeamSize(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set Entropy threshold
|
|
||||||
func (context *context) SetEntropyThold(t float32) {
|
|
||||||
context.params.SetEntropyThold(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set Temperature
|
|
||||||
func (context *context) SetTemperature(t float32) {
|
|
||||||
context.params.SetTemperature(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set the fallback temperature incrementation
|
|
||||||
// Pass -1.0 to disable this feature
|
|
||||||
func (context *context) SetTemperatureFallback(t float32) {
|
|
||||||
context.params.SetTemperatureFallback(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set initial prompt
|
// Set initial prompt
|
||||||
func (context *context) SetInitialPrompt(prompt string) {
|
func (context *context) SetInitialPrompt(prompt string) {
|
||||||
context.params.SetInitialPrompt(prompt)
|
context.params.SetInitialPrompt(prompt)
|
||||||
|
@ -4,90 +4,52 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/ggerganov/whisper.cpp/bindings/go/pkg/whisper"
|
// Packages
|
||||||
"github.com/go-audio/wav"
|
whisper "github.com/ggerganov/whisper.cpp/bindings/go/pkg/whisper"
|
||||||
assert "github.com/stretchr/testify/assert"
|
assert "github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSetLanguage(t *testing.T) {
|
const (
|
||||||
assert := assert.New(t)
|
ModelPath = "../../models/ggml-tiny.bin"
|
||||||
|
SamplePath = "../../samples/jfk.wav"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Test_Whisper_000(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
if _, err := os.Stat(ModelPath); os.IsNotExist(err) {
|
||||||
|
t.Skip("Skipping test, model not found:", ModelPath)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(SamplePath); os.IsNotExist(err) {
|
||||||
|
t.Skip("Skipping test, sample not found:", SamplePath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load model
|
||||||
|
model, err := whisper.New(ModelPath)
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.NotNil(model)
|
||||||
|
assert.NoError(model.Close())
|
||||||
|
|
||||||
|
t.Log("languages=", model.Languages())
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_Whisper_001(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
if _, err := os.Stat(ModelPath); os.IsNotExist(err) {
|
||||||
|
t.Skip("Skipping test, model not found:", ModelPath)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(SamplePath); os.IsNotExist(err) {
|
||||||
|
t.Skip("Skipping test, sample not found:", SamplePath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load model
|
||||||
model, err := whisper.New(ModelPath)
|
model, err := whisper.New(ModelPath)
|
||||||
assert.NoError(err)
|
assert.NoError(err)
|
||||||
assert.NotNil(model)
|
assert.NotNil(model)
|
||||||
defer model.Close()
|
defer model.Close()
|
||||||
|
|
||||||
context, err := model.NewContext()
|
// Get context for decoding
|
||||||
|
ctx, err := model.NewContext()
|
||||||
assert.NoError(err)
|
assert.NoError(err)
|
||||||
|
assert.NotNil(ctx)
|
||||||
|
|
||||||
// This returns an error since
|
|
||||||
// the model 'models/ggml-small.en.bin'
|
|
||||||
// that is loaded is not multilingual
|
|
||||||
err = context.SetLanguage("en")
|
|
||||||
assert.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestContextModelIsMultilingual(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
context, err := model.NewContext()
|
|
||||||
assert.NoError(err)
|
|
||||||
|
|
||||||
isMultilingual := context.IsMultilingual()
|
|
||||||
|
|
||||||
// This returns false since
|
|
||||||
// the model 'models/ggml-small.en.bin'
|
|
||||||
// that is loaded is not multilingual
|
|
||||||
assert.False(isMultilingual)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLanguage(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
context, err := model.NewContext()
|
|
||||||
assert.NoError(err)
|
|
||||||
|
|
||||||
// This always returns en since
|
|
||||||
// the model 'models/ggml-small.en.bin'
|
|
||||||
// that is loaded is not multilingual
|
|
||||||
expectedLanguage := "en"
|
|
||||||
actualLanguage := context.Language()
|
|
||||||
assert.Equal(expectedLanguage, actualLanguage)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestProcess(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
fh, err := os.Open(SamplePath)
|
|
||||||
assert.NoError(err)
|
|
||||||
defer fh.Close()
|
|
||||||
|
|
||||||
// Decode the WAV file - load the full buffer
|
|
||||||
dec := wav.NewDecoder(fh)
|
|
||||||
buf, err := dec.FullPCMBuffer()
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.Equal(uint16(1), dec.NumChans)
|
|
||||||
|
|
||||||
data := buf.AsFloat32Buffer().Data
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
context, err := model.NewContext()
|
|
||||||
assert.NoError(err)
|
|
||||||
|
|
||||||
err = context.Process(data, nil, nil)
|
|
||||||
assert.NoError(err)
|
|
||||||
}
|
}
|
||||||
|
@ -38,22 +38,17 @@ type Context interface {
|
|||||||
IsMultilingual() bool // Return true if the model is multilingual.
|
IsMultilingual() bool // Return true if the model is multilingual.
|
||||||
Language() string // Get language
|
Language() string // Get language
|
||||||
|
|
||||||
SetOffset(time.Duration) // Set offset
|
SetOffset(time.Duration) // Set offset
|
||||||
SetDuration(time.Duration) // Set duration
|
SetDuration(time.Duration) // Set duration
|
||||||
SetThreads(uint) // Set number of threads to use
|
SetThreads(uint) // Set number of threads to use
|
||||||
SetSplitOnWord(bool) // Set split on word flag
|
SetSplitOnWord(bool) // Set split on word flag
|
||||||
SetTokenThreshold(float32) // Set timestamp token probability threshold
|
SetTokenThreshold(float32) // Set timestamp token probability threshold
|
||||||
SetTokenSumThreshold(float32) // Set timestamp token sum probability threshold
|
SetTokenSumThreshold(float32) // Set timestamp token sum probability threshold
|
||||||
SetMaxSegmentLength(uint) // Set max segment length in characters
|
SetMaxSegmentLength(uint) // Set max segment length in characters
|
||||||
SetTokenTimestamps(bool) // Set token timestamps flag
|
SetTokenTimestamps(bool) // Set token timestamps flag
|
||||||
SetMaxTokensPerSegment(uint) // Set max tokens per segment (0 = no limit)
|
SetMaxTokensPerSegment(uint) // Set max tokens per segment (0 = no limit)
|
||||||
SetAudioCtx(uint) // Set audio encoder context
|
SetAudioCtx(uint) // Set audio encoder context
|
||||||
SetMaxContext(n int) // Set maximum number of text context tokens to store
|
SetInitialPrompt(prompt string) // Set initial prompt
|
||||||
SetBeamSize(n int) // Set Beam Size
|
|
||||||
SetEntropyThold(t float32) // Set Entropy threshold
|
|
||||||
SetInitialPrompt(prompt string) // Set initial prompt
|
|
||||||
SetTemperature(t float32) // Set temperature
|
|
||||||
SetTemperatureFallback(t float32) // Set temperature incrementation
|
|
||||||
|
|
||||||
// Process mono audio data and return any errors.
|
// Process mono audio data and return any errors.
|
||||||
// If defined, newly generated segments are passed to the
|
// If defined, newly generated segments are passed to the
|
||||||
|
@ -1,91 +0,0 @@
|
|||||||
package whisper_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/ggerganov/whisper.cpp/bindings/go/pkg/whisper"
|
|
||||||
assert "github.com/stretchr/testify/assert"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestNew(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
t.Run("valid model path", func(t *testing.T) {
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("invalid model path", func(t *testing.T) {
|
|
||||||
invalidModelPath := "invalid-model-path.bin"
|
|
||||||
model, err := whisper.New(invalidModelPath)
|
|
||||||
assert.Error(err)
|
|
||||||
assert.Nil(model)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestClose(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
|
|
||||||
err = model.Close()
|
|
||||||
assert.NoError(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestNewContext(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
context, err := model.NewContext()
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(context)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestIsMultilingual(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
isMultilingual := model.IsMultilingual()
|
|
||||||
|
|
||||||
// This returns false since
|
|
||||||
// the model 'models/ggml-small.en.bin'
|
|
||||||
// that is loaded is not multilingual
|
|
||||||
assert.False(isMultilingual)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLanguages(t *testing.T) {
|
|
||||||
assert := assert.New(t)
|
|
||||||
|
|
||||||
model, err := whisper.New(ModelPath)
|
|
||||||
assert.NoError(err)
|
|
||||||
assert.NotNil(model)
|
|
||||||
defer model.Close()
|
|
||||||
|
|
||||||
expectedLanguages := []string{
|
|
||||||
"en", "zh", "de", "es", "ru", "ko", "fr", "ja", "pt", "tr", "pl",
|
|
||||||
"ca", "nl", "ar", "sv", "it", "id", "hi", "fi", "vi", "he", "uk",
|
|
||||||
"el", "ms", "cs", "ro", "da", "hu", "ta", "no", "th", "ur", "hr",
|
|
||||||
"bg", "lt", "la", "mi", "ml", "cy", "sk", "te", "fa", "lv", "bn",
|
|
||||||
"sr", "az", "sl", "kn", "et", "mk", "br", "eu", "is", "hy", "ne",
|
|
||||||
"mn", "bs", "kk", "sq", "sw", "gl", "mr", "pa", "si", "km", "sn",
|
|
||||||
"yo", "so", "af", "oc", "ka", "be", "tg", "sd", "gu", "am", "yi",
|
|
||||||
"lo", "uz", "fo", "ht", "ps", "tk", "nn", "mt", "sa", "lb", "my",
|
|
||||||
"bo", "tl", "mg", "as", "tt", "haw", "ln", "ha", "ba", "jw", "su",
|
|
||||||
}
|
|
||||||
|
|
||||||
actualLanguages := model.Languages()
|
|
||||||
|
|
||||||
assert.Equal(expectedLanguages, actualLanguages)
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
package whisper_test
|
|
||||||
|
|
||||||
const (
|
|
||||||
ModelPath = "../../models/ggml-small.en.bin"
|
|
||||||
SamplePath = "../../samples/jfk.wav"
|
|
||||||
)
|
|
@ -9,7 +9,7 @@ import (
|
|||||||
// CGO
|
// CGO
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#cgo LDFLAGS: -lwhisper -lm -lstdc++ -fopenmp
|
#cgo LDFLAGS: -lwhisper -lm -lstdc++
|
||||||
#cgo darwin LDFLAGS: -framework Accelerate -framework Metal -framework Foundation -framework CoreGraphics
|
#cgo darwin LDFLAGS: -framework Accelerate -framework Metal -framework Foundation -framework CoreGraphics
|
||||||
#include <whisper.h>
|
#include <whisper.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
@ -67,5 +67,5 @@ copy /y ..\..\build\bin\Release\whisper.dll build\generated\resources\main\win32
|
|||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
The license for the Java bindings is the same as the license for the rest of the whisper.cpp project, which is the MIT License. See the `LICENSE` file for more details.
|
The license for the Go bindings is the same as the license for the rest of the whisper.cpp project, which is the MIT License. See the `LICENSE` file for more details.
|
||||||
|
|
||||||
|
@ -181,11 +181,11 @@ public class WhisperFullParams extends Structure {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Flag to suppress non-speech tokens. */
|
/** Flag to suppress non-speech tokens. */
|
||||||
public CBool suppress_nst;
|
public CBool suppress_non_speech_tokens;
|
||||||
|
|
||||||
/** Flag to suppress non-speech tokens. */
|
/** Flag to suppress non-speech tokens. */
|
||||||
public void suppressNonSpeechTokens(boolean enable) {
|
public void suppressNonSpeechTokens(boolean enable) {
|
||||||
suppress_nst = enable ? CBool.TRUE : CBool.FALSE;
|
suppress_non_speech_tokens = enable ? CBool.TRUE : CBool.FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Initial decoding temperature. */
|
/** Initial decoding temperature. */
|
||||||
@ -315,7 +315,7 @@ public class WhisperFullParams extends Structure {
|
|||||||
"print_special", "print_progress", "print_realtime", "print_timestamps", "token_timestamps",
|
"print_special", "print_progress", "print_realtime", "print_timestamps", "token_timestamps",
|
||||||
"thold_pt", "thold_ptsum", "max_len", "split_on_word", "max_tokens", "audio_ctx",
|
"thold_pt", "thold_ptsum", "max_len", "split_on_word", "max_tokens", "audio_ctx",
|
||||||
"tdrz_enable", "suppress_regex", "initial_prompt", "prompt_tokens", "prompt_n_tokens", "language", "detect_language",
|
"tdrz_enable", "suppress_regex", "initial_prompt", "prompt_tokens", "prompt_n_tokens", "language", "detect_language",
|
||||||
"suppress_blank", "suppress_nst", "temperature", "max_initial_ts", "length_penalty",
|
"suppress_blank", "suppress_non_speech_tokens", "temperature", "max_initial_ts", "length_penalty",
|
||||||
"temperature_inc", "entropy_thold", "logprob_thold", "no_speech_thold", "greedy", "beam_search",
|
"temperature_inc", "entropy_thold", "logprob_thold", "no_speech_thold", "greedy", "beam_search",
|
||||||
"new_segment_callback", "new_segment_callback_user_data",
|
"new_segment_callback", "new_segment_callback_user_data",
|
||||||
"progress_callback", "progress_callback_user_data",
|
"progress_callback", "progress_callback_user_data",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "whisper.cpp",
|
"name": "whisper.cpp",
|
||||||
"version": "1.7.4",
|
"version": "1.6.2",
|
||||||
"description": "Whisper speech recognition",
|
"description": "Whisper speech recognition",
|
||||||
"main": "whisper.js",
|
"main": "whisper.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
3
bindings/ruby/.gitignore
vendored
3
bindings/ruby/.gitignore
vendored
@ -1,3 +0,0 @@
|
|||||||
LICENSE
|
|
||||||
pkg/
|
|
||||||
lib/whisper.*
|
|
@ -1,245 +0,0 @@
|
|||||||
whispercpp
|
|
||||||
==========
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Ruby bindings for [whisper.cpp][], an interface of automatic speech recognition model.
|
|
||||||
|
|
||||||
Installation
|
|
||||||
------------
|
|
||||||
|
|
||||||
Install the gem and add to the application's Gemfile by executing:
|
|
||||||
|
|
||||||
$ bundle add whispercpp
|
|
||||||
|
|
||||||
If bundler is not being used to manage dependencies, install the gem by executing:
|
|
||||||
|
|
||||||
$ gem install whispercpp
|
|
||||||
|
|
||||||
Usage
|
|
||||||
-----
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
require "whisper"
|
|
||||||
|
|
||||||
whisper = Whisper::Context.new("base")
|
|
||||||
|
|
||||||
params = Whisper::Params.new(
|
|
||||||
language: "en",
|
|
||||||
offset: 10_000,
|
|
||||||
duration: 60_000,
|
|
||||||
max_text_tokens: 300,
|
|
||||||
translate: true,
|
|
||||||
print_timestamps: false,
|
|
||||||
initial_prompt: "Initial prompt here."
|
|
||||||
)
|
|
||||||
|
|
||||||
whisper.transcribe("path/to/audio.wav", params) do |whole_text|
|
|
||||||
puts whole_text
|
|
||||||
end
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
### Preparing model ###
|
|
||||||
|
|
||||||
Some models are prepared up-front:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
base_en = Whisper::Model.pre_converted_models["base.en"]
|
|
||||||
whisper = Whisper::Context.new(base_en)
|
|
||||||
```
|
|
||||||
|
|
||||||
At first time you use a model, it is downloaded automatically. After that, downloaded cached file is used. To clear cache, call `#clear_cache`:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
Whisper::Model.pre_converted_models["base"].clear_cache
|
|
||||||
```
|
|
||||||
|
|
||||||
You also can use shorthand for pre-converted models:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
whisper = Whisper::Context.new("base.en")
|
|
||||||
```
|
|
||||||
|
|
||||||
You can see the list of prepared model names by `Whisper::Model.pre_converted_models.keys`:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
puts Whisper::Model.pre_converted_models.keys
|
|
||||||
# tiny
|
|
||||||
# tiny.en
|
|
||||||
# tiny-q5_1
|
|
||||||
# tiny.en-q5_1
|
|
||||||
# tiny-q8_0
|
|
||||||
# base
|
|
||||||
# base.en
|
|
||||||
# base-q5_1
|
|
||||||
# base.en-q5_1
|
|
||||||
# base-q8_0
|
|
||||||
# :
|
|
||||||
# :
|
|
||||||
```
|
|
||||||
|
|
||||||
You can also use local model files you prepared:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
whisper = Whisper::Context.new("path/to/your/model.bin")
|
|
||||||
```
|
|
||||||
|
|
||||||
Or, you can download model files:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
whisper = Whisper::Context.new("https://example.net/uri/of/your/model.bin")
|
|
||||||
# Or
|
|
||||||
whisper = Whisper::Context.new(URI("https://example.net/uri/of/your/model.bin"))
|
|
||||||
```
|
|
||||||
|
|
||||||
See [models][] page for details.
|
|
||||||
|
|
||||||
### Preparing audio file ###
|
|
||||||
|
|
||||||
Currently, whisper.cpp accepts only 16-bit WAV files.
|
|
||||||
|
|
||||||
API
|
|
||||||
---
|
|
||||||
|
|
||||||
### Segments ###
|
|
||||||
|
|
||||||
Once `Whisper::Context#transcribe` called, you can retrieve segments by `#each_segment`:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
def format_time(time_ms)
|
|
||||||
sec, decimal_part = time_ms.divmod(1000)
|
|
||||||
min, sec = sec.divmod(60)
|
|
||||||
hour, min = min.divmod(60)
|
|
||||||
"%02d:%02d:%02d.%03d" % [hour, min, sec, decimal_part]
|
|
||||||
end
|
|
||||||
|
|
||||||
whisper
|
|
||||||
.transcribe("path/to/audio.wav", params)
|
|
||||||
.each_segment.with_index do |segment, index|
|
|
||||||
line = "[%{nth}: %{st} --> %{ed}] %{text}" % {
|
|
||||||
nth: index + 1,
|
|
||||||
st: format_time(segment.start_time),
|
|
||||||
ed: format_time(segment.end_time),
|
|
||||||
text: segment.text
|
|
||||||
}
|
|
||||||
line << " (speaker turned)" if segment.speaker_next_turn?
|
|
||||||
puts line
|
|
||||||
end
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
You can also add hook to params called on new segment:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
# Add hook before calling #transcribe
|
|
||||||
params.on_new_segment do |segment|
|
|
||||||
line = "[%{st} --> %{ed}] %{text}" % {
|
|
||||||
st: format_time(segment.start_time),
|
|
||||||
ed: format_time(segment.end_time),
|
|
||||||
text: segment.text
|
|
||||||
}
|
|
||||||
line << " (speaker turned)" if segment.speaker_next_turn?
|
|
||||||
puts line
|
|
||||||
end
|
|
||||||
|
|
||||||
whisper.transcribe("path/to/audio.wav", params)
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
### Models ###
|
|
||||||
|
|
||||||
You can see model information:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
whisper = Whisper::Context.new("base")
|
|
||||||
model = whisper.model
|
|
||||||
|
|
||||||
model.n_vocab # => 51864
|
|
||||||
model.n_audio_ctx # => 1500
|
|
||||||
model.n_audio_state # => 512
|
|
||||||
model.n_audio_head # => 8
|
|
||||||
model.n_audio_layer # => 6
|
|
||||||
model.n_text_ctx # => 448
|
|
||||||
model.n_text_state # => 512
|
|
||||||
model.n_text_head # => 8
|
|
||||||
model.n_text_layer # => 6
|
|
||||||
model.n_mels # => 80
|
|
||||||
model.ftype # => 1
|
|
||||||
model.type # => "base"
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
### Logging ###
|
|
||||||
|
|
||||||
You can set log callback:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
prefix = "[MyApp] "
|
|
||||||
log_callback = ->(level, buffer, user_data) {
|
|
||||||
case level
|
|
||||||
when Whisper::LOG_LEVEL_NONE
|
|
||||||
puts "#{user_data}none: #{buffer}"
|
|
||||||
when Whisper::LOG_LEVEL_INFO
|
|
||||||
puts "#{user_data}info: #{buffer}"
|
|
||||||
when Whisper::LOG_LEVEL_WARN
|
|
||||||
puts "#{user_data}warn: #{buffer}"
|
|
||||||
when Whisper::LOG_LEVEL_ERROR
|
|
||||||
puts "#{user_data}error: #{buffer}"
|
|
||||||
when Whisper::LOG_LEVEL_DEBUG
|
|
||||||
puts "#{user_data}debug: #{buffer}"
|
|
||||||
when Whisper::LOG_LEVEL_CONT
|
|
||||||
puts "#{user_data}same to previous: #{buffer}"
|
|
||||||
end
|
|
||||||
}
|
|
||||||
Whisper.log_set log_callback, prefix
|
|
||||||
```
|
|
||||||
|
|
||||||
Using this feature, you are also able to suppress log:
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
Whisper.log_set ->(level, buffer, user_data) {
|
|
||||||
# do nothing
|
|
||||||
}, nil
|
|
||||||
Whisper::Context.new("base")
|
|
||||||
```
|
|
||||||
|
|
||||||
### Low-level API to transcribe ###
|
|
||||||
|
|
||||||
You can also call `Whisper::Context#full` and `#full_parallel` with a Ruby array as samples. Although `#transcribe` with audio file path is recommended because it extracts PCM samples in C++ and is fast, `#full` and `#full_parallel` give you flexibility.
|
|
||||||
|
|
||||||
```ruby
|
|
||||||
require "whisper"
|
|
||||||
require "wavefile"
|
|
||||||
|
|
||||||
reader = WaveFile::Reader.new("path/to/audio.wav", WaveFile::Format.new(:mono, :float, 16000))
|
|
||||||
samples = reader.enum_for(:each_buffer).map(&:samples).flatten
|
|
||||||
|
|
||||||
whisper = Whisper::Context.new("base")
|
|
||||||
whisper
|
|
||||||
.full(Whisper::Params.new, samples)
|
|
||||||
.each_segment do |segment|
|
|
||||||
puts segment.text
|
|
||||||
end
|
|
||||||
```
|
|
||||||
|
|
||||||
The second argument `samples` may be an array, an object with `length` and `each` method, or a MemoryView. If you can prepare audio data as C array and export it as a MemoryView, whispercpp accepts and works with it with zero copy.
|
|
||||||
|
|
||||||
Development
|
|
||||||
-----------
|
|
||||||
|
|
||||||
% git clone https://github.com/ggerganov/whisper.cpp.git
|
|
||||||
% cd whisper.cpp/bindings/ruby
|
|
||||||
% rake test
|
|
||||||
|
|
||||||
First call of `rake test` builds an extension and downloads a model for testing. After that, you add tests in `tests` directory and modify `ext/ruby_whisper.cpp`.
|
|
||||||
|
|
||||||
If something seems wrong on build, running `rake clean` solves some cases.
|
|
||||||
|
|
||||||
License
|
|
||||||
-------
|
|
||||||
|
|
||||||
The same to [whisper.cpp][].
|
|
||||||
|
|
||||||
[whisper.cpp]: https://github.com/ggerganov/whisper.cpp
|
|
||||||
[models]: https://github.com/ggerganov/whisper.cpp/tree/master/models
|
|
@ -1,66 +1,12 @@
|
|||||||
require 'rake/clean'
|
require 'rake/clean'
|
||||||
require "bundler/gem_tasks"
|
require 'rubygems/package'
|
||||||
require "rake/testtask"
|
|
||||||
require_relative "extsources"
|
|
||||||
|
|
||||||
SOURCES = FileList[]
|
desc 'Build gem'
|
||||||
|
task :package do
|
||||||
EXTSOURCES.each do |src|
|
spec_source = File.read File.join(File.dirname(__FILE__),'whispercpp.gemspec')
|
||||||
basename = src.pathmap("%f")
|
spec = nil
|
||||||
dest = basename == "LICENSE" ? basename : src.pathmap("%{../..,ext}p")
|
# see: http://gist.github.com/16215
|
||||||
dir = dest.pathmap("%d")
|
Thread.new { spec = eval("#{spec_source}") }.join
|
||||||
file src
|
spec.validate
|
||||||
directory dir
|
Gem::Package.build(spec)
|
||||||
file dest => [src, dir] do |t|
|
|
||||||
cp t.source, t.name
|
|
||||||
end
|
|
||||||
SOURCES.include dest
|
|
||||||
end
|
end
|
||||||
|
|
||||||
CLEAN.include SOURCES
|
|
||||||
CLEAN.include FileList["ext/**/*.o", "ext/**/*.metal", "ext/**/*.tmp", "ext/whisper.{so,bundle,dll}"]
|
|
||||||
|
|
||||||
SRC = FileList["ext/*.{c,cpp,h}"]
|
|
||||||
|
|
||||||
task build: SOURCES
|
|
||||||
|
|
||||||
directory "pkg"
|
|
||||||
CLOBBER.include "pkg"
|
|
||||||
|
|
||||||
LIB_NAME = "whisper".ext(RbConfig::CONFIG["DLEXT"])
|
|
||||||
SO_FILE = File.join("ext", LIB_NAME)
|
|
||||||
LIB_FILE = File.join("lib", LIB_NAME)
|
|
||||||
|
|
||||||
file "ext/Makefile" => SRC + ["ext/extconf.rb"] + SOURCES do |t|
|
|
||||||
chdir "ext" do
|
|
||||||
ruby "extconf.rb"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
file SO_FILE => "ext/Makefile" do |t|
|
|
||||||
chdir "ext" do
|
|
||||||
sh "make"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
CLEAN.include SO_FILE
|
|
||||||
|
|
||||||
directory "lib"
|
|
||||||
file LIB_FILE => [SO_FILE, "lib"] do |t|
|
|
||||||
copy t.source, t.name
|
|
||||||
end
|
|
||||||
CLEAN.include LIB_FILE
|
|
||||||
|
|
||||||
Rake::TestTask.new do |t|
|
|
||||||
t.test_files = FileList["tests/test_*.rb"]
|
|
||||||
end
|
|
||||||
|
|
||||||
TEST_MEMORY_VIEW = "tests/jfk_reader/jfk_reader.#{RbConfig::CONFIG['DLEXT']}"
|
|
||||||
file TEST_MEMORY_VIEW => "tests/jfk_reader/jfk_reader.c" do |t|
|
|
||||||
chdir "tests/jfk_reader" do
|
|
||||||
ruby "extconf.rb"
|
|
||||||
sh "make"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
CLEAN.include "tests/jfk_reader/jfk_reader.{o,#{RbConfig::CONFIG['DLEXT']}}"
|
|
||||||
|
|
||||||
task test: [LIB_FILE, TEST_MEMORY_VIEW]
|
|
||||||
|
16
bindings/ruby/ext/.gitignore
vendored
16
bindings/ruby/ext/.gitignore
vendored
@ -1,11 +1,9 @@
|
|||||||
Makefile
|
Makefile
|
||||||
whisper.so
|
ggml.c
|
||||||
|
ggml.h
|
||||||
|
ggml-alloc.c
|
||||||
|
ggml-alloc.h
|
||||||
whisper.bundle
|
whisper.bundle
|
||||||
whisper.dll
|
whisper.cpp
|
||||||
scripts/get-flags.mk
|
whisper.h
|
||||||
*.o
|
dr_wav.h
|
||||||
/*/**/*.c
|
|
||||||
/*/**/*.cpp
|
|
||||||
/*/**/*.h
|
|
||||||
/*/**/*.m
|
|
||||||
/*/**/*.metal
|
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
ggml/src/ggml-cpu/ggml-cpu-cpp.o: \
|
|
||||||
ggml/src/ggml-cpu/ggml-cpu.cpp \
|
|
||||||
ggml/include/ggml-backend.h \
|
|
||||||
ggml/include/ggml.h \
|
|
||||||
ggml/include/ggml-alloc.h \
|
|
||||||
ggml/src/ggml-backend-impl.h \
|
|
||||||
ggml/include/ggml-cpu.h \
|
|
||||||
ggml/src/ggml-impl.h
|
|
||||||
$(CXX) $(CXXFLAGS) -c $< -o $@
|
|
@ -1,10 +1,23 @@
|
|||||||
require 'mkmf'
|
require 'mkmf'
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','whisper.cpp')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','whisper.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','whisper-mel.hpp')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml.c')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-impl.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-alloc.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-alloc.c')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-backend-impl.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-backend.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-backend.c')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-common.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-quants.h')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','ggml-quants.c')} .")
|
||||||
|
system("cp #{File.join(File.dirname(__FILE__),'..','..','..','examples','dr_wav.h')} .")
|
||||||
|
|
||||||
|
|
||||||
# need to use c++ compiler flags
|
# need to use c++ compiler flags
|
||||||
$CXXFLAGS << ' -std=c++17'
|
$CXXFLAGS << ' -std=c++11'
|
||||||
|
|
||||||
$LDFLAGS << ' -lstdc++'
|
|
||||||
|
|
||||||
# Set to true when building binary gems
|
# Set to true when building binary gems
|
||||||
if enable_config('static-stdlib', false)
|
if enable_config('static-stdlib', false)
|
||||||
$LDFLAGS << ' -static-libgcc -static-libstdc++'
|
$LDFLAGS << ' -static-libgcc -static-libstdc++'
|
||||||
@ -15,192 +28,4 @@ if enable_config('march-tune-native', false)
|
|||||||
$CXXFLAGS << ' -march=native -mtune=native'
|
$CXXFLAGS << ' -march=native -mtune=native'
|
||||||
end
|
end
|
||||||
|
|
||||||
if ENV['WHISPER_METAL']
|
|
||||||
$GGML_METAL ||= true
|
|
||||||
$DEPRECATE_WARNING ||= true
|
|
||||||
end
|
|
||||||
|
|
||||||
$UNAME_S = `uname -s`.chomp
|
|
||||||
$UNAME_P = `uname -p`.chomp
|
|
||||||
$UNAME_M = `uname -m`.chomp
|
|
||||||
|
|
||||||
if $UNAME_S == 'Darwin'
|
|
||||||
unless ENV['GGML_NO_METAL']
|
|
||||||
$GGML_METAL ||= true
|
|
||||||
end
|
|
||||||
$GGML_NO_OPENMP ||= true
|
|
||||||
end
|
|
||||||
|
|
||||||
if $GGML_METAL
|
|
||||||
$GGML_METAL_EMBED_LIBRARY = true
|
|
||||||
end
|
|
||||||
|
|
||||||
$MK_CPPFLAGS = '-Iggml/include -Iggml/src -Iggml/src/ggml-cpu -Iinclude -Isrc -Iexamples'
|
|
||||||
$MK_CFLAGS = '-std=c11 -fPIC'
|
|
||||||
$MK_CXXFLAGS = '-std=c++17 -fPIC'
|
|
||||||
$MK_NVCCFLAGS = '-std=c++17'
|
|
||||||
$MK_LDFLAGS = ''
|
|
||||||
|
|
||||||
$OBJ_GGML = []
|
|
||||||
$OBJ_WHISPER = []
|
|
||||||
$OBJ_COMMON = []
|
|
||||||
$OBJ_SDL = []
|
|
||||||
|
|
||||||
$MK_CPPFLAGS << ' -D_XOPEN_SOURCE=600'
|
|
||||||
|
|
||||||
if $UNAME_S == 'Linux'
|
|
||||||
$MK_CPPFLAGS << ' -D_GNU_SOURCE'
|
|
||||||
end
|
|
||||||
|
|
||||||
if $UNAME_S == 'Darwin'
|
|
||||||
$MK_CPPFLAGS << ' -D_DARWIN_C_SOURCE'
|
|
||||||
end
|
|
||||||
|
|
||||||
if ENV['WHISPER_DEBUG']
|
|
||||||
$MK_CFLAGS << ' -O0 -g'
|
|
||||||
$MK_CXXFLAGS << ' -O0 -g'
|
|
||||||
$MK_LDFLAGS << ' -g'
|
|
||||||
$MK_NVCCFLAGS << ' -O0 -g'
|
|
||||||
else
|
|
||||||
$MK_CPPFLAGS << ' -DNDEBUG'
|
|
||||||
$MK_CFLAGS << ' -O3'
|
|
||||||
$MK_CXXFLAGS << ' -O3'
|
|
||||||
$MK_NVCCFLAGS << ' -O3'
|
|
||||||
end
|
|
||||||
|
|
||||||
$WARN_FLAGS =
|
|
||||||
' -Wall' <<
|
|
||||||
' -Wextra' <<
|
|
||||||
' -Wpedantic' <<
|
|
||||||
' -Wcast-qual' <<
|
|
||||||
' -Wno-unused-function'
|
|
||||||
|
|
||||||
$MK_CFLAGS <<
|
|
||||||
$WARN_FLAGS <<
|
|
||||||
' -Wshadow' <<
|
|
||||||
' -Wstrict-prototypes' <<
|
|
||||||
' -Wpointer-arith' <<
|
|
||||||
' -Wmissing-prototypes' <<
|
|
||||||
' -Werror=implicit-int' <<
|
|
||||||
' -Werror=implicit-function-declaration'
|
|
||||||
|
|
||||||
$MK_CXXFLAGS <<
|
|
||||||
$WARN_FLAGS <<
|
|
||||||
' -Wmissing-declarations' <<
|
|
||||||
' -Wmissing-noreturn'
|
|
||||||
|
|
||||||
unless `#{cc_command} #{$LDFLAGS} -Wl,-v 2>&1`.chomp.include? 'dyld-1015.7'
|
|
||||||
$MK_CPPFLAGS << ' -DHAVE_BUGGY_APPLE_LINKER'
|
|
||||||
end
|
|
||||||
|
|
||||||
if %w[Linux Darwin FreeBSD NetBSD OpenBSD Haiku].include? $UNAME_S
|
|
||||||
$MK_CFLAGS << ' -pthread'
|
|
||||||
$MK_CXXFLAGS << ' -pthread'
|
|
||||||
end
|
|
||||||
|
|
||||||
unless $_WIN32
|
|
||||||
$DSO_EXT = '.so'
|
|
||||||
else
|
|
||||||
$DSO_EXT = '.dll'
|
|
||||||
end
|
|
||||||
|
|
||||||
unless ENV['RISCV']
|
|
||||||
if %w[x86_64 i686 amd64].include? $UNAME_M
|
|
||||||
$HOST_CXXFLAGS ||= ''
|
|
||||||
|
|
||||||
$MK_CFLAGS << ' -march=native -mtune=native'
|
|
||||||
$HOST_CXXFLAGS << ' -march=native -mtune=native'
|
|
||||||
end
|
|
||||||
else
|
|
||||||
$MK_CFLAGS << ' -march=rv64gcv -mabi=lp64d'
|
|
||||||
$MK_CXXFLAGS << ' -march=rv64gcv -mabi=lp64d'
|
|
||||||
end
|
|
||||||
|
|
||||||
unless ENV['GGML_NO_ACCELERATE']
|
|
||||||
if $UNAME_S == 'Darwin'
|
|
||||||
$MK_CPPFLAGS << ' -DGGML_USE_ACCELERATE -DGGML_USE_BLAS -DGGML_BLAS_USE_ACCELERATE'
|
|
||||||
$MK_CPPFLAGS << ' -DACCELERATE_NEW_LAPACK'
|
|
||||||
$MK_CPPFLAGS << ' -DACCELERATE_LAPACK_ILP64'
|
|
||||||
$MK_LDFLAGS << ' -framework Accelerate'
|
|
||||||
$OBJ_GGML << 'ggml/src/ggml-blas/ggml-blas.o'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
if ENV['GGML_OPENBLAS']
|
|
||||||
$MK_CPPFLAGS << " -DGGML_USE_BLAS #{`pkg-config --cflags-only-I openblas`.chomp}"
|
|
||||||
$MK_CFLAGS << " #{`pkg-config --cflags-only-other openblas)`.chomp}"
|
|
||||||
$MK_LDFLAGS << " #{`pkg-config --libs openblas`}"
|
|
||||||
$OBJ_GGML << 'ggml/src/ggml-blas/ggml-blas.o'
|
|
||||||
end
|
|
||||||
|
|
||||||
if ENV['GGML_OPENBLAS64']
|
|
||||||
$MK_CPPFLAGS << " -DGGML_USE_BLAS #{`pkg-config --cflags-only-I openblas64`.chomp}"
|
|
||||||
$MK_CFLAGS << " #{`pkg-config --cflags-only-other openblas64)`.chomp}"
|
|
||||||
$MK_LDFLAGS << " #{`pkg-config --libs openblas64`}"
|
|
||||||
$OBJ_GGML << 'ggml/src/ggml-blas/ggml-blas.o'
|
|
||||||
end
|
|
||||||
|
|
||||||
if $GGML_METAL
|
|
||||||
$MK_CPPFLAGS << ' -DGGML_USE_METAL'
|
|
||||||
$MK_LDFLAGS << ' -framework Foundation -framework Metal -framework MetalKit'
|
|
||||||
$OBJ_GGML << 'ggml/src/ggml-metal/ggml-metal.o'
|
|
||||||
|
|
||||||
if ENV['GGML_METAL_NDEBUG']
|
|
||||||
$MK_CPPFLAGS << ' -DGGML_METAL_NDEBUG'
|
|
||||||
end
|
|
||||||
|
|
||||||
if $GGML_METAL_EMBED_LIBRARY
|
|
||||||
$MK_CPPFLAGS << ' -DGGML_METAL_EMBED_LIBRARY'
|
|
||||||
$OBJ_GGML << 'ggml/src/ggml-metal/ggml-metal-embed.o'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
$OBJ_GGML <<
|
|
||||||
'ggml/src/ggml.o' <<
|
|
||||||
'ggml/src/ggml-alloc.o' <<
|
|
||||||
'ggml/src/ggml-backend.o' <<
|
|
||||||
'ggml/src/ggml-backend-reg.o' <<
|
|
||||||
'ggml/src/ggml-opt.o' <<
|
|
||||||
'ggml/src/ggml-quants.o' <<
|
|
||||||
'ggml/src/ggml-threading.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu-cpp.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu-aarch64.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu-hbm.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu-quants.o' <<
|
|
||||||
'ggml/src/ggml-cpu/ggml-cpu-traits.o'
|
|
||||||
|
|
||||||
$OBJ_WHISPER <<
|
|
||||||
'src/whisper.o'
|
|
||||||
|
|
||||||
$objs = $OBJ_GGML + $OBJ_WHISPER + $OBJ_COMMON + $OBJ_SDL
|
|
||||||
$objs <<
|
|
||||||
"ruby_whisper.o" <<
|
|
||||||
"ruby_whisper_context.o" <<
|
|
||||||
"ruby_whisper_transcribe.o" <<
|
|
||||||
"ruby_whisper_params.o" <<
|
|
||||||
"ruby_whisper_error.o" <<
|
|
||||||
"ruby_whisper_segment.o" <<
|
|
||||||
"ruby_whisper_model.o"
|
|
||||||
|
|
||||||
$CPPFLAGS = "#{$MK_CPPFLAGS} #{$CPPFLAGS}"
|
|
||||||
$CFLAGS = "#{$CPPFLAGS} #{$MK_CFLAGS} #{$GF_CFLAGS} #{$CFLAGS}"
|
|
||||||
$BASE_CXXFLAGS = "#{$MK_CXXFLAGS} #{$CXXFLAGS}"
|
|
||||||
$CXXFLAGS = "#{$BASE_CXXFLAGS} #{$HOST_CXXFLAGS} #{$GF_CXXFLAGS} #{$CPPFLAGS}"
|
|
||||||
$NVCCFLAGS = "#{$MK_NVCCFLAGS} #{$NVCCFLAGS}"
|
|
||||||
$LDFLAGS = "#{$MK_LDFLAGS} #{$LDFLAGS}"
|
|
||||||
|
|
||||||
create_makefile('whisper')
|
create_makefile('whisper')
|
||||||
|
|
||||||
File.open 'Makefile', 'a' do |file|
|
|
||||||
file.puts 'include scripts/get-flags.mk'
|
|
||||||
file.puts 'include cpu.mk'
|
|
||||||
|
|
||||||
if $GGML_METAL
|
|
||||||
file.puts 'include metal.mk'
|
|
||||||
|
|
||||||
if $GGML_METAL_EMBED_LIBRARY
|
|
||||||
file.puts 'include metal-embed.mk'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
141
bindings/ruby/ext/ggml-backend-impl.h
Normal file
141
bindings/ruby/ext/ggml-backend-impl.h
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
// ggml-backend internal header
|
||||||
|
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend buffer
|
||||||
|
//
|
||||||
|
|
||||||
|
// buffer type
|
||||||
|
typedef void * ggml_backend_buffer_type_context_t;
|
||||||
|
|
||||||
|
struct ggml_backend_buffer_type_i {
|
||||||
|
const char * (*GGML_CALL get_name) (ggml_backend_buffer_type_t buft);
|
||||||
|
ggml_backend_buffer_t (*GGML_CALL alloc_buffer) (ggml_backend_buffer_type_t buft, size_t size);
|
||||||
|
size_t (*GGML_CALL get_alignment) (ggml_backend_buffer_type_t buft); // tensor alignment
|
||||||
|
size_t (*GGML_CALL get_max_size) (ggml_backend_buffer_type_t buft); // allocation max size
|
||||||
|
size_t (*GGML_CALL get_alloc_size) (ggml_backend_buffer_type_t buft, const struct ggml_tensor * tensor); // data size needed to allocate the tensor, including padding
|
||||||
|
bool (*GGML_CALL supports_backend)(ggml_backend_buffer_type_t buft, ggml_backend_t backend); // check if the buffer type is usable by the backend
|
||||||
|
// check if tensor data is in host memory
|
||||||
|
// should be equivalent to supports_backend(buft, ggml_backend_cpu_init())
|
||||||
|
bool (*GGML_CALL is_host) (ggml_backend_buffer_type_t buft);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ggml_backend_buffer_type {
|
||||||
|
struct ggml_backend_buffer_type_i iface;
|
||||||
|
ggml_backend_buffer_type_context_t context;
|
||||||
|
};
|
||||||
|
|
||||||
|
// buffer
|
||||||
|
typedef void * ggml_backend_buffer_context_t;
|
||||||
|
|
||||||
|
struct ggml_backend_buffer_i {
|
||||||
|
const char * (*GGML_CALL get_name) (ggml_backend_buffer_t buffer);
|
||||||
|
void (*GGML_CALL free_buffer)(ggml_backend_buffer_t buffer);
|
||||||
|
void * (*GGML_CALL get_base) (ggml_backend_buffer_t buffer);
|
||||||
|
void (*GGML_CALL init_tensor)(ggml_backend_buffer_t buffer, struct ggml_tensor * tensor);
|
||||||
|
void (*GGML_CALL set_tensor) (ggml_backend_buffer_t buffer, struct ggml_tensor * tensor, const void * data, size_t offset, size_t size);
|
||||||
|
void (*GGML_CALL get_tensor) (ggml_backend_buffer_t buffer, const struct ggml_tensor * tensor, void * data, size_t offset, size_t size);
|
||||||
|
bool (*GGML_CALL cpy_tensor) (ggml_backend_buffer_t buffer, const struct ggml_tensor * src, struct ggml_tensor * dst); // dst is in the buffer, src may be in any buffer
|
||||||
|
void (*GGML_CALL clear) (ggml_backend_buffer_t buffer, uint8_t value);
|
||||||
|
void (*GGML_CALL reset) (ggml_backend_buffer_t buffer); // reset any internal state due to tensor initialization, such as tensor extras
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ggml_backend_buffer {
|
||||||
|
struct ggml_backend_buffer_i iface;
|
||||||
|
ggml_backend_buffer_type_t buft;
|
||||||
|
ggml_backend_buffer_context_t context;
|
||||||
|
size_t size;
|
||||||
|
enum ggml_backend_buffer_usage usage;
|
||||||
|
};
|
||||||
|
|
||||||
|
GGML_CALL ggml_backend_buffer_t ggml_backend_buffer_init(
|
||||||
|
ggml_backend_buffer_type_t buft,
|
||||||
|
struct ggml_backend_buffer_i iface,
|
||||||
|
ggml_backend_buffer_context_t context,
|
||||||
|
size_t size);
|
||||||
|
|
||||||
|
// do not use directly, use ggml_backend_tensor_copy instead
|
||||||
|
bool ggml_backend_buffer_copy_tensor(const struct ggml_tensor * src, struct ggml_tensor * dst);
|
||||||
|
|
||||||
|
// buffer that contains a collection of buffers
|
||||||
|
GGML_CALL ggml_backend_buffer_t ggml_backend_multi_buffer_alloc_buffer(ggml_backend_buffer_t * buffers, size_t n_buffers);
|
||||||
|
GGML_CALL bool ggml_backend_buffer_is_multi_buffer(ggml_backend_buffer_t buffer);
|
||||||
|
GGML_CALL void ggml_backend_multi_buffer_set_usage(ggml_backend_buffer_t buffer, enum ggml_backend_buffer_usage usage);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend
|
||||||
|
//
|
||||||
|
|
||||||
|
typedef void * ggml_backend_context_t;
|
||||||
|
|
||||||
|
struct ggml_backend_i {
|
||||||
|
const char * (*GGML_CALL get_name)(ggml_backend_t backend);
|
||||||
|
|
||||||
|
void (*GGML_CALL free)(ggml_backend_t backend);
|
||||||
|
|
||||||
|
// buffer allocation
|
||||||
|
ggml_backend_buffer_type_t (*GGML_CALL get_default_buffer_type)(ggml_backend_t backend);
|
||||||
|
|
||||||
|
// (optional) asynchronous tensor data access
|
||||||
|
void (*GGML_CALL set_tensor_async)(ggml_backend_t backend, struct ggml_tensor * tensor, const void * data, size_t offset, size_t size);
|
||||||
|
void (*GGML_CALL get_tensor_async)(ggml_backend_t backend, const struct ggml_tensor * tensor, void * data, size_t offset, size_t size);
|
||||||
|
bool (*GGML_CALL cpy_tensor_async)(ggml_backend_t backend_src, ggml_backend_t backend_dst, const struct ggml_tensor * src, struct ggml_tensor * dst);
|
||||||
|
|
||||||
|
// (optional) complete all pending operations
|
||||||
|
void (*GGML_CALL synchronize)(ggml_backend_t backend);
|
||||||
|
|
||||||
|
// compute graph with a plan (not used currently)
|
||||||
|
ggml_backend_graph_plan_t (*GGML_CALL graph_plan_create) (ggml_backend_t backend, const struct ggml_cgraph * cgraph);
|
||||||
|
void (*GGML_CALL graph_plan_free) (ggml_backend_t backend, ggml_backend_graph_plan_t plan);
|
||||||
|
|
||||||
|
// compute graph with a plan
|
||||||
|
enum ggml_status (*GGML_CALL graph_plan_compute)(ggml_backend_t backend, ggml_backend_graph_plan_t plan);
|
||||||
|
// compute graph without a plan (async)
|
||||||
|
enum ggml_status (*GGML_CALL graph_compute) (ggml_backend_t backend, struct ggml_cgraph * cgraph);
|
||||||
|
|
||||||
|
// check if the backend supports an operation
|
||||||
|
bool (*GGML_CALL supports_op)(ggml_backend_t backend, const struct ggml_tensor * op);
|
||||||
|
|
||||||
|
// check if the backend wants to run an operation, even if the weights are allocated in a CPU buffer
|
||||||
|
// these should be expensive operations with large batch sizes that may benefit from running on this backend
|
||||||
|
// even if the weight has to be copied from the CPU temporarily
|
||||||
|
bool (*GGML_CALL offload_op)(ggml_backend_t backend, const struct ggml_tensor * op);
|
||||||
|
|
||||||
|
// (optional) event synchronization
|
||||||
|
ggml_backend_event_t (*GGML_CALL event_new) (ggml_backend_t backend);
|
||||||
|
void (*GGML_CALL event_free) (ggml_backend_event_t event);
|
||||||
|
void (*GGML_CALL event_record) (ggml_backend_event_t event);
|
||||||
|
void (*GGML_CALL event_wait) (ggml_backend_t backend, ggml_backend_event_t event);
|
||||||
|
void (*GGML_CALL event_synchronize) (ggml_backend_event_t event);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ggml_backend {
|
||||||
|
ggml_guid_t guid;
|
||||||
|
|
||||||
|
struct ggml_backend_i iface;
|
||||||
|
ggml_backend_context_t context;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ggml_backend_event {
|
||||||
|
ggml_backend_t backend;
|
||||||
|
void * context;
|
||||||
|
};
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend registry
|
||||||
|
//
|
||||||
|
|
||||||
|
typedef ggml_backend_t (*GGML_CALL ggml_backend_init_fn)(const char * params, void * user_data);
|
||||||
|
|
||||||
|
GGML_CALL void ggml_backend_register(const char * name, ggml_backend_init_fn init_fn, ggml_backend_buffer_type_t default_buffer_type, void * user_data);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
2095
bindings/ruby/ext/ggml-backend.c
Normal file
2095
bindings/ruby/ext/ggml-backend.c
Normal file
File diff suppressed because it is too large
Load Diff
233
bindings/ruby/ext/ggml-backend.h
Normal file
233
bindings/ruby/ext/ggml-backend.h
Normal file
@ -0,0 +1,233 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-alloc.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct ggml_backend_buffer_type * ggml_backend_buffer_type_t;
|
||||||
|
typedef struct ggml_backend_buffer * ggml_backend_buffer_t;
|
||||||
|
typedef struct ggml_backend_event * ggml_backend_event_t;
|
||||||
|
typedef struct ggml_backend * ggml_backend_t;
|
||||||
|
typedef void * ggml_backend_graph_plan_t;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend buffer
|
||||||
|
//
|
||||||
|
|
||||||
|
// buffer type
|
||||||
|
GGML_API const char * ggml_backend_buft_name (ggml_backend_buffer_type_t buft);
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_t ggml_backend_buft_alloc_buffer (ggml_backend_buffer_type_t buft, size_t size);
|
||||||
|
GGML_API size_t ggml_backend_buft_get_alignment (ggml_backend_buffer_type_t buft);
|
||||||
|
GGML_API size_t ggml_backend_buft_get_max_size (ggml_backend_buffer_type_t buft);
|
||||||
|
GGML_API GGML_CALL size_t ggml_backend_buft_get_alloc_size (ggml_backend_buffer_type_t buft, struct ggml_tensor * tensor);
|
||||||
|
GGML_API bool ggml_backend_buft_supports_backend(ggml_backend_buffer_type_t buft, ggml_backend_t backend);
|
||||||
|
GGML_API bool ggml_backend_buft_is_host (ggml_backend_buffer_type_t buft);
|
||||||
|
|
||||||
|
// buffer
|
||||||
|
enum ggml_backend_buffer_usage {
|
||||||
|
GGML_BACKEND_BUFFER_USAGE_ANY = 0,
|
||||||
|
GGML_BACKEND_BUFFER_USAGE_WEIGHTS = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
GGML_API const char * ggml_backend_buffer_name (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API void ggml_backend_buffer_free (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API void * ggml_backend_buffer_get_base (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API size_t ggml_backend_buffer_get_size (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_buffer_init_tensor (ggml_backend_buffer_t buffer, struct ggml_tensor * tensor);
|
||||||
|
GGML_API size_t ggml_backend_buffer_get_alignment (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API size_t ggml_backend_buffer_get_max_size (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API size_t ggml_backend_buffer_get_alloc_size(ggml_backend_buffer_t buffer, struct ggml_tensor * tensor);
|
||||||
|
GGML_API void ggml_backend_buffer_clear (ggml_backend_buffer_t buffer, uint8_t value);
|
||||||
|
GGML_API bool ggml_backend_buffer_is_host (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API void ggml_backend_buffer_set_usage (ggml_backend_buffer_t buffer, enum ggml_backend_buffer_usage usage);
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_buffer_get_type (ggml_backend_buffer_t buffer);
|
||||||
|
GGML_API void ggml_backend_buffer_reset (ggml_backend_buffer_t buffer);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend
|
||||||
|
//
|
||||||
|
|
||||||
|
GGML_API ggml_guid_t ggml_backend_guid(ggml_backend_t backend);
|
||||||
|
GGML_API const char * ggml_backend_name(ggml_backend_t backend);
|
||||||
|
GGML_API void ggml_backend_free(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_get_default_buffer_type(ggml_backend_t backend);
|
||||||
|
GGML_API ggml_backend_buffer_t ggml_backend_alloc_buffer(ggml_backend_t backend, size_t size);
|
||||||
|
GGML_API size_t ggml_backend_get_alignment(ggml_backend_t backend);
|
||||||
|
GGML_API size_t ggml_backend_get_max_size(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_tensor_set_async(ggml_backend_t backend, struct ggml_tensor * tensor, const void * data, size_t offset, size_t size);
|
||||||
|
GGML_API void ggml_backend_tensor_get_async(ggml_backend_t backend, const struct ggml_tensor * tensor, void * data, size_t offset, size_t size);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL void ggml_backend_tensor_set( struct ggml_tensor * tensor, const void * data, size_t offset, size_t size);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_tensor_get(const struct ggml_tensor * tensor, void * data, size_t offset, size_t size);
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_synchronize(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API ggml_backend_graph_plan_t ggml_backend_graph_plan_create(ggml_backend_t backend, struct ggml_cgraph * cgraph);
|
||||||
|
GGML_API void ggml_backend_graph_plan_free (ggml_backend_t backend, ggml_backend_graph_plan_t plan);
|
||||||
|
|
||||||
|
GGML_API enum ggml_status ggml_backend_graph_plan_compute (ggml_backend_t backend, ggml_backend_graph_plan_t plan);
|
||||||
|
GGML_API enum ggml_status ggml_backend_graph_compute (ggml_backend_t backend, struct ggml_cgraph * cgraph);
|
||||||
|
GGML_API enum ggml_status ggml_backend_graph_compute_async(ggml_backend_t backend, struct ggml_cgraph * cgraph);
|
||||||
|
GGML_API bool ggml_backend_supports_op(ggml_backend_t backend, const struct ggml_tensor * op);
|
||||||
|
GGML_API bool ggml_backend_offload_op(ggml_backend_t backend, const struct ggml_tensor * op);
|
||||||
|
|
||||||
|
// tensor copy between different backends
|
||||||
|
GGML_API void ggml_backend_tensor_copy(struct ggml_tensor * src, struct ggml_tensor * dst);
|
||||||
|
|
||||||
|
// asynchronous copy
|
||||||
|
// the copy is performed after all the currently queued operations in backend_src
|
||||||
|
// backend_dst will wait for the copy to complete before performing other operations
|
||||||
|
// automatic fallback to sync copy if async is not supported
|
||||||
|
GGML_API void ggml_backend_tensor_copy_async(ggml_backend_t backend_src, ggml_backend_t backend_dst, struct ggml_tensor * src, struct ggml_tensor * dst);
|
||||||
|
|
||||||
|
// events
|
||||||
|
GGML_API ggml_backend_event_t ggml_backend_event_new (ggml_backend_t backend);
|
||||||
|
GGML_API void ggml_backend_event_free (ggml_backend_event_t event);
|
||||||
|
GGML_API void ggml_backend_event_record (ggml_backend_event_t event);
|
||||||
|
GGML_API void ggml_backend_event_synchronize(ggml_backend_event_t event);
|
||||||
|
GGML_API void ggml_backend_event_wait (ggml_backend_t backend, ggml_backend_event_t event); // wait async on event
|
||||||
|
|
||||||
|
//
|
||||||
|
// CPU backend
|
||||||
|
//
|
||||||
|
|
||||||
|
GGML_API ggml_backend_t ggml_backend_cpu_init(void);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL bool ggml_backend_is_cpu (ggml_backend_t backend);
|
||||||
|
GGML_API void ggml_backend_cpu_set_n_threads (ggml_backend_t backend_cpu, int n_threads);
|
||||||
|
GGML_API void ggml_backend_cpu_set_abort_callback(ggml_backend_t backend_cpu, ggml_abort_callback abort_callback, void * abort_callback_data);
|
||||||
|
|
||||||
|
// Create a backend buffer from an existing pointer
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_t ggml_backend_cpu_buffer_from_ptr(void * ptr, size_t size);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_cpu_buffer_type(void);
|
||||||
|
|
||||||
|
#ifdef GGML_USE_CPU_HBM
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_cpu_hbm_buffer_type(void);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend registry
|
||||||
|
//
|
||||||
|
|
||||||
|
// The backend registry is a registry of all the available backends, and allows initializing backends in a generic way
|
||||||
|
|
||||||
|
GGML_API size_t ggml_backend_reg_get_count(void);
|
||||||
|
GGML_API size_t ggml_backend_reg_find_by_name(const char * name);
|
||||||
|
GGML_API ggml_backend_t ggml_backend_reg_init_backend_from_str(const char * backend_str); // str is name[:params]
|
||||||
|
GGML_API const char * ggml_backend_reg_get_name(size_t i);
|
||||||
|
GGML_API ggml_backend_t ggml_backend_reg_init_backend(size_t i, const char * params); // params is backend-specific
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_reg_get_default_buffer_type(size_t i);
|
||||||
|
GGML_API ggml_backend_buffer_t ggml_backend_reg_alloc_buffer(size_t i, size_t size);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Backend scheduler
|
||||||
|
//
|
||||||
|
|
||||||
|
// The backend scheduler allows for multiple backends to be used together
|
||||||
|
// Handles compute buffer allocation, assignment of tensors to backends, and copying of tensors between backends
|
||||||
|
// The backends are selected based on:
|
||||||
|
// - the backend that supports the operation
|
||||||
|
// - the location of the pre-allocated tensors (e.g. the weights)
|
||||||
|
/*
|
||||||
|
Example usage:
|
||||||
|
|
||||||
|
// operations that use tensors allocated in a buffer with USAGE_WEIGHTS will be assigned
|
||||||
|
// preferrably to run on the same backend as the buffer
|
||||||
|
ggml_backend_buffer_set_usage(buf_weights, GGML_BACKEND_BUFFER_USAGE_WEIGHTS);
|
||||||
|
|
||||||
|
sched = ggml_backend_sched_new({backend_gpu, backend_gpu2, backend_cpu}, NULL, num_backends, GGML_DEFAULT_GRAPH_SIZE, false);
|
||||||
|
|
||||||
|
// initialize buffers from a max size graph (optional)
|
||||||
|
reserve_graph = build_graph(sched, max_batch_size);
|
||||||
|
|
||||||
|
// manually assign nodes to a backend (optional, should not be needed in most cases)
|
||||||
|
struct ggml_tensor * node = ggml_mul_mat(ctx, ...);
|
||||||
|
ggml_backend_sched_set_tensor_backend(sched, node, backend_gpu);
|
||||||
|
|
||||||
|
ggml_backend_sched_reserve(sched, reserve_graph);
|
||||||
|
|
||||||
|
// compute
|
||||||
|
graph = build_graph(sched);
|
||||||
|
ggml_backend_sched_graph_compute(sched, graph);
|
||||||
|
|
||||||
|
// if there are graph inputs:
|
||||||
|
ggml_backend_sched_reset(sched);
|
||||||
|
ggml_backend_sched_alloc_graph(sched, graph);
|
||||||
|
ggml_backend_tensor_set(input_tensor, ...);
|
||||||
|
ggml_backend_sched_graph_compute(sched, graph);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct ggml_backend_sched;
|
||||||
|
typedef struct ggml_backend_sched * ggml_backend_sched_t;
|
||||||
|
|
||||||
|
// when ask == true, the scheduler wants to know if the user wants to observe this node
|
||||||
|
// this allows the scheduler to batch nodes together in order to evaluate them in a single call
|
||||||
|
//
|
||||||
|
// when ask == false, the scheduler is passing the node tensor to the user for observation
|
||||||
|
// if the user returns false, the scheduler will cancel the graph compute
|
||||||
|
//
|
||||||
|
typedef bool (*ggml_backend_sched_eval_callback)(struct ggml_tensor * t, bool ask, void * user_data);
|
||||||
|
|
||||||
|
// Initialize a backend scheduler
|
||||||
|
GGML_API ggml_backend_sched_t ggml_backend_sched_new(ggml_backend_t * backends, ggml_backend_buffer_type_t * bufts, int n_backends, size_t graph_size, bool parallel);
|
||||||
|
GGML_API void ggml_backend_sched_free(ggml_backend_sched_t sched);
|
||||||
|
|
||||||
|
// Initialize backend buffers from a measure graph
|
||||||
|
GGML_API bool ggml_backend_sched_reserve(ggml_backend_sched_t sched, struct ggml_cgraph * measure_graph);
|
||||||
|
|
||||||
|
// Get the number of splits of the last graph
|
||||||
|
GGML_API int ggml_backend_sched_get_n_splits(ggml_backend_sched_t sched);
|
||||||
|
GGML_API int ggml_backend_sched_get_n_copies(ggml_backend_sched_t sched);
|
||||||
|
|
||||||
|
GGML_API size_t ggml_backend_sched_get_buffer_size(ggml_backend_sched_t sched, ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_sched_set_tensor_backend(ggml_backend_sched_t sched, struct ggml_tensor * node, ggml_backend_t backend);
|
||||||
|
GGML_API ggml_backend_t ggml_backend_sched_get_tensor_backend(ggml_backend_sched_t sched, struct ggml_tensor * node);
|
||||||
|
|
||||||
|
// Allocate and compute graph on the backend scheduler
|
||||||
|
GGML_API bool ggml_backend_sched_alloc_graph(ggml_backend_sched_t sched, struct ggml_cgraph * graph);
|
||||||
|
GGML_API enum ggml_status ggml_backend_sched_graph_compute(ggml_backend_sched_t sched, struct ggml_cgraph * graph);
|
||||||
|
GGML_API enum ggml_status ggml_backend_sched_graph_compute_async(ggml_backend_sched_t sched, struct ggml_cgraph * graph);
|
||||||
|
GGML_API void ggml_backend_sched_synchronize(ggml_backend_sched_t sched);
|
||||||
|
|
||||||
|
// Reset all assignments and allocators - must be called before changing the node backends
|
||||||
|
GGML_API void ggml_backend_sched_reset(ggml_backend_sched_t sched);
|
||||||
|
|
||||||
|
// Set a callback to be called for each resulting node during graph compute
|
||||||
|
GGML_API void ggml_backend_sched_set_eval_callback(ggml_backend_sched_t sched, ggml_backend_sched_eval_callback callback, void * user_data);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Utils
|
||||||
|
//
|
||||||
|
|
||||||
|
struct ggml_backend_graph_copy {
|
||||||
|
ggml_backend_buffer_t buffer;
|
||||||
|
struct ggml_context * ctx_allocated;
|
||||||
|
struct ggml_context * ctx_unallocated;
|
||||||
|
struct ggml_cgraph * graph;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Copy a graph to a different backend
|
||||||
|
GGML_API struct ggml_backend_graph_copy ggml_backend_graph_copy(ggml_backend_t backend, struct ggml_cgraph * graph);
|
||||||
|
GGML_API void ggml_backend_graph_copy_free(struct ggml_backend_graph_copy copy);
|
||||||
|
|
||||||
|
typedef bool (*GGML_CALL ggml_backend_eval_callback)(int node_index, struct ggml_tensor * t1, struct ggml_tensor * t2, void * user_data);
|
||||||
|
|
||||||
|
// Compare the output of two backends
|
||||||
|
GGML_API bool ggml_backend_compare_graph_backend(ggml_backend_t backend1, ggml_backend_t backend2, struct ggml_cgraph * graph, ggml_backend_eval_callback callback, void * user_data);
|
||||||
|
|
||||||
|
// Tensor initialization
|
||||||
|
GGML_API void ggml_backend_tensor_alloc(ggml_backend_buffer_t buffer, struct ggml_tensor * tensor, void * addr);
|
||||||
|
GGML_API void ggml_backend_view_init(ggml_backend_buffer_t buffer, struct ggml_tensor * tensor);
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
1853
bindings/ruby/ext/ggml-common.h
Normal file
1853
bindings/ruby/ext/ggml-common.h
Normal file
File diff suppressed because it is too large
Load Diff
43
bindings/ruby/ext/ggml-cuda.h
Normal file
43
bindings/ruby/ext/ggml-cuda.h
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#ifdef GGML_USE_HIPBLAS
|
||||||
|
#define GGML_CUDA_NAME "ROCm"
|
||||||
|
#define GGML_CUBLAS_NAME "hipBLAS"
|
||||||
|
#else
|
||||||
|
#define GGML_CUDA_NAME "CUDA"
|
||||||
|
#define GGML_CUBLAS_NAME "cuBLAS"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define GGML_CUDA_MAX_DEVICES 16
|
||||||
|
|
||||||
|
// backend API
|
||||||
|
GGML_API GGML_CALL ggml_backend_t ggml_backend_cuda_init(int device);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL bool ggml_backend_is_cuda(ggml_backend_t backend);
|
||||||
|
|
||||||
|
// device buffer
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_cuda_buffer_type(int device);
|
||||||
|
|
||||||
|
// split tensor buffer that splits matrices by rows across multiple devices
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_cuda_split_buffer_type(const float * tensor_split);
|
||||||
|
|
||||||
|
// pinned host buffer for use with the CPU backend for faster copies between CPU and GPU
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_cuda_host_buffer_type(void);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL int ggml_backend_cuda_get_device_count(void);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_cuda_get_device_description(int device, char * description, size_t description_size);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_cuda_get_device_memory(int device, size_t * free, size_t * total);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL bool ggml_backend_cuda_register_host_buffer(void * buffer, size_t size);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_cuda_unregister_host_buffer(void * buffer);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
272
bindings/ruby/ext/ggml-impl.h
Normal file
272
bindings/ruby/ext/ggml-impl.h
Normal file
@ -0,0 +1,272 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
|
||||||
|
// GGML internal header
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h> // load `stdlib.h` before other headers to work around MinGW bug: https://sourceforge.net/p/mingw-w64/bugs/192/
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <string.h> // memcpy
|
||||||
|
#include <math.h> // fabsf
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// static_assert should be a #define, but if it's not,
|
||||||
|
// fall back to the _Static_assert C11 keyword.
|
||||||
|
// if C99 - static_assert is noop
|
||||||
|
// ref: https://stackoverflow.com/a/53923785/4039976
|
||||||
|
#ifndef __cplusplus
|
||||||
|
#ifndef static_assert
|
||||||
|
#if defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201100L)
|
||||||
|
#define static_assert(cond, msg) _Static_assert(cond, msg)
|
||||||
|
#else
|
||||||
|
#define static_assert(cond, msg) struct global_scope_noop_trick
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// __FMA__ and __F16C__ are not defined in MSVC, however they are implied with AVX2/AVX512
|
||||||
|
#if defined(_MSC_VER) && (defined(__AVX2__) || defined(__AVX512F__))
|
||||||
|
#ifndef __FMA__
|
||||||
|
#define __FMA__
|
||||||
|
#endif
|
||||||
|
#ifndef __F16C__
|
||||||
|
#define __F16C__
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// __SSE3__ and __SSSE3__ are not defined in MSVC, but SSE3/SSSE3 are present when AVX/AVX2/AVX512 are available
|
||||||
|
#if defined(_MSC_VER) && (defined(__AVX__) || defined(__AVX2__) || defined(__AVX512F__))
|
||||||
|
#ifndef __SSE3__
|
||||||
|
#define __SSE3__
|
||||||
|
#endif
|
||||||
|
#ifndef __SSSE3__
|
||||||
|
#define __SSSE3__
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// 16-bit float
|
||||||
|
// on Arm, we use __fp16
|
||||||
|
// on x86, we use uint16_t
|
||||||
|
#if defined(__ARM_NEON) && !defined(_MSC_VER)
|
||||||
|
|
||||||
|
// if YCM cannot find <arm_neon.h>, make a symbolic link to it, for example:
|
||||||
|
//
|
||||||
|
// $ ln -sfn /Library/Developer/CommandLineTools/usr/lib/clang/13.1.6/include/arm_neon.h ./src/
|
||||||
|
//
|
||||||
|
#include <arm_neon.h>
|
||||||
|
|
||||||
|
typedef __fp16 ggml_fp16_internal_t;
|
||||||
|
|
||||||
|
#define GGML_COMPUTE_FP16_TO_FP32(x) ggml_compute_fp16_to_fp32(x)
|
||||||
|
#define GGML_COMPUTE_FP32_TO_FP16(x) ggml_compute_fp32_to_fp16(x)
|
||||||
|
|
||||||
|
#define GGML_FP16_TO_FP32(x) ggml_compute_fp16_to_fp32(x)
|
||||||
|
|
||||||
|
static inline float ggml_compute_fp16_to_fp32(ggml_fp16_t h) {
|
||||||
|
ggml_fp16_internal_t tmp;
|
||||||
|
memcpy(&tmp, &h, sizeof(ggml_fp16_t));
|
||||||
|
return (float)tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ggml_fp16_t ggml_compute_fp32_to_fp16(float f) {
|
||||||
|
ggml_fp16_t res;
|
||||||
|
ggml_fp16_internal_t tmp = f;
|
||||||
|
memcpy(&res, &tmp, sizeof(ggml_fp16_t));
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
typedef uint16_t ggml_fp16_internal_t;
|
||||||
|
|
||||||
|
#ifdef __wasm_simd128__
|
||||||
|
#include <wasm_simd128.h>
|
||||||
|
#else
|
||||||
|
#ifdef __POWER9_VECTOR__
|
||||||
|
#include <altivec.h>
|
||||||
|
#undef bool
|
||||||
|
#define bool _Bool
|
||||||
|
#else
|
||||||
|
#if defined(_MSC_VER) || defined(__MINGW32__)
|
||||||
|
#include <intrin.h>
|
||||||
|
#else
|
||||||
|
#if defined(__AVX__) || defined(__AVX2__) || defined(__AVX512F__) || defined(__SSSE3__) || defined(__SSE3__)
|
||||||
|
#if !defined(__riscv)
|
||||||
|
#include <immintrin.h>
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __riscv_v_intrinsic
|
||||||
|
#include <riscv_vector.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __F16C__
|
||||||
|
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
#define GGML_COMPUTE_FP16_TO_FP32(x) _mm_cvtss_f32(_mm_cvtph_ps(_mm_cvtsi32_si128(x)))
|
||||||
|
#define GGML_COMPUTE_FP32_TO_FP16(x) _mm_extract_epi16(_mm_cvtps_ph(_mm_set_ss(x), 0), 0)
|
||||||
|
#else
|
||||||
|
#define GGML_COMPUTE_FP16_TO_FP32(x) _cvtsh_ss(x)
|
||||||
|
#define GGML_COMPUTE_FP32_TO_FP16(x) _cvtss_sh(x, 0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#elif defined(__POWER9_VECTOR__)
|
||||||
|
|
||||||
|
#define GGML_COMPUTE_FP16_TO_FP32(x) ggml_compute_fp16_to_fp32(x)
|
||||||
|
#define GGML_COMPUTE_FP32_TO_FP16(x) ggml_compute_fp32_to_fp16(x)
|
||||||
|
/* the inline asm below is about 12% faster than the lookup method */
|
||||||
|
#define GGML_FP16_TO_FP32(x) GGML_COMPUTE_FP16_TO_FP32(x)
|
||||||
|
#define GGML_FP32_TO_FP16(x) GGML_COMPUTE_FP32_TO_FP16(x)
|
||||||
|
|
||||||
|
static inline float ggml_compute_fp16_to_fp32(ggml_fp16_t h) {
|
||||||
|
register float f;
|
||||||
|
register double d;
|
||||||
|
__asm__(
|
||||||
|
"mtfprd %0,%2\n"
|
||||||
|
"xscvhpdp %0,%0\n"
|
||||||
|
"frsp %1,%0\n" :
|
||||||
|
/* temp */ "=d"(d),
|
||||||
|
/* out */ "=f"(f):
|
||||||
|
/* in */ "r"(h));
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ggml_fp16_t ggml_compute_fp32_to_fp16(float f) {
|
||||||
|
register double d;
|
||||||
|
register ggml_fp16_t r;
|
||||||
|
__asm__( /* xscvdphp can work on double or single precision */
|
||||||
|
"xscvdphp %0,%2\n"
|
||||||
|
"mffprd %1,%0\n" :
|
||||||
|
/* temp */ "=d"(d),
|
||||||
|
/* out */ "=r"(r):
|
||||||
|
/* in */ "f"(f));
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
// FP16 <-> FP32
|
||||||
|
// ref: https://github.com/Maratyszcza/FP16
|
||||||
|
|
||||||
|
static inline float fp32_from_bits(uint32_t w) {
|
||||||
|
union {
|
||||||
|
uint32_t as_bits;
|
||||||
|
float as_value;
|
||||||
|
} fp32;
|
||||||
|
fp32.as_bits = w;
|
||||||
|
return fp32.as_value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline uint32_t fp32_to_bits(float f) {
|
||||||
|
union {
|
||||||
|
float as_value;
|
||||||
|
uint32_t as_bits;
|
||||||
|
} fp32;
|
||||||
|
fp32.as_value = f;
|
||||||
|
return fp32.as_bits;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline float ggml_compute_fp16_to_fp32(ggml_fp16_t h) {
|
||||||
|
const uint32_t w = (uint32_t) h << 16;
|
||||||
|
const uint32_t sign = w & UINT32_C(0x80000000);
|
||||||
|
const uint32_t two_w = w + w;
|
||||||
|
|
||||||
|
const uint32_t exp_offset = UINT32_C(0xE0) << 23;
|
||||||
|
#if defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) || defined(__GNUC__) && !defined(__STRICT_ANSI__)
|
||||||
|
const float exp_scale = 0x1.0p-112f;
|
||||||
|
#else
|
||||||
|
const float exp_scale = fp32_from_bits(UINT32_C(0x7800000));
|
||||||
|
#endif
|
||||||
|
const float normalized_value = fp32_from_bits((two_w >> 4) + exp_offset) * exp_scale;
|
||||||
|
|
||||||
|
const uint32_t magic_mask = UINT32_C(126) << 23;
|
||||||
|
const float magic_bias = 0.5f;
|
||||||
|
const float denormalized_value = fp32_from_bits((two_w >> 17) | magic_mask) - magic_bias;
|
||||||
|
|
||||||
|
const uint32_t denormalized_cutoff = UINT32_C(1) << 27;
|
||||||
|
const uint32_t result = sign |
|
||||||
|
(two_w < denormalized_cutoff ? fp32_to_bits(denormalized_value) : fp32_to_bits(normalized_value));
|
||||||
|
return fp32_from_bits(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ggml_fp16_t ggml_compute_fp32_to_fp16(float f) {
|
||||||
|
#if defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) || defined(__GNUC__) && !defined(__STRICT_ANSI__)
|
||||||
|
const float scale_to_inf = 0x1.0p+112f;
|
||||||
|
const float scale_to_zero = 0x1.0p-110f;
|
||||||
|
#else
|
||||||
|
const float scale_to_inf = fp32_from_bits(UINT32_C(0x77800000));
|
||||||
|
const float scale_to_zero = fp32_from_bits(UINT32_C(0x08800000));
|
||||||
|
#endif
|
||||||
|
float base = (fabsf(f) * scale_to_inf) * scale_to_zero;
|
||||||
|
|
||||||
|
const uint32_t w = fp32_to_bits(f);
|
||||||
|
const uint32_t shl1_w = w + w;
|
||||||
|
const uint32_t sign = w & UINT32_C(0x80000000);
|
||||||
|
uint32_t bias = shl1_w & UINT32_C(0xFF000000);
|
||||||
|
if (bias < UINT32_C(0x71000000)) {
|
||||||
|
bias = UINT32_C(0x71000000);
|
||||||
|
}
|
||||||
|
|
||||||
|
base = fp32_from_bits((bias >> 1) + UINT32_C(0x07800000)) + base;
|
||||||
|
const uint32_t bits = fp32_to_bits(base);
|
||||||
|
const uint32_t exp_bits = (bits >> 13) & UINT32_C(0x00007C00);
|
||||||
|
const uint32_t mantissa_bits = bits & UINT32_C(0x00000FFF);
|
||||||
|
const uint32_t nonsign = exp_bits + mantissa_bits;
|
||||||
|
return (sign >> 16) | (shl1_w > UINT32_C(0xFF000000) ? UINT16_C(0x7E00) : nonsign);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define GGML_COMPUTE_FP16_TO_FP32(x) ggml_compute_fp16_to_fp32(x)
|
||||||
|
#define GGML_COMPUTE_FP32_TO_FP16(x) ggml_compute_fp32_to_fp16(x)
|
||||||
|
|
||||||
|
#endif // __F16C__
|
||||||
|
|
||||||
|
#endif // __ARM_NEON
|
||||||
|
|
||||||
|
// precomputed f32 table for f16 (256 KB)
|
||||||
|
// defined in ggml.c, initialized in ggml_init()
|
||||||
|
extern float ggml_table_f32_f16[1 << 16];
|
||||||
|
|
||||||
|
// On ARM NEON, it's quicker to directly convert x -> x instead of calling into ggml_lookup_fp16_to_fp32,
|
||||||
|
// so we define GGML_FP16_TO_FP32 and GGML_FP32_TO_FP16 elsewhere for NEON.
|
||||||
|
// This is also true for POWER9.
|
||||||
|
#if !defined(GGML_FP16_TO_FP32)
|
||||||
|
inline static float ggml_lookup_fp16_to_fp32(ggml_fp16_t f) {
|
||||||
|
uint16_t s;
|
||||||
|
memcpy(&s, &f, sizeof(uint16_t));
|
||||||
|
return ggml_table_f32_f16[s];
|
||||||
|
}
|
||||||
|
|
||||||
|
#define GGML_FP16_TO_FP32(x) ggml_lookup_fp16_to_fp32(x)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(GGML_FP32_TO_FP16)
|
||||||
|
#define GGML_FP32_TO_FP16(x) GGML_COMPUTE_FP32_TO_FP16(x)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define GGML_HASHTABLE_FULL ((size_t)-1)
|
||||||
|
#define GGML_HASHTABLE_ALREADY_EXISTS ((size_t)-2)
|
||||||
|
|
||||||
|
struct ggml_hash_set ggml_hash_set_new(size_t size);
|
||||||
|
|
||||||
|
bool ggml_hash_contains (const struct ggml_hash_set hash_set, struct ggml_tensor * key);
|
||||||
|
|
||||||
|
// returns GGML_HASHTABLE_FULL if table is full, otherwise the current index of the key or where it should be inserted
|
||||||
|
size_t ggml_hash_find (const struct ggml_hash_set hash_set, struct ggml_tensor * key);
|
||||||
|
|
||||||
|
// returns GGML_HASHTABLE_ALREADY_EXISTS if key already exists, index otherwise, asserts if table is full
|
||||||
|
size_t ggml_hash_insert ( struct ggml_hash_set hash_set, struct ggml_tensor * key);
|
||||||
|
|
||||||
|
// return index, asserts if table is full
|
||||||
|
size_t ggml_hash_find_or_insert( struct ggml_hash_set hash_set, struct ggml_tensor * key);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
46
bindings/ruby/ext/ggml-kompute.h
Normal file
46
bindings/ruby/ext/ggml-kompute.h
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct ggml_vk_device {
|
||||||
|
int index;
|
||||||
|
int type; // same as VkPhysicalDeviceType
|
||||||
|
size_t heapSize;
|
||||||
|
const char * name;
|
||||||
|
const char * vendor;
|
||||||
|
int subgroupSize;
|
||||||
|
uint64_t bufferAlignment;
|
||||||
|
uint64_t maxAlloc;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ggml_vk_device * ggml_vk_available_devices(size_t memoryRequired, size_t * count);
|
||||||
|
bool ggml_vk_get_device(struct ggml_vk_device * device, size_t memoryRequired, const char * name);
|
||||||
|
bool ggml_vk_has_vulkan(void);
|
||||||
|
bool ggml_vk_has_device(void);
|
||||||
|
struct ggml_vk_device ggml_vk_current_device(void);
|
||||||
|
|
||||||
|
//
|
||||||
|
// backend API
|
||||||
|
//
|
||||||
|
|
||||||
|
// forward declaration
|
||||||
|
typedef struct ggml_backend * ggml_backend_t;
|
||||||
|
|
||||||
|
GGML_API ggml_backend_t ggml_backend_kompute_init(int device);
|
||||||
|
|
||||||
|
GGML_API bool ggml_backend_is_kompute(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_kompute_buffer_type(int device);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
66
bindings/ruby/ext/ggml-metal.h
Normal file
66
bindings/ruby/ext/ggml-metal.h
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
// An interface allowing to compute ggml_cgraph with Metal
|
||||||
|
//
|
||||||
|
// This is a fully functional interface that extends ggml with GPU support for Apple devices.
|
||||||
|
// A similar interface can be created for other GPU backends (e.g. Vulkan, CUDA, OpenCL, etc.)
|
||||||
|
//
|
||||||
|
// How it works?
|
||||||
|
//
|
||||||
|
// As long as your program can create and evaluate a ggml_cgraph on the CPU, you can use this
|
||||||
|
// interface to evaluate the same graph on the GPU. Instead of using ggml_graph_compute(), you
|
||||||
|
// use ggml_metal_graph_compute() (or ggml_vulkan_graph_compute(), etc.)
|
||||||
|
//
|
||||||
|
// You only need to make sure that all memory buffers that you used during the graph creation
|
||||||
|
// are mapped to the device memory with the ggml_metal_add_buffer() function. This mapping is
|
||||||
|
// used during the graph evaluation to determine the arguments of the compute kernels.
|
||||||
|
//
|
||||||
|
// Synchronization between device and host memory (for example for input and output tensors)
|
||||||
|
// is done with the ggml_metal_set_tensor() and ggml_metal_get_tensor() functions.
|
||||||
|
//
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
// max memory buffers that can be mapped to the device
|
||||||
|
#define GGML_METAL_MAX_BUFFERS 64
|
||||||
|
|
||||||
|
struct ggml_tensor;
|
||||||
|
struct ggml_cgraph;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//
|
||||||
|
// backend API
|
||||||
|
// user-code should use only these functions
|
||||||
|
//
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_metal_log_set_callback(ggml_log_callback log_callback, void * user_data);
|
||||||
|
|
||||||
|
GGML_API ggml_backend_t ggml_backend_metal_init(void);
|
||||||
|
|
||||||
|
GGML_API bool ggml_backend_is_metal(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_t ggml_backend_metal_buffer_from_ptr(void * data, size_t size, size_t max_size);
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_metal_set_n_cb(ggml_backend_t backend, int n_cb);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_metal_buffer_type(void);
|
||||||
|
|
||||||
|
// helper to check if the device supports a specific family
|
||||||
|
// ideally, the user code should be doing these checks
|
||||||
|
// ref: https://developer.apple.com/metal/Metal-Feature-Set-Tables.pdf
|
||||||
|
GGML_API bool ggml_backend_metal_supports_family(ggml_backend_t backend, int family);
|
||||||
|
|
||||||
|
// capture all command buffers committed the next time `ggml_backend_graph_compute` is called
|
||||||
|
GGML_API void ggml_backend_metal_capture_next_compute(ggml_backend_t backend);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
36
bindings/ruby/ext/ggml-opencl.h
Normal file
36
bindings/ruby/ext/ggml-opencl.h
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
GGML_API void ggml_cl_init(void);
|
||||||
|
|
||||||
|
GGML_API void ggml_cl_mul(const struct ggml_tensor * src0, const struct ggml_tensor * src1, struct ggml_tensor * dst);
|
||||||
|
GGML_API void ggml_cl_add(const struct ggml_tensor * src0, const struct ggml_tensor * src1, struct ggml_tensor * dst);
|
||||||
|
GGML_API bool ggml_cl_can_mul_mat(const struct ggml_tensor * src0, const struct ggml_tensor * src1, const struct ggml_tensor * dst);
|
||||||
|
GGML_API size_t ggml_cl_mul_mat_get_wsize(const struct ggml_tensor * src0, const struct ggml_tensor * src1, struct ggml_tensor * dst);
|
||||||
|
GGML_API void ggml_cl_mul_mat(const struct ggml_tensor * src0, const struct ggml_tensor * src1, struct ggml_tensor * dst, void * wdata, size_t wsize);
|
||||||
|
|
||||||
|
// GGML_API void * ggml_cl_host_malloc(size_t size);
|
||||||
|
// GGML_API void ggml_cl_host_free(void * ptr);
|
||||||
|
|
||||||
|
GGML_API void ggml_cl_free_data(const struct ggml_tensor* tensor);
|
||||||
|
|
||||||
|
GGML_API void ggml_cl_transform_tensor(void * data, struct ggml_tensor * tensor);
|
||||||
|
|
||||||
|
// backend API
|
||||||
|
|
||||||
|
// GGML_API ggml_backend_t ggml_backend_opencl_init(void);
|
||||||
|
|
||||||
|
// GGML_API bool ggml_backend_is_opencl(ggml_backend_t backend);
|
||||||
|
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_opencl_buffer_type(void);
|
||||||
|
// GGML_API ggml_backend_buffer_type_t ggml_backend_opencl_host_buffer_type(void);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
12678
bindings/ruby/ext/ggml-quants.c
Normal file
12678
bindings/ruby/ext/ggml-quants.c
Normal file
File diff suppressed because it is too large
Load Diff
133
bindings/ruby/ext/ggml-quants.h
Normal file
133
bindings/ruby/ext/ggml-quants.h
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#define GGML_COMMON_DECL_C
|
||||||
|
#include "ggml-common.h"
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
|
||||||
|
// GGML internal header
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Quantization
|
||||||
|
void quantize_row_q4_0_reference(const float * GGML_RESTRICT x, block_q4_0 * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q4_1_reference(const float * GGML_RESTRICT x, block_q4_1 * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_0_reference(const float * GGML_RESTRICT x, block_q5_0 * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_1_reference(const float * GGML_RESTRICT x, block_q5_1 * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_0_reference(const float * GGML_RESTRICT x, block_q8_0 * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_1_reference(const float * GGML_RESTRICT x, block_q8_1 * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void quantize_row_q2_K_reference(const float * GGML_RESTRICT x, block_q2_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q3_K_reference(const float * GGML_RESTRICT x, block_q3_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q4_K_reference(const float * GGML_RESTRICT x, block_q4_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_K_reference(const float * GGML_RESTRICT x, block_q5_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q6_K_reference(const float * GGML_RESTRICT x, block_q6_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_K_reference(const float * GGML_RESTRICT x, block_q8_K * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void quantize_row_iq3_xxs_reference(const float * GGML_RESTRICT x, block_iq3_xxs * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq4_nl_reference (const float * GGML_RESTRICT x, block_iq4_nl * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq4_xs_reference (const float * GGML_RESTRICT x, block_iq4_xs * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq3_s_reference (const float * GGML_RESTRICT x, block_iq3_s * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq2_s_reference (const float * GGML_RESTRICT x, block_iq2_s * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void quantize_row_q4_0(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q4_1(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_0(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_1(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_0(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_1(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void quantize_row_q2_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q3_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q4_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q5_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q6_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_q8_K(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void quantize_row_iq3_xxs(const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq4_nl (const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq4_xs (const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq3_s (const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
void quantize_row_iq2_s (const float * GGML_RESTRICT x, void * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
// Dequantization
|
||||||
|
void dequantize_row_q4_0(const block_q4_0 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q4_1(const block_q4_1 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q5_0(const block_q5_0 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q5_1(const block_q5_1 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q8_0(const block_q8_0 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
//void dequantize_row_q8_1(const block_q8_1 * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void dequantize_row_q2_K(const block_q2_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q3_K(const block_q3_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q4_K(const block_q4_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q5_K(const block_q5_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q6_K(const block_q6_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_q8_K(const block_q8_K * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
void dequantize_row_iq2_xxs(const block_iq2_xxs * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq2_xs (const block_iq2_xs * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq2_s (const block_iq2_s * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq3_xxs(const block_iq3_xxs * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq1_s (const block_iq1_s * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq1_m (const block_iq1_m * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq4_nl (const block_iq4_nl * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq4_xs (const block_iq4_xs * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
void dequantize_row_iq3_s (const block_iq3_s * GGML_RESTRICT x, float * GGML_RESTRICT y, int64_t k);
|
||||||
|
|
||||||
|
// Dot product
|
||||||
|
void ggml_vec_dot_q4_0_q8_0(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q4_1_q8_1(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q5_0_q8_0(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q5_1_q8_1(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q8_0_q8_0(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
|
||||||
|
void ggml_vec_dot_q2_K_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q3_K_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q4_K_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q5_K_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_q6_K_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
|
||||||
|
void ggml_vec_dot_iq2_xxs_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq2_xs_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq2_s_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq3_xxs_q8_K(int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq1_s_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq1_m_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq4_nl_q8_0 (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq4_xs_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
void ggml_vec_dot_iq3_s_q8_K (int n, float * GGML_RESTRICT s, size_t bs, const void * GGML_RESTRICT vx, size_t bx, const void * GGML_RESTRICT vy, size_t by, int nrc);
|
||||||
|
|
||||||
|
// Quantization utilizing an importance matrix (a.k.a. "Activation aWare Quantization")
|
||||||
|
size_t quantize_iq2_xxs(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq2_xs (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq2_s (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq3_xxs(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq1_s (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq1_m (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq4_nl (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq4_xs (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_iq3_s (const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
|
||||||
|
size_t quantize_q2_K(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q3_K(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q4_K(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q5_K(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q6_K(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q4_0(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q4_1(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q5_0(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q5_1(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
size_t quantize_q8_0(const float * GGML_RESTRICT src, void * GGML_RESTRICT dst, int64_t nrows, int64_t n_per_row, const float * imatrix);
|
||||||
|
|
||||||
|
void iq2xs_init_impl(enum ggml_type type);
|
||||||
|
void iq2xs_free_impl(enum ggml_type type);
|
||||||
|
void iq3xs_init_impl(int grid_size);
|
||||||
|
void iq3xs_free_impl(int grid_size);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
49
bindings/ruby/ext/ggml-sycl.h
Normal file
49
bindings/ruby/ext/ggml-sycl.h
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
//
|
||||||
|
// MIT license
|
||||||
|
// Copyright (C) 2024 Intel Corporation
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
//
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define GGML_SYCL_MAX_DEVICES 48
|
||||||
|
#define GGML_SYCL_NAME "SYCL"
|
||||||
|
|
||||||
|
// backend API
|
||||||
|
GGML_API ggml_backend_t ggml_backend_sycl_init(int device);
|
||||||
|
|
||||||
|
// devide buffer
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_sycl_buffer_type(int device);
|
||||||
|
|
||||||
|
// split tensor buffer that splits matrices by rows across multiple devices
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_sycl_split_buffer_type(const float * tensor_split);
|
||||||
|
|
||||||
|
// pinned host buffer for use with the CPU backend for faster copies between CPU and GPU
|
||||||
|
GGML_API ggml_backend_buffer_type_t ggml_backend_sycl_host_buffer_type(void);
|
||||||
|
|
||||||
|
GGML_API void ggml_backend_sycl_print_sycl_devices(void);
|
||||||
|
GGML_API GGML_CALL void ggml_sycl_get_gpu_list(int *id_list, int max_len);
|
||||||
|
GGML_API GGML_CALL void ggml_sycl_get_device_description(int device, char *description, size_t description_size);
|
||||||
|
GGML_API GGML_CALL int ggml_backend_sycl_get_device_count();
|
||||||
|
GGML_API GGML_CALL void ggml_backend_sycl_get_device_memory(int device, size_t *free, size_t *total);
|
||||||
|
GGML_API GGML_CALL int ggml_backend_sycl_get_device_index(int device_id);
|
||||||
|
|
||||||
|
// TODO: these are temporary
|
||||||
|
// ref: https://github.com/ggerganov/llama.cpp/pull/6022#issuecomment-1992615670
|
||||||
|
GGML_API GGML_CALL int ggml_backend_sycl_get_device_id(int device_index);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_sycl_set_single_device_mode(int main_gpu_id);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_sycl_set_mul_device_mode();
|
||||||
|
|
||||||
|
// SYCL doesn't support registering host memory, keep here for reference
|
||||||
|
// GGML_API GGML_CALL bool ggml_backend_sycl_register_host_buffer(void * buffer, size_t size);
|
||||||
|
// GGML_API GGML_CALL void ggml_backend_sycl_unregister_host_buffer(void * buffer);
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
29
bindings/ruby/ext/ggml-vulkan.h
Normal file
29
bindings/ruby/ext/ggml-vulkan.h
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ggml.h"
|
||||||
|
#include "ggml-backend.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define GGML_VK_NAME "Vulkan"
|
||||||
|
#define GGML_VK_MAX_DEVICES 16
|
||||||
|
|
||||||
|
GGML_API void ggml_vk_instance_init(void);
|
||||||
|
|
||||||
|
// backend API
|
||||||
|
GGML_API GGML_CALL ggml_backend_t ggml_backend_vk_init(size_t dev_num);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL bool ggml_backend_is_vk(ggml_backend_t backend);
|
||||||
|
GGML_API GGML_CALL int ggml_backend_vk_get_device_count(void);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_vk_get_device_description(int device, char * description, size_t description_size);
|
||||||
|
GGML_API GGML_CALL void ggml_backend_vk_get_device_memory(int device, size_t * free, size_t * total);
|
||||||
|
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_vk_buffer_type(size_t dev_num);
|
||||||
|
// pinned host buffer for use with the CPU backend for faster copies between CPU and GPU
|
||||||
|
GGML_API GGML_CALL ggml_backend_buffer_type_t ggml_backend_vk_host_buffer_type(void);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
@ -1,17 +0,0 @@
|
|||||||
ggml/src/ggml-metal/ggml-metal-embed.o: \
|
|
||||||
ggml/src/ggml-metal/ggml-metal.metal \
|
|
||||||
ggml/src/ggml-metal/ggml-metal-impl.h \
|
|
||||||
ggml/src/ggml-common.h
|
|
||||||
@echo "Embedding Metal library"
|
|
||||||
@sed -e '/__embed_ggml-common.h__/r ggml/src/ggml-common.h' -e '/__embed_ggml-common.h__/d' < ggml/src/ggml-metal/ggml-metal.metal > ggml/src/ggml-metal/ggml-metal-embed.metal.tmp
|
|
||||||
@sed -e '/#include "ggml-metal-impl.h"/r ggml/src/ggml-metal/ggml-metal-impl.h' -e '/#include "ggml-metal-impl.h"/d' < ggml/src/ggml-metal/ggml-metal-embed.metal.tmp > ggml/src/ggml-metal/ggml-metal-embed.metal
|
|
||||||
$(eval TEMP_ASSEMBLY=$(shell mktemp -d))
|
|
||||||
@echo ".section __DATA, __ggml_metallib" > $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
@echo ".globl _ggml_metallib_start" >> $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
@echo "_ggml_metallib_start:" >> $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
@echo ".incbin \"ggml/src/ggml-metal/ggml-metal-embed.metal\"" >> $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
@echo ".globl _ggml_metallib_end" >> $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
@echo "_ggml_metallib_end:" >> $(TEMP_ASSEMBLY)/ggml-metal-embed.s
|
|
||||||
$(CC) $(CFLAGS) -c $(TEMP_ASSEMBLY)/ggml-metal-embed.s -o $@
|
|
||||||
@rm -f ${TEMP_ASSEMBLY}/ggml-metal-embed.s
|
|
||||||
@rmdir ${TEMP_ASSEMBLY}
|
|
@ -1,6 +0,0 @@
|
|||||||
ggml/src/ggml-metal/ggml-metal.o: \
|
|
||||||
ggml/src/ggml-metal/ggml-metal.m \
|
|
||||||
ggml/src/ggml-metal/ggml-metal-impl.h \
|
|
||||||
ggml/include/ggml-metal.h \
|
|
||||||
ggml/include/ggml.h
|
|
||||||
$(CC) $(CFLAGS) -c $< -o $@
|
|
@ -1,164 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include <ruby/memory_view.h>
|
|
||||||
#include "ruby_whisper.h"
|
|
||||||
|
|
||||||
VALUE mWhisper;
|
|
||||||
VALUE cContext;
|
|
||||||
VALUE cParams;
|
|
||||||
VALUE eError;
|
|
||||||
|
|
||||||
VALUE cSegment;
|
|
||||||
VALUE cModel;
|
|
||||||
|
|
||||||
ID id_to_s;
|
|
||||||
ID id_call;
|
|
||||||
ID id___method__;
|
|
||||||
ID id_to_enum;
|
|
||||||
ID id_length;
|
|
||||||
ID id_next;
|
|
||||||
ID id_new;
|
|
||||||
ID id_to_path;
|
|
||||||
ID id_URI;
|
|
||||||
ID id_pre_converted_models;
|
|
||||||
|
|
||||||
static bool is_log_callback_finalized = false;
|
|
||||||
|
|
||||||
// High level API
|
|
||||||
extern VALUE ruby_whisper_segment_allocate(VALUE klass);
|
|
||||||
|
|
||||||
extern void init_ruby_whisper_context(VALUE *mWhisper);
|
|
||||||
extern void init_ruby_whisper_params(VALUE *mWhisper);
|
|
||||||
extern void init_ruby_whisper_error(VALUE *mWhisper);
|
|
||||||
extern void init_ruby_whisper_segment(VALUE *mWhisper, VALUE *cSegment);
|
|
||||||
extern void init_ruby_whisper_model(VALUE *mWhisper);
|
|
||||||
extern void register_callbacks(ruby_whisper_params *rwp, VALUE *context);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* lang_max_id -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE ruby_whisper_s_lang_max_id(VALUE self) {
|
|
||||||
return INT2NUM(whisper_lang_max_id());
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* lang_id(lang_name) -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE ruby_whisper_s_lang_id(VALUE self, VALUE lang) {
|
|
||||||
const char * lang_str = StringValueCStr(lang);
|
|
||||||
const int id = whisper_lang_id(lang_str);
|
|
||||||
if (-1 == id) {
|
|
||||||
rb_raise(rb_eArgError, "language not found: %s", lang_str);
|
|
||||||
}
|
|
||||||
return INT2NUM(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* lang_str(lang_id) -> String
|
|
||||||
*/
|
|
||||||
static VALUE ruby_whisper_s_lang_str(VALUE self, VALUE id) {
|
|
||||||
const int lang_id = NUM2INT(id);
|
|
||||||
const char * str = whisper_lang_str(lang_id);
|
|
||||||
if (NULL == str) {
|
|
||||||
rb_raise(rb_eIndexError, "id %d outside of language id", lang_id);
|
|
||||||
}
|
|
||||||
return rb_str_new2(str);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* lang_str(lang_id) -> String
|
|
||||||
*/
|
|
||||||
static VALUE ruby_whisper_s_lang_str_full(VALUE self, VALUE id) {
|
|
||||||
const int lang_id = NUM2INT(id);
|
|
||||||
const char * str_full = whisper_lang_str_full(lang_id);
|
|
||||||
if (NULL == str_full) {
|
|
||||||
rb_raise(rb_eIndexError, "id %d outside of language id", lang_id);
|
|
||||||
}
|
|
||||||
return rb_str_new2(str_full);
|
|
||||||
}
|
|
||||||
|
|
||||||
static VALUE ruby_whisper_s_finalize_log_callback(VALUE self, VALUE id) {
|
|
||||||
is_log_callback_finalized = true;
|
|
||||||
return Qnil;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
ruby_whisper_log_callback(enum ggml_log_level level, const char * buffer, void * user_data) {
|
|
||||||
if (is_log_callback_finalized) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
VALUE log_callback = rb_iv_get(mWhisper, "log_callback");
|
|
||||||
VALUE udata = rb_iv_get(mWhisper, "user_data");
|
|
||||||
rb_funcall(log_callback, id_call, 3, INT2NUM(level), rb_str_new2(buffer), udata);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* log_set ->(level, buffer, user_data) { ... }, user_data -> nil
|
|
||||||
*/
|
|
||||||
static VALUE ruby_whisper_s_log_set(VALUE self, VALUE log_callback, VALUE user_data) {
|
|
||||||
VALUE old_callback = rb_iv_get(self, "log_callback");
|
|
||||||
if (!NIL_P(old_callback)) {
|
|
||||||
rb_undefine_finalizer(old_callback);
|
|
||||||
}
|
|
||||||
|
|
||||||
rb_iv_set(self, "log_callback", log_callback);
|
|
||||||
rb_iv_set(self, "user_data", user_data);
|
|
||||||
|
|
||||||
VALUE finalize_log_callback = rb_funcall(mWhisper, rb_intern("method"), 1, rb_str_new2("finalize_log_callback"));
|
|
||||||
rb_define_finalizer(log_callback, finalize_log_callback);
|
|
||||||
|
|
||||||
whisper_log_set(ruby_whisper_log_callback, NULL);
|
|
||||||
|
|
||||||
return Qnil;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void rb_whisper_model_mark(ruby_whisper_model *rwm) {
|
|
||||||
rb_gc_mark(rwm->context);
|
|
||||||
}
|
|
||||||
|
|
||||||
static VALUE ruby_whisper_model_allocate(VALUE klass) {
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
rwm = ALLOC(ruby_whisper_model);
|
|
||||||
return Data_Wrap_Struct(klass, rb_whisper_model_mark, RUBY_DEFAULT_FREE, rwm);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Init_whisper() {
|
|
||||||
id_to_s = rb_intern("to_s");
|
|
||||||
id_call = rb_intern("call");
|
|
||||||
id___method__ = rb_intern("__method__");
|
|
||||||
id_to_enum = rb_intern("to_enum");
|
|
||||||
id_length = rb_intern("length");
|
|
||||||
id_next = rb_intern("next");
|
|
||||||
id_new = rb_intern("new");
|
|
||||||
id_to_path = rb_intern("to_path");
|
|
||||||
id_URI = rb_intern("URI");
|
|
||||||
id_pre_converted_models = rb_intern("pre_converted_models");
|
|
||||||
|
|
||||||
mWhisper = rb_define_module("Whisper");
|
|
||||||
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_NONE", INT2NUM(GGML_LOG_LEVEL_NONE));
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_INFO", INT2NUM(GGML_LOG_LEVEL_INFO));
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_WARN", INT2NUM(GGML_LOG_LEVEL_WARN));
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_ERROR", INT2NUM(GGML_LOG_LEVEL_ERROR));
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_DEBUG", INT2NUM(GGML_LOG_LEVEL_DEBUG));
|
|
||||||
rb_define_const(mWhisper, "LOG_LEVEL_CONT", INT2NUM(GGML_LOG_LEVEL_CONT));
|
|
||||||
|
|
||||||
rb_define_singleton_method(mWhisper, "lang_max_id", ruby_whisper_s_lang_max_id, 0);
|
|
||||||
rb_define_singleton_method(mWhisper, "lang_id", ruby_whisper_s_lang_id, 1);
|
|
||||||
rb_define_singleton_method(mWhisper, "lang_str", ruby_whisper_s_lang_str, 1);
|
|
||||||
rb_define_singleton_method(mWhisper, "lang_str_full", ruby_whisper_s_lang_str_full, 1);
|
|
||||||
rb_define_singleton_method(mWhisper, "log_set", ruby_whisper_s_log_set, 2);
|
|
||||||
rb_define_private_method(rb_singleton_class(mWhisper), "finalize_log_callback", ruby_whisper_s_finalize_log_callback, 1);
|
|
||||||
|
|
||||||
init_ruby_whisper_context(&mWhisper);
|
|
||||||
init_ruby_whisper_params(&mWhisper);
|
|
||||||
init_ruby_whisper_error(&mWhisper);
|
|
||||||
init_ruby_whisper_segment(&mWhisper, &cContext);
|
|
||||||
init_ruby_whisper_model(&mWhisper);
|
|
||||||
|
|
||||||
rb_require("whisper/model/uri");
|
|
||||||
}
|
|
418
bindings/ruby/ext/ruby_whisper.cpp
Normal file
418
bindings/ruby/ext/ruby_whisper.cpp
Normal file
@ -0,0 +1,418 @@
|
|||||||
|
#include <ruby.h>
|
||||||
|
#include "ruby_whisper.h"
|
||||||
|
#define DR_WAV_IMPLEMENTATION
|
||||||
|
#include "dr_wav.h"
|
||||||
|
#include <cmath>
|
||||||
|
#include <fstream>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <string>
|
||||||
|
#include <thread>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define BOOL_PARAMS_SETTER(self, prop, value) \
|
||||||
|
ruby_whisper_params *rwp; \
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp); \
|
||||||
|
if (value == Qfalse || value == Qnil) { \
|
||||||
|
rwp->params.prop = false; \
|
||||||
|
} else { \
|
||||||
|
rwp->params.prop = true; \
|
||||||
|
} \
|
||||||
|
return value; \
|
||||||
|
|
||||||
|
#define BOOL_PARAMS_GETTER(self, prop) \
|
||||||
|
ruby_whisper_params *rwp; \
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp); \
|
||||||
|
if (rwp->params.prop) { \
|
||||||
|
return Qtrue; \
|
||||||
|
} else { \
|
||||||
|
return Qfalse; \
|
||||||
|
}
|
||||||
|
|
||||||
|
VALUE mWhisper;
|
||||||
|
VALUE cContext;
|
||||||
|
VALUE cParams;
|
||||||
|
|
||||||
|
static void ruby_whisper_free(ruby_whisper *rw) {
|
||||||
|
if (rw->context) {
|
||||||
|
whisper_free(rw->context);
|
||||||
|
rw->context = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static void ruby_whisper_params_free(ruby_whisper_params *rwp) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void rb_whisper_mark(ruby_whisper *rw) {
|
||||||
|
// call rb_gc_mark on any ruby references in rw
|
||||||
|
}
|
||||||
|
|
||||||
|
void rb_whisper_free(ruby_whisper *rw) {
|
||||||
|
ruby_whisper_free(rw);
|
||||||
|
free(rw);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rb_whisper_params_mark(ruby_whisper_params *rwp) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void rb_whisper_params_free(ruby_whisper_params *rwp) {
|
||||||
|
ruby_whisper_params_free(rwp);
|
||||||
|
free(rwp);
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE ruby_whisper_allocate(VALUE klass) {
|
||||||
|
ruby_whisper *rw;
|
||||||
|
rw = ALLOC(ruby_whisper);
|
||||||
|
rw->context = NULL;
|
||||||
|
return Data_Wrap_Struct(klass, rb_whisper_mark, rb_whisper_free, rw);
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE ruby_whisper_params_allocate(VALUE klass) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
rwp = ALLOC(ruby_whisper_params);
|
||||||
|
rwp->params = whisper_full_default_params(WHISPER_SAMPLING_GREEDY);
|
||||||
|
return Data_Wrap_Struct(klass, rb_whisper_params_mark, rb_whisper_params_free, rwp);
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE ruby_whisper_initialize(int argc, VALUE *argv, VALUE self) {
|
||||||
|
ruby_whisper *rw;
|
||||||
|
VALUE whisper_model_file_path;
|
||||||
|
|
||||||
|
// TODO: we can support init from buffer here too maybe another ruby object to expose
|
||||||
|
rb_scan_args(argc, argv, "01", &whisper_model_file_path);
|
||||||
|
Data_Get_Struct(self, ruby_whisper, rw);
|
||||||
|
|
||||||
|
if (!rb_respond_to(whisper_model_file_path, rb_intern("to_s"))) {
|
||||||
|
rb_raise(rb_eRuntimeError, "Expected file path to model to initialize Whisper::Context");
|
||||||
|
}
|
||||||
|
rw->context = whisper_init_from_file_with_params(StringValueCStr(whisper_model_file_path), whisper_context_default_params());
|
||||||
|
if (rw->context == nullptr) {
|
||||||
|
rb_raise(rb_eRuntimeError, "error: failed to initialize whisper context");
|
||||||
|
}
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* transcribe a single file
|
||||||
|
* can emit to a block results
|
||||||
|
*
|
||||||
|
**/
|
||||||
|
static VALUE ruby_whisper_transcribe(int argc, VALUE *argv, VALUE self) {
|
||||||
|
ruby_whisper *rw;
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
VALUE wave_file_path, blk, params;
|
||||||
|
|
||||||
|
rb_scan_args(argc, argv, "02&", &wave_file_path, ¶ms, &blk);
|
||||||
|
Data_Get_Struct(self, ruby_whisper, rw);
|
||||||
|
Data_Get_Struct(params, ruby_whisper_params, rwp);
|
||||||
|
|
||||||
|
if (!rb_respond_to(wave_file_path, rb_intern("to_s"))) {
|
||||||
|
rb_raise(rb_eRuntimeError, "Expected file path to wave file");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string fname_inp = StringValueCStr(wave_file_path);
|
||||||
|
|
||||||
|
std::vector<float> pcmf32; // mono-channel F32 PCM
|
||||||
|
std::vector<std::vector<float>> pcmf32s; // stereo-channel F32 PCM
|
||||||
|
|
||||||
|
// WAV input - this is directly from main.cpp example
|
||||||
|
{
|
||||||
|
drwav wav;
|
||||||
|
std::vector<uint8_t> wav_data; // used for pipe input from stdin
|
||||||
|
|
||||||
|
if (fname_inp == "-") {
|
||||||
|
{
|
||||||
|
uint8_t buf[1024];
|
||||||
|
while (true) {
|
||||||
|
const size_t n = fread(buf, 1, sizeof(buf), stdin);
|
||||||
|
if (n == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
wav_data.insert(wav_data.end(), buf, buf + n);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (drwav_init_memory(&wav, wav_data.data(), wav_data.size(), nullptr) == false) {
|
||||||
|
fprintf(stderr, "error: failed to open WAV file from stdin\n");
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: read %zu bytes from stdin\n", __func__, wav_data.size());
|
||||||
|
} else if (drwav_init_file(&wav, fname_inp.c_str(), nullptr) == false) {
|
||||||
|
fprintf(stderr, "error: failed to open '%s' as WAV file\n", fname_inp.c_str());
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wav.channels != 1 && wav.channels != 2) {
|
||||||
|
fprintf(stderr, "WAV file '%s' must be mono or stereo\n", fname_inp.c_str());
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rwp->diarize && wav.channels != 2 && rwp->params.print_timestamps == false) {
|
||||||
|
fprintf(stderr, "WAV file '%s' must be stereo for diarization and timestamps have to be enabled\n", fname_inp.c_str());
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wav.sampleRate != WHISPER_SAMPLE_RATE) {
|
||||||
|
fprintf(stderr, "WAV file '%s' must be %i kHz\n", fname_inp.c_str(), WHISPER_SAMPLE_RATE/1000);
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wav.bitsPerSample != 16) {
|
||||||
|
fprintf(stderr, "WAV file '%s' must be 16-bit\n", fname_inp.c_str());
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint64_t n = wav_data.empty() ? wav.totalPCMFrameCount : wav_data.size()/(wav.channels*wav.bitsPerSample/8);
|
||||||
|
|
||||||
|
std::vector<int16_t> pcm16;
|
||||||
|
pcm16.resize(n*wav.channels);
|
||||||
|
drwav_read_pcm_frames_s16(&wav, n, pcm16.data());
|
||||||
|
drwav_uninit(&wav);
|
||||||
|
|
||||||
|
// convert to mono, float
|
||||||
|
pcmf32.resize(n);
|
||||||
|
if (wav.channels == 1) {
|
||||||
|
for (uint64_t i = 0; i < n; i++) {
|
||||||
|
pcmf32[i] = float(pcm16[i])/32768.0f;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (uint64_t i = 0; i < n; i++) {
|
||||||
|
pcmf32[i] = float(pcm16[2*i] + pcm16[2*i + 1])/65536.0f;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rwp->diarize) {
|
||||||
|
// convert to stereo, float
|
||||||
|
pcmf32s.resize(2);
|
||||||
|
|
||||||
|
pcmf32s[0].resize(n);
|
||||||
|
pcmf32s[1].resize(n);
|
||||||
|
for (uint64_t i = 0; i < n; i++) {
|
||||||
|
pcmf32s[0][i] = float(pcm16[2*i])/32768.0f;
|
||||||
|
pcmf32s[1][i] = float(pcm16[2*i + 1])/32768.0f;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
{
|
||||||
|
static bool is_aborted = false; // NOTE: this should be atomic to avoid data race
|
||||||
|
|
||||||
|
rwp->params.encoder_begin_callback = [](struct whisper_context * /*ctx*/, struct whisper_state * /*state*/, void * user_data) {
|
||||||
|
bool is_aborted = *(bool*)user_data;
|
||||||
|
return !is_aborted;
|
||||||
|
};
|
||||||
|
rwp->params.encoder_begin_callback_user_data = &is_aborted;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (whisper_full_parallel(rw->context, rwp->params, pcmf32.data(), pcmf32.size(), 1) != 0) {
|
||||||
|
fprintf(stderr, "failed to process audio\n");
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
const int n_segments = whisper_full_n_segments(rw->context);
|
||||||
|
VALUE output = rb_str_new2("");
|
||||||
|
for (int i = 0; i < n_segments; ++i) {
|
||||||
|
const char * text = whisper_full_get_segment_text(rw->context, i);
|
||||||
|
output = rb_str_concat(output, rb_str_new2(text));
|
||||||
|
}
|
||||||
|
VALUE idCall = rb_intern("call");
|
||||||
|
if (blk != Qnil) {
|
||||||
|
rb_funcall(blk, idCall, 1, output);
|
||||||
|
}
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* params.language = "auto" | "en", etc...
|
||||||
|
*/
|
||||||
|
static VALUE ruby_whisper_params_set_language(VALUE self, VALUE value) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
if (value == Qfalse || value == Qnil) {
|
||||||
|
rwp->params.language = "auto";
|
||||||
|
} else {
|
||||||
|
rwp->params.language = StringValueCStr(value);
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_language(VALUE self) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
if (rwp->params.language) {
|
||||||
|
return rb_str_new2(rwp->params.language);
|
||||||
|
} else {
|
||||||
|
return rb_str_new2("auto");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_translate(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, translate, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_translate(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, translate)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_no_context(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, no_context, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_no_context(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, no_context)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_single_segment(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, single_segment, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_single_segment(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, single_segment)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_print_special(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, print_special, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_print_special(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, print_special)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_print_progress(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, print_progress, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_print_progress(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, print_progress)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_print_realtime(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, print_realtime, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_print_realtime(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, print_realtime)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_print_timestamps(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, print_timestamps, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_print_timestamps(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, print_timestamps)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_suppress_blank(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, suppress_blank, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_suppress_blank(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, suppress_blank)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_suppress_non_speech_tokens(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, suppress_non_speech_tokens, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_suppress_non_speech_tokens(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, suppress_non_speech_tokens)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_token_timestamps(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, token_timestamps)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_token_timestamps(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, token_timestamps, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_split_on_word(VALUE self) {
|
||||||
|
BOOL_PARAMS_GETTER(self, split_on_word)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_split_on_word(VALUE self, VALUE value) {
|
||||||
|
BOOL_PARAMS_SETTER(self, split_on_word, value)
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_diarize(VALUE self) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
if (rwp->diarize) {
|
||||||
|
return Qtrue;
|
||||||
|
} else {
|
||||||
|
return Qfalse;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_diarize(VALUE self, VALUE value) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
if (value == Qfalse || value == Qnil) {
|
||||||
|
rwp->diarize = false;
|
||||||
|
} else {
|
||||||
|
rwp->diarize = true;
|
||||||
|
} \
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE ruby_whisper_params_get_offset(VALUE self) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
return INT2NUM(rwp->params.offset_ms);
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_offset(VALUE self, VALUE value) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
rwp->params.offset_ms = NUM2INT(value);
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_get_duration(VALUE self) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
return INT2NUM(rwp->params.duration_ms);
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_duration(VALUE self, VALUE value) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
rwp->params.duration_ms = NUM2INT(value);
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE ruby_whisper_params_get_max_text_tokens(VALUE self) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
return INT2NUM(rwp->params.n_max_text_ctx);
|
||||||
|
}
|
||||||
|
static VALUE ruby_whisper_params_set_max_text_tokens(VALUE self, VALUE value) {
|
||||||
|
ruby_whisper_params *rwp;
|
||||||
|
Data_Get_Struct(self, ruby_whisper_params, rwp);
|
||||||
|
rwp->params.n_max_text_ctx = NUM2INT(value);
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Init_whisper() {
|
||||||
|
mWhisper = rb_define_module("Whisper");
|
||||||
|
cContext = rb_define_class_under(mWhisper, "Context", rb_cObject);
|
||||||
|
cParams = rb_define_class_under(mWhisper, "Params", rb_cObject);
|
||||||
|
|
||||||
|
rb_define_alloc_func(cContext, ruby_whisper_allocate);
|
||||||
|
rb_define_method(cContext, "initialize", ruby_whisper_initialize, -1);
|
||||||
|
|
||||||
|
rb_define_method(cContext, "transcribe", ruby_whisper_transcribe, -1);
|
||||||
|
|
||||||
|
rb_define_alloc_func(cParams, ruby_whisper_params_allocate);
|
||||||
|
|
||||||
|
rb_define_method(cParams, "language=", ruby_whisper_params_set_language, 1);
|
||||||
|
rb_define_method(cParams, "language", ruby_whisper_params_get_language, 0);
|
||||||
|
rb_define_method(cParams, "translate=", ruby_whisper_params_set_translate, 1);
|
||||||
|
rb_define_method(cParams, "translate", ruby_whisper_params_get_translate, 0);
|
||||||
|
rb_define_method(cParams, "no_context=", ruby_whisper_params_set_no_context, 1);
|
||||||
|
rb_define_method(cParams, "no_context", ruby_whisper_params_get_no_context, 0);
|
||||||
|
rb_define_method(cParams, "single_segment=", ruby_whisper_params_set_single_segment, 1);
|
||||||
|
rb_define_method(cParams, "single_segment", ruby_whisper_params_get_single_segment, 0);
|
||||||
|
rb_define_method(cParams, "print_special", ruby_whisper_params_get_print_special, 0);
|
||||||
|
rb_define_method(cParams, "print_special=", ruby_whisper_params_set_print_special, 1);
|
||||||
|
rb_define_method(cParams, "print_progress", ruby_whisper_params_get_print_progress, 0);
|
||||||
|
rb_define_method(cParams, "print_progress=", ruby_whisper_params_set_print_progress, 1);
|
||||||
|
rb_define_method(cParams, "print_realtime", ruby_whisper_params_get_print_realtime, 0);
|
||||||
|
rb_define_method(cParams, "print_realtime=", ruby_whisper_params_set_print_realtime, 1);
|
||||||
|
rb_define_method(cParams, "print_timestamps", ruby_whisper_params_get_print_timestamps, 0);
|
||||||
|
rb_define_method(cParams, "print_timestamps=", ruby_whisper_params_set_print_timestamps, 1);
|
||||||
|
rb_define_method(cParams, "suppress_blank", ruby_whisper_params_get_suppress_blank, 0);
|
||||||
|
rb_define_method(cParams, "suppress_blank=", ruby_whisper_params_set_suppress_blank, 1);
|
||||||
|
rb_define_method(cParams, "suppress_non_speech_tokens", ruby_whisper_params_get_suppress_non_speech_tokens, 0);
|
||||||
|
rb_define_method(cParams, "suppress_non_speech_tokens=", ruby_whisper_params_set_suppress_non_speech_tokens, 1);
|
||||||
|
rb_define_method(cParams, "token_timestamps", ruby_whisper_params_get_token_timestamps, 0);
|
||||||
|
rb_define_method(cParams, "token_timestamps=", ruby_whisper_params_set_token_timestamps, 1);
|
||||||
|
rb_define_method(cParams, "split_on_word", ruby_whisper_params_get_split_on_word, 0);
|
||||||
|
rb_define_method(cParams, "split_on_word=", ruby_whisper_params_set_split_on_word, 1);
|
||||||
|
rb_define_method(cParams, "diarize", ruby_whisper_params_get_diarize, 0);
|
||||||
|
rb_define_method(cParams, "diarize=", ruby_whisper_params_set_diarize, 1);
|
||||||
|
|
||||||
|
rb_define_method(cParams, "offset", ruby_whisper_params_get_offset, 0);
|
||||||
|
rb_define_method(cParams, "offset=", ruby_whisper_params_set_offset, 1);
|
||||||
|
rb_define_method(cParams, "duration", ruby_whisper_params_get_duration, 0);
|
||||||
|
rb_define_method(cParams, "duration=", ruby_whisper_params_set_duration, 1);
|
||||||
|
|
||||||
|
rb_define_method(cParams, "max_text_tokens", ruby_whisper_params_get_max_text_tokens, 0);
|
||||||
|
rb_define_method(cParams, "max_text_tokens=", ruby_whisper_params_set_max_text_tokens, 1);
|
||||||
|
}
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
@ -1,15 +1,8 @@
|
|||||||
#ifndef RUBY_WHISPER_H
|
#ifndef __RUBY_WHISPER_H
|
||||||
#define RUBY_WHISPER_H
|
#define __RUBY_WHISPER_H
|
||||||
|
|
||||||
#include "whisper.h"
|
#include "whisper.h"
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
VALUE *context;
|
|
||||||
VALUE user_data;
|
|
||||||
VALUE callback;
|
|
||||||
VALUE callbacks;
|
|
||||||
} ruby_whisper_callback_container;
|
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
struct whisper_context *context;
|
struct whisper_context *context;
|
||||||
} ruby_whisper;
|
} ruby_whisper;
|
||||||
@ -17,18 +10,6 @@ typedef struct {
|
|||||||
typedef struct {
|
typedef struct {
|
||||||
struct whisper_full_params params;
|
struct whisper_full_params params;
|
||||||
bool diarize;
|
bool diarize;
|
||||||
ruby_whisper_callback_container *new_segment_callback_container;
|
|
||||||
ruby_whisper_callback_container *progress_callback_container;
|
|
||||||
ruby_whisper_callback_container *abort_callback_container;
|
|
||||||
} ruby_whisper_params;
|
} ruby_whisper_params;
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
VALUE context;
|
|
||||||
int index;
|
|
||||||
} ruby_whisper_segment;
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
VALUE context;
|
|
||||||
} ruby_whisper_model;
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -1,613 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include <ruby/memory_view.h>
|
|
||||||
#include "ruby_whisper.h"
|
|
||||||
|
|
||||||
extern ID id_to_s;
|
|
||||||
extern ID id___method__;
|
|
||||||
extern ID id_to_enum;
|
|
||||||
extern ID id_length;
|
|
||||||
extern ID id_next;
|
|
||||||
extern ID id_new;
|
|
||||||
extern ID id_to_path;
|
|
||||||
extern ID id_URI;
|
|
||||||
extern ID id_pre_converted_models;
|
|
||||||
|
|
||||||
extern VALUE cContext;
|
|
||||||
extern VALUE eError;
|
|
||||||
extern VALUE cModel;
|
|
||||||
|
|
||||||
extern VALUE ruby_whisper_transcribe(int argc, VALUE *argv, VALUE self);
|
|
||||||
extern VALUE rb_whisper_model_initialize(VALUE context);
|
|
||||||
extern VALUE rb_whisper_segment_initialize(VALUE context, int index);
|
|
||||||
extern void register_callbacks(ruby_whisper_params *rwp, VALUE *context);
|
|
||||||
|
|
||||||
static void
|
|
||||||
ruby_whisper_free(ruby_whisper *rw)
|
|
||||||
{
|
|
||||||
if (rw->context) {
|
|
||||||
whisper_free(rw->context);
|
|
||||||
rw->context = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
rb_whisper_mark(ruby_whisper *rw)
|
|
||||||
{
|
|
||||||
// call rb_gc_mark on any ruby references in rw
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
rb_whisper_free(ruby_whisper *rw)
|
|
||||||
{
|
|
||||||
ruby_whisper_free(rw);
|
|
||||||
free(rw);
|
|
||||||
}
|
|
||||||
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_allocate(VALUE klass)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
rw = ALLOC(ruby_whisper);
|
|
||||||
rw->context = NULL;
|
|
||||||
return Data_Wrap_Struct(klass, rb_whisper_mark, rb_whisper_free, rw);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* new("base.en") -> Whisper::Context
|
|
||||||
* new("path/to/model.bin") -> Whisper::Context
|
|
||||||
* new(Whisper::Model::URI.new("https://example.net/uri/of/model.bin")) -> Whisper::Context
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_initialize(int argc, VALUE *argv, VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
VALUE whisper_model_file_path;
|
|
||||||
|
|
||||||
// TODO: we can support init from buffer here too maybe another ruby object to expose
|
|
||||||
rb_scan_args(argc, argv, "01", &whisper_model_file_path);
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
|
|
||||||
VALUE pre_converted_models = rb_funcall(cModel, id_pre_converted_models, 0);
|
|
||||||
VALUE pre_converted_model = rb_hash_aref(pre_converted_models, whisper_model_file_path);
|
|
||||||
if (!NIL_P(pre_converted_model)) {
|
|
||||||
whisper_model_file_path = pre_converted_model;
|
|
||||||
}
|
|
||||||
if (TYPE(whisper_model_file_path) == T_STRING) {
|
|
||||||
const char * whisper_model_file_path_str = StringValueCStr(whisper_model_file_path);
|
|
||||||
if (strncmp("http://", whisper_model_file_path_str, 7) == 0 || strncmp("https://", whisper_model_file_path_str, 8) == 0) {
|
|
||||||
VALUE uri_class = rb_const_get(cModel, id_URI);
|
|
||||||
whisper_model_file_path = rb_class_new_instance(1, &whisper_model_file_path, uri_class);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (rb_obj_is_kind_of(whisper_model_file_path, rb_path2class("URI::HTTP"))) {
|
|
||||||
VALUE uri_class = rb_const_get(cModel, id_URI);
|
|
||||||
whisper_model_file_path = rb_class_new_instance(1, &whisper_model_file_path, uri_class);
|
|
||||||
}
|
|
||||||
if (rb_respond_to(whisper_model_file_path, id_to_path)) {
|
|
||||||
whisper_model_file_path = rb_funcall(whisper_model_file_path, id_to_path, 0);
|
|
||||||
}
|
|
||||||
if (!rb_respond_to(whisper_model_file_path, id_to_s)) {
|
|
||||||
rb_raise(rb_eRuntimeError, "Expected file path to model to initialize Whisper::Context");
|
|
||||||
}
|
|
||||||
rw->context = whisper_init_from_file_with_params(StringValueCStr(whisper_model_file_path), whisper_context_default_params());
|
|
||||||
if (rw->context == NULL) {
|
|
||||||
rb_raise(rb_eRuntimeError, "error: failed to initialize whisper context");
|
|
||||||
}
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_vocab -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_vocab(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_vocab(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_audio_ctx -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_audio_ctx(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_ctx(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_audio_state -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_audio_state(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_state(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_audio_head -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_audio_head(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_head(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_audio_layer -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_audio_layer(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_layer(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_text_ctx -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_text_ctx(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_ctx(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_text_state -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_text_state(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_state(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_text_head -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_text_head(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_head(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_text_layer -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_text_layer(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_layer(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_n_mels -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_n_mels(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_mels(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_ftype -> Integer
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_ftype(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_ftype(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model_type -> String
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_model_type(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return rb_str_new2(whisper_model_type_readable(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Run the entire model: PCM -> log mel spectrogram -> encoder -> decoder -> text
|
|
||||||
* Not thread safe for same context
|
|
||||||
* Uses the specified decoding strategy to obtain the text.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full(params, samples, n_samples) -> nil
|
|
||||||
* full(params, samples) -> nil
|
|
||||||
*
|
|
||||||
* The second argument +samples+ must be an array of samples, respond to :length, or be a MemoryView of an array of float. It must be 32 bit float PCM audio data.
|
|
||||||
*/
|
|
||||||
VALUE ruby_whisper_full(int argc, VALUE *argv, VALUE self)
|
|
||||||
{
|
|
||||||
if (argc < 2 || argc > 3) {
|
|
||||||
rb_raise(rb_eArgError, "wrong number of arguments (given %d, expected 2..3)", argc);
|
|
||||||
}
|
|
||||||
|
|
||||||
ruby_whisper *rw;
|
|
||||||
ruby_whisper_params *rwp;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
VALUE params = argv[0];
|
|
||||||
Data_Get_Struct(params, ruby_whisper_params, rwp);
|
|
||||||
VALUE samples = argv[1];
|
|
||||||
int n_samples;
|
|
||||||
rb_memory_view_t view;
|
|
||||||
const bool memory_view_available_p = rb_memory_view_available_p(samples);
|
|
||||||
if (argc == 3) {
|
|
||||||
n_samples = NUM2INT(argv[2]);
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
if (RARRAY_LEN(samples) < n_samples) {
|
|
||||||
rb_raise(rb_eArgError, "samples length %ld is less than n_samples %d", RARRAY_LEN(samples), n_samples);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Should check when samples.respond_to?(:length)?
|
|
||||||
} else {
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
n_samples = RARRAY_LEN(samples);
|
|
||||||
} else if (memory_view_available_p) {
|
|
||||||
if (!rb_memory_view_get(samples, &view, RUBY_MEMORY_VIEW_SIMPLE)) {
|
|
||||||
view.obj = Qnil;
|
|
||||||
rb_raise(rb_eArgError, "unable to get a memory view");
|
|
||||||
}
|
|
||||||
n_samples = view.byte_size / view.item_size;
|
|
||||||
} else if (rb_respond_to(samples, id_length)) {
|
|
||||||
n_samples = NUM2INT(rb_funcall(samples, id_length, 0));
|
|
||||||
} else {
|
|
||||||
rb_raise(rb_eArgError, "samples must respond to :length or be a MemoryView of an array of flaot when n_samples is not given");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
float * c_samples = (float *)malloc(n_samples * sizeof(float));
|
|
||||||
if (memory_view_available_p) {
|
|
||||||
c_samples = (float *)view.data;
|
|
||||||
} else {
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
for (int i = 0; i < n_samples; i++) {
|
|
||||||
c_samples[i] = RFLOAT_VALUE(rb_ary_entry(samples, i));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// TODO: use rb_block_call
|
|
||||||
VALUE iter = rb_funcall(samples, id_to_enum, 1, rb_str_new2("each"));
|
|
||||||
for (int i = 0; i < n_samples; i++) {
|
|
||||||
// TODO: check if iter is exhausted and raise ArgumentError appropriately
|
|
||||||
VALUE sample = rb_funcall(iter, id_next, 0);
|
|
||||||
c_samples[i] = RFLOAT_VALUE(sample);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
register_callbacks(rwp, &self);
|
|
||||||
const int result = whisper_full(rw->context, rwp->params, c_samples, n_samples);
|
|
||||||
if (0 == result) {
|
|
||||||
return self;
|
|
||||||
} else {
|
|
||||||
rb_exc_raise(rb_funcall(eError, id_new, 1, result));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Split the input audio in chunks and process each chunk separately using whisper_full_with_state()
|
|
||||||
* Result is stored in the default state of the context
|
|
||||||
* Not thread safe if executed in parallel on the same context.
|
|
||||||
* It seems this approach can offer some speedup in some cases.
|
|
||||||
* However, the transcription accuracy can be worse at the beginning and end of each chunk.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_parallel(params, samples) -> nil
|
|
||||||
* full_parallel(params, samples, n_samples) -> nil
|
|
||||||
* full_parallel(params, samples, n_samples, n_processors) -> nil
|
|
||||||
* full_parallel(params, samples, nil, n_processors) -> nil
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_parallel(int argc, VALUE *argv,VALUE self)
|
|
||||||
{
|
|
||||||
if (argc < 2 || argc > 4) {
|
|
||||||
rb_raise(rb_eArgError, "wrong number of arguments (given %d, expected 2..3)", argc);
|
|
||||||
}
|
|
||||||
|
|
||||||
ruby_whisper *rw;
|
|
||||||
ruby_whisper_params *rwp;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
VALUE params = argv[0];
|
|
||||||
Data_Get_Struct(params, ruby_whisper_params, rwp);
|
|
||||||
VALUE samples = argv[1];
|
|
||||||
int n_samples;
|
|
||||||
int n_processors;
|
|
||||||
rb_memory_view_t view;
|
|
||||||
const bool memory_view_available_p = rb_memory_view_available_p(samples);
|
|
||||||
switch (argc) {
|
|
||||||
case 2:
|
|
||||||
n_processors = 1;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
n_processors = 1;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
n_processors = NUM2INT(argv[3]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (argc >= 3 && !NIL_P(argv[2])) {
|
|
||||||
n_samples = NUM2INT(argv[2]);
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
if (RARRAY_LEN(samples) < n_samples) {
|
|
||||||
rb_raise(rb_eArgError, "samples length %ld is less than n_samples %d", RARRAY_LEN(samples), n_samples);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Should check when samples.respond_to?(:length)?
|
|
||||||
} else if (memory_view_available_p) {
|
|
||||||
if (!rb_memory_view_get(samples, &view, RUBY_MEMORY_VIEW_SIMPLE)) {
|
|
||||||
view.obj = Qnil;
|
|
||||||
rb_raise(rb_eArgError, "unable to get a memory view");
|
|
||||||
}
|
|
||||||
n_samples = view.byte_size / view.item_size;
|
|
||||||
} else {
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
n_samples = RARRAY_LEN(samples);
|
|
||||||
} else if (rb_respond_to(samples, id_length)) {
|
|
||||||
n_samples = NUM2INT(rb_funcall(samples, id_length, 0));
|
|
||||||
} else {
|
|
||||||
rb_raise(rb_eArgError, "samples must respond to :length or be a MemoryView of an array of flaot when n_samples is not given");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
float * c_samples = (float *)malloc(n_samples * sizeof(float));
|
|
||||||
if (memory_view_available_p) {
|
|
||||||
c_samples = (float *)view.data;
|
|
||||||
} else {
|
|
||||||
if (TYPE(samples) == T_ARRAY) {
|
|
||||||
for (int i = 0; i < n_samples; i++) {
|
|
||||||
c_samples[i] = RFLOAT_VALUE(rb_ary_entry(samples, i));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// FIXME: use rb_block_call
|
|
||||||
VALUE iter = rb_funcall(samples, id_to_enum, 1, rb_str_new2("each"));
|
|
||||||
for (int i = 0; i < n_samples; i++) {
|
|
||||||
// TODO: check if iter is exhausted and raise ArgumentError
|
|
||||||
VALUE sample = rb_funcall(iter, id_next, 0);
|
|
||||||
c_samples[i] = RFLOAT_VALUE(sample);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
register_callbacks(rwp, &self);
|
|
||||||
const int result = whisper_full_parallel(rw->context, rwp->params, c_samples, n_samples, n_processors);
|
|
||||||
if (0 == result) {
|
|
||||||
return self;
|
|
||||||
} else {
|
|
||||||
rb_exc_raise(rb_funcall(eError, id_new, 1, result));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Number of segments.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_n_segments -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_n_segments(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_full_n_segments(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Language ID, which can be converted to string by Whisper.lang_str and Whisper.lang_str_full.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_lang_id -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_lang_id(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_full_lang_id(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int ruby_whisper_full_check_segment_index(const ruby_whisper * rw, const VALUE i_segment)
|
|
||||||
{
|
|
||||||
const int c_i_segment = NUM2INT(i_segment);
|
|
||||||
if (c_i_segment < 0 || c_i_segment >= whisper_full_n_segments(rw->context)) {
|
|
||||||
rb_raise(rb_eIndexError, "segment index %d out of range", c_i_segment);
|
|
||||||
}
|
|
||||||
return c_i_segment;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Start time of a segment indexed by +segment_index+ in centiseconds (10 times milliseconds).
|
|
||||||
*
|
|
||||||
* full_get_segment_t0(3) # => 1668 (16680 ms)
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_get_segment_t0(segment_index) -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment_t0(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
const int c_i_segment = ruby_whisper_full_check_segment_index(rw, i_segment);
|
|
||||||
const int64_t t0 = whisper_full_get_segment_t0(rw->context, c_i_segment);
|
|
||||||
return INT2NUM(t0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* End time of a segment indexed by +segment_index+ in centiseconds (10 times milliseconds).
|
|
||||||
*
|
|
||||||
* full_get_segment_t1(3) # => 1668 (16680 ms)
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_get_segment_t1(segment_index) -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment_t1(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
const int c_i_segment = ruby_whisper_full_check_segment_index(rw, i_segment);
|
|
||||||
const int64_t t1 = whisper_full_get_segment_t1(rw->context, c_i_segment);
|
|
||||||
return INT2NUM(t1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Whether the next segment indexed by +segment_index+ is predicated as a speaker turn.
|
|
||||||
*
|
|
||||||
* full_get_segment_speacker_turn_next(3) # => true
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_get_segment_speacker_turn_next(segment_index) -> bool
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment_speaker_turn_next(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
const int c_i_segment = ruby_whisper_full_check_segment_index(rw, i_segment);
|
|
||||||
const bool speaker_turn_next = whisper_full_get_segment_speaker_turn_next(rw->context, c_i_segment);
|
|
||||||
return speaker_turn_next ? Qtrue : Qfalse;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Text of a segment indexed by +segment_index+.
|
|
||||||
*
|
|
||||||
* full_get_segment_text(3) # => "ask not what your country can do for you, ..."
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* full_get_segment_text(segment_index) -> String
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment_text(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
const int c_i_segment = ruby_whisper_full_check_segment_index(rw, i_segment);
|
|
||||||
const char * text = whisper_full_get_segment_text(rw->context, c_i_segment);
|
|
||||||
return rb_str_new2(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* full_get_segment_no_speech_prob(segment_index) -> Float
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment_no_speech_prob(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
const int c_i_segment = ruby_whisper_full_check_segment_index(rw, i_segment);
|
|
||||||
const float no_speech_prob = whisper_full_get_segment_no_speech_prob(rw->context, c_i_segment);
|
|
||||||
return DBL2NUM(no_speech_prob);
|
|
||||||
}
|
|
||||||
|
|
||||||
// High level API
|
|
||||||
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_full_get_segment(VALUE self, VALUE i_segment)
|
|
||||||
{
|
|
||||||
return rb_whisper_segment_initialize(self, NUM2INT(i_segment));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Yields each Whisper::Segment:
|
|
||||||
*
|
|
||||||
* whisper.transcribe("path/to/audio.wav", params)
|
|
||||||
* whisper.each_segment do |segment|
|
|
||||||
* puts segment.text
|
|
||||||
* end
|
|
||||||
*
|
|
||||||
* Returns an Enumerator if no block given:
|
|
||||||
*
|
|
||||||
* whisper.transcribe("path/to/audio.wav", params)
|
|
||||||
* enum = whisper.each_segment
|
|
||||||
* enum.to_a # => [#<Whisper::Segment>, ...]
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* each_segment {|segment| ... }
|
|
||||||
* each_segment -> Enumerator
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_each_segment(VALUE self)
|
|
||||||
{
|
|
||||||
if (!rb_block_given_p()) {
|
|
||||||
const VALUE method_name = rb_funcall(self, id___method__, 0);
|
|
||||||
return rb_funcall(self, id_to_enum, 1, method_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
|
|
||||||
const int n_segments = whisper_full_n_segments(rw->context);
|
|
||||||
for (int i = 0; i < n_segments; ++i) {
|
|
||||||
rb_yield(rb_whisper_segment_initialize(self, i));
|
|
||||||
}
|
|
||||||
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* model -> Whisper::Model
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_get_model(VALUE self)
|
|
||||||
{
|
|
||||||
return rb_whisper_model_initialize(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
init_ruby_whisper_context(VALUE *mWhisper)
|
|
||||||
{
|
|
||||||
cContext = rb_define_class_under(*mWhisper, "Context", rb_cObject);
|
|
||||||
|
|
||||||
rb_define_alloc_func(cContext, ruby_whisper_allocate);
|
|
||||||
rb_define_method(cContext, "initialize", ruby_whisper_initialize, -1);
|
|
||||||
|
|
||||||
rb_define_method(cContext, "transcribe", ruby_whisper_transcribe, -1);
|
|
||||||
rb_define_method(cContext, "model_n_vocab", ruby_whisper_model_n_vocab, 0);
|
|
||||||
rb_define_method(cContext, "model_n_audio_ctx", ruby_whisper_model_n_audio_ctx, 0);
|
|
||||||
rb_define_method(cContext, "model_n_audio_state", ruby_whisper_model_n_audio_state, 0);
|
|
||||||
rb_define_method(cContext, "model_n_audio_head", ruby_whisper_model_n_audio_head, 0);
|
|
||||||
rb_define_method(cContext, "model_n_audio_layer", ruby_whisper_model_n_audio_layer, 0);
|
|
||||||
rb_define_method(cContext, "model_n_text_ctx", ruby_whisper_model_n_text_ctx, 0);
|
|
||||||
rb_define_method(cContext, "model_n_text_state", ruby_whisper_model_n_text_state, 0);
|
|
||||||
rb_define_method(cContext, "model_n_text_head", ruby_whisper_model_n_text_head, 0);
|
|
||||||
rb_define_method(cContext, "model_n_text_layer", ruby_whisper_model_n_text_layer, 0);
|
|
||||||
rb_define_method(cContext, "model_n_mels", ruby_whisper_model_n_mels, 0);
|
|
||||||
rb_define_method(cContext, "model_ftype", ruby_whisper_model_ftype, 0);
|
|
||||||
rb_define_method(cContext, "model_type", ruby_whisper_model_type, 0);
|
|
||||||
rb_define_method(cContext, "full_n_segments", ruby_whisper_full_n_segments, 0);
|
|
||||||
rb_define_method(cContext, "full_lang_id", ruby_whisper_full_lang_id, 0);
|
|
||||||
rb_define_method(cContext, "full_get_segment_t0", ruby_whisper_full_get_segment_t0, 1);
|
|
||||||
rb_define_method(cContext, "full_get_segment_t1", ruby_whisper_full_get_segment_t1, 1);
|
|
||||||
rb_define_method(cContext, "full_get_segment_speaker_turn_next", ruby_whisper_full_get_segment_speaker_turn_next, 1);
|
|
||||||
rb_define_method(cContext, "full_get_segment_text", ruby_whisper_full_get_segment_text, 1);
|
|
||||||
rb_define_method(cContext, "full_get_segment_no_speech_prob", ruby_whisper_full_get_segment_no_speech_prob, 1);
|
|
||||||
rb_define_method(cContext, "full", ruby_whisper_full, -1);
|
|
||||||
rb_define_method(cContext, "full_parallel", ruby_whisper_full_parallel, -1);
|
|
||||||
|
|
||||||
// High leve
|
|
||||||
rb_define_method(cContext, "full_get_segment", ruby_whisper_full_get_segment, 1);
|
|
||||||
rb_define_method(cContext, "each_segment", ruby_whisper_each_segment, 0);
|
|
||||||
|
|
||||||
rb_define_method(cContext, "model", ruby_whisper_get_model, 0);
|
|
||||||
}
|
|
@ -1,52 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
|
|
||||||
extern VALUE eError;
|
|
||||||
|
|
||||||
VALUE ruby_whisper_error_initialize(VALUE self, VALUE code)
|
|
||||||
{
|
|
||||||
const int c_code = NUM2INT(code);
|
|
||||||
const char *raw_message;
|
|
||||||
switch (c_code) {
|
|
||||||
case -2:
|
|
||||||
raw_message = "failed to compute log mel spectrogram";
|
|
||||||
break;
|
|
||||||
case -3:
|
|
||||||
raw_message = "failed to auto-detect language";
|
|
||||||
break;
|
|
||||||
case -4:
|
|
||||||
raw_message = "too many decoders requested";
|
|
||||||
break;
|
|
||||||
case -5:
|
|
||||||
raw_message = "audio_ctx is larger than the maximum allowed";
|
|
||||||
break;
|
|
||||||
case -6:
|
|
||||||
raw_message = "failed to encode";
|
|
||||||
break;
|
|
||||||
case -7:
|
|
||||||
raw_message = "whisper_kv_cache_init() failed for self-attention cache";
|
|
||||||
break;
|
|
||||||
case -8:
|
|
||||||
raw_message = "failed to decode";
|
|
||||||
break;
|
|
||||||
case -9:
|
|
||||||
raw_message = "failed to decode";
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
raw_message = "unknown error";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
const VALUE message = rb_str_new2(raw_message);
|
|
||||||
rb_call_super(1, &message);
|
|
||||||
rb_iv_set(self, "@code", code);
|
|
||||||
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
init_ruby_whisper_error(VALUE *mWhisper)
|
|
||||||
{
|
|
||||||
eError = rb_define_class_under(*mWhisper, "Error", rb_eStandardError);
|
|
||||||
|
|
||||||
rb_define_attr(eError, "code", true, false);
|
|
||||||
rb_define_method(eError, "initialize", ruby_whisper_error_initialize, 1);
|
|
||||||
}
|
|
@ -1,210 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include "ruby_whisper.h"
|
|
||||||
|
|
||||||
extern VALUE cModel;
|
|
||||||
|
|
||||||
static void rb_whisper_model_mark(ruby_whisper_model *rwm) {
|
|
||||||
rb_gc_mark(rwm->context);
|
|
||||||
}
|
|
||||||
|
|
||||||
static VALUE ruby_whisper_model_allocate(VALUE klass) {
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
rwm = ALLOC(ruby_whisper_model);
|
|
||||||
return Data_Wrap_Struct(klass, rb_whisper_model_mark, RUBY_DEFAULT_FREE, rwm);
|
|
||||||
}
|
|
||||||
|
|
||||||
VALUE rb_whisper_model_initialize(VALUE context) {
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
const VALUE model = ruby_whisper_model_allocate(cModel);
|
|
||||||
Data_Get_Struct(model, ruby_whisper_model, rwm);
|
|
||||||
rwm->context = context;
|
|
||||||
return model;
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_vocab -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_vocab(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_vocab(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_audio_ctx -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_audio_ctx(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_ctx(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_audio_state -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_audio_state(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_state(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_audio_head -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_audio_head(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_head(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_audio_layer -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_audio_layer(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_audio_layer(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_text_ctx -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_text_ctx(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_ctx(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_text_state -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_text_state(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_state(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_text_head -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_text_head(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_head(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_text_layer -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_text_layer(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_text_layer(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* n_mels -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_n_mels(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_n_mels(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* ftype -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_ftype(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return INT2NUM(whisper_model_ftype(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* type -> String
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_model_type(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_model *rwm;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_model, rwm);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rwm->context, ruby_whisper, rw);
|
|
||||||
return rb_str_new2(whisper_model_type_readable(rw->context));
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
init_ruby_whisper_model(VALUE *mWhisper)
|
|
||||||
{
|
|
||||||
cModel = rb_define_class_under(*mWhisper, "Model", rb_cObject);
|
|
||||||
|
|
||||||
rb_define_alloc_func(cModel, ruby_whisper_model_allocate);
|
|
||||||
rb_define_method(cModel, "n_vocab", ruby_whisper_model_n_vocab, 0);
|
|
||||||
rb_define_method(cModel, "n_audio_ctx", ruby_whisper_model_n_audio_ctx, 0);
|
|
||||||
rb_define_method(cModel, "n_audio_state", ruby_whisper_model_n_audio_state, 0);
|
|
||||||
rb_define_method(cModel, "n_audio_head", ruby_whisper_model_n_audio_head, 0);
|
|
||||||
rb_define_method(cModel, "n_audio_layer", ruby_whisper_model_n_audio_layer, 0);
|
|
||||||
rb_define_method(cModel, "n_text_ctx", ruby_whisper_model_n_text_ctx, 0);
|
|
||||||
rb_define_method(cModel, "n_text_state", ruby_whisper_model_n_text_state, 0);
|
|
||||||
rb_define_method(cModel, "n_text_head", ruby_whisper_model_n_text_head, 0);
|
|
||||||
rb_define_method(cModel, "n_text_layer", ruby_whisper_model_n_text_layer, 0);
|
|
||||||
rb_define_method(cModel, "n_mels", ruby_whisper_model_n_mels, 0);
|
|
||||||
rb_define_method(cModel, "ftype", ruby_whisper_model_ftype, 0);
|
|
||||||
rb_define_method(cModel, "type", ruby_whisper_model_type, 0);
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
@ -1,123 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include "ruby_whisper.h"
|
|
||||||
|
|
||||||
extern VALUE cSegment;
|
|
||||||
|
|
||||||
static void
|
|
||||||
rb_whisper_segment_mark(ruby_whisper_segment *rws)
|
|
||||||
{
|
|
||||||
rb_gc_mark(rws->context);
|
|
||||||
}
|
|
||||||
|
|
||||||
VALUE
|
|
||||||
ruby_whisper_segment_allocate(VALUE klass)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
rws = ALLOC(ruby_whisper_segment);
|
|
||||||
return Data_Wrap_Struct(klass, rb_whisper_segment_mark, RUBY_DEFAULT_FREE, rws);
|
|
||||||
}
|
|
||||||
|
|
||||||
VALUE
|
|
||||||
rb_whisper_segment_initialize(VALUE context, int index)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
const VALUE segment = ruby_whisper_segment_allocate(cSegment);
|
|
||||||
Data_Get_Struct(segment, ruby_whisper_segment, rws);
|
|
||||||
rws->context = context;
|
|
||||||
rws->index = index;
|
|
||||||
return segment;
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Start time in milliseconds.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* start_time -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_segment_get_start_time(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_segment, rws);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rws->context, ruby_whisper, rw);
|
|
||||||
const int64_t t0 = whisper_full_get_segment_t0(rw->context, rws->index);
|
|
||||||
// able to multiply 10 without overflow because to_timestamp() in whisper.cpp does it
|
|
||||||
return INT2NUM(t0 * 10);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* End time in milliseconds.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* end_time -> Integer
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_segment_get_end_time(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_segment, rws);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rws->context, ruby_whisper, rw);
|
|
||||||
const int64_t t1 = whisper_full_get_segment_t1(rw->context, rws->index);
|
|
||||||
// able to multiply 10 without overflow because to_timestamp() in whisper.cpp does it
|
|
||||||
return INT2NUM(t1 * 10);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Whether the next segment is predicted as a speaker turn.
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* speaker_turn_next? -> bool
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_segment_get_speaker_turn_next(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_segment, rws);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rws->context, ruby_whisper, rw);
|
|
||||||
return whisper_full_get_segment_speaker_turn_next(rw->context, rws->index) ? Qtrue : Qfalse;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* text -> String
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_segment_get_text(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_segment, rws);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rws->context, ruby_whisper, rw);
|
|
||||||
const char * text = whisper_full_get_segment_text(rw->context, rws->index);
|
|
||||||
return rb_str_new2(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* call-seq:
|
|
||||||
* no_speech_prob -> Float
|
|
||||||
*/
|
|
||||||
static VALUE
|
|
||||||
ruby_whisper_segment_get_no_speech_prob(VALUE self)
|
|
||||||
{
|
|
||||||
ruby_whisper_segment *rws;
|
|
||||||
Data_Get_Struct(self, ruby_whisper_segment, rws);
|
|
||||||
ruby_whisper *rw;
|
|
||||||
Data_Get_Struct(rws->context, ruby_whisper, rw);
|
|
||||||
return DBL2NUM(whisper_full_get_segment_no_speech_prob(rw->context, rws->index));
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
init_ruby_whisper_segment(VALUE *mWhisper, VALUE *cContext)
|
|
||||||
{
|
|
||||||
cSegment = rb_define_class_under(*mWhisper, "Segment", rb_cObject);
|
|
||||||
|
|
||||||
rb_define_alloc_func(cSegment, ruby_whisper_segment_allocate);
|
|
||||||
rb_define_method(cSegment, "start_time", ruby_whisper_segment_get_start_time, 0);
|
|
||||||
rb_define_method(cSegment, "end_time", ruby_whisper_segment_get_end_time, 0);
|
|
||||||
rb_define_method(cSegment, "speaker_next_turn?", ruby_whisper_segment_get_speaker_turn_next, 0);
|
|
||||||
rb_define_method(cSegment, "text", ruby_whisper_segment_get_text, 0);
|
|
||||||
rb_define_method(cSegment, "no_speech_prob", ruby_whisper_segment_get_no_speech_prob, 0);
|
|
||||||
}
|
|
@ -1,159 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include "ruby_whisper.h"
|
|
||||||
#define DR_WAV_IMPLEMENTATION
|
|
||||||
#include "dr_wav.h"
|
|
||||||
#include <string>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern ID id_to_s;
|
|
||||||
extern ID id_call;
|
|
||||||
|
|
||||||
extern void
|
|
||||||
register_callbacks(ruby_whisper_params * rwp, VALUE * self);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* transcribe a single file
|
|
||||||
* can emit to a block results
|
|
||||||
*
|
|
||||||
* params = Whisper::Params.new
|
|
||||||
* params.duration = 60_000
|
|
||||||
* whisper.transcribe "path/to/audio.wav", params do |text|
|
|
||||||
* puts text
|
|
||||||
* end
|
|
||||||
*
|
|
||||||
* call-seq:
|
|
||||||
* transcribe(path_to_audio, params) {|text| ...}
|
|
||||||
**/
|
|
||||||
VALUE
|
|
||||||
ruby_whisper_transcribe(int argc, VALUE *argv, VALUE self) {
|
|
||||||
ruby_whisper *rw;
|
|
||||||
ruby_whisper_params *rwp;
|
|
||||||
VALUE wave_file_path, blk, params;
|
|
||||||
|
|
||||||
rb_scan_args(argc, argv, "02&", &wave_file_path, ¶ms, &blk);
|
|
||||||
Data_Get_Struct(self, ruby_whisper, rw);
|
|
||||||
Data_Get_Struct(params, ruby_whisper_params, rwp);
|
|
||||||
|
|
||||||
if (!rb_respond_to(wave_file_path, id_to_s)) {
|
|
||||||
rb_raise(rb_eRuntimeError, "Expected file path to wave file");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string fname_inp = StringValueCStr(wave_file_path);
|
|
||||||
|
|
||||||
std::vector<float> pcmf32; // mono-channel F32 PCM
|
|
||||||
std::vector<std::vector<float>> pcmf32s; // stereo-channel F32 PCM
|
|
||||||
|
|
||||||
// WAV input - this is directly from main.cpp example
|
|
||||||
{
|
|
||||||
drwav wav;
|
|
||||||
std::vector<uint8_t> wav_data; // used for pipe input from stdin
|
|
||||||
|
|
||||||
if (fname_inp == "-") {
|
|
||||||
{
|
|
||||||
uint8_t buf[1024];
|
|
||||||
while (true) {
|
|
||||||
const size_t n = fread(buf, 1, sizeof(buf), stdin);
|
|
||||||
if (n == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
wav_data.insert(wav_data.end(), buf, buf + n);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (drwav_init_memory(&wav, wav_data.data(), wav_data.size(), nullptr) == false) {
|
|
||||||
fprintf(stderr, "error: failed to open WAV file from stdin\n");
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
fprintf(stderr, "%s: read %zu bytes from stdin\n", __func__, wav_data.size());
|
|
||||||
} else if (drwav_init_file(&wav, fname_inp.c_str(), nullptr) == false) {
|
|
||||||
fprintf(stderr, "error: failed to open '%s' as WAV file\n", fname_inp.c_str());
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wav.channels != 1 && wav.channels != 2) {
|
|
||||||
fprintf(stderr, "WAV file '%s' must be mono or stereo\n", fname_inp.c_str());
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rwp->diarize && wav.channels != 2 && rwp->params.print_timestamps == false) {
|
|
||||||
fprintf(stderr, "WAV file '%s' must be stereo for diarization and timestamps have to be enabled\n", fname_inp.c_str());
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wav.sampleRate != WHISPER_SAMPLE_RATE) {
|
|
||||||
fprintf(stderr, "WAV file '%s' must be %i kHz\n", fname_inp.c_str(), WHISPER_SAMPLE_RATE/1000);
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wav.bitsPerSample != 16) {
|
|
||||||
fprintf(stderr, "WAV file '%s' must be 16-bit\n", fname_inp.c_str());
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
const uint64_t n = wav_data.empty() ? wav.totalPCMFrameCount : wav_data.size()/(wav.channels*wav.bitsPerSample/8);
|
|
||||||
|
|
||||||
std::vector<int16_t> pcm16;
|
|
||||||
pcm16.resize(n*wav.channels);
|
|
||||||
drwav_read_pcm_frames_s16(&wav, n, pcm16.data());
|
|
||||||
drwav_uninit(&wav);
|
|
||||||
|
|
||||||
// convert to mono, float
|
|
||||||
pcmf32.resize(n);
|
|
||||||
if (wav.channels == 1) {
|
|
||||||
for (uint64_t i = 0; i < n; i++) {
|
|
||||||
pcmf32[i] = float(pcm16[i])/32768.0f;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (uint64_t i = 0; i < n; i++) {
|
|
||||||
pcmf32[i] = float((int32_t)pcm16[2*i] + pcm16[2*i + 1])/65536.0f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rwp->diarize) {
|
|
||||||
// convert to stereo, float
|
|
||||||
pcmf32s.resize(2);
|
|
||||||
|
|
||||||
pcmf32s[0].resize(n);
|
|
||||||
pcmf32s[1].resize(n);
|
|
||||||
for (uint64_t i = 0; i < n; i++) {
|
|
||||||
pcmf32s[0][i] = float(pcm16[2*i])/32768.0f;
|
|
||||||
pcmf32s[1][i] = float(pcm16[2*i + 1])/32768.0f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
static bool is_aborted = false; // NOTE: this should be atomic to avoid data race
|
|
||||||
|
|
||||||
rwp->params.encoder_begin_callback = [](struct whisper_context * /*ctx*/, struct whisper_state * /*state*/, void * user_data) {
|
|
||||||
bool is_aborted = *(bool*)user_data;
|
|
||||||
return !is_aborted;
|
|
||||||
};
|
|
||||||
rwp->params.encoder_begin_callback_user_data = &is_aborted;
|
|
||||||
}
|
|
||||||
|
|
||||||
register_callbacks(rwp, &self);
|
|
||||||
|
|
||||||
if (whisper_full_parallel(rw->context, rwp->params, pcmf32.data(), pcmf32.size(), 1) != 0) {
|
|
||||||
fprintf(stderr, "failed to process audio\n");
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
const int n_segments = whisper_full_n_segments(rw->context);
|
|
||||||
VALUE output = rb_str_new2("");
|
|
||||||
for (int i = 0; i < n_segments; ++i) {
|
|
||||||
const char * text = whisper_full_get_segment_text(rw->context, i);
|
|
||||||
output = rb_str_concat(output, rb_str_new2(text));
|
|
||||||
}
|
|
||||||
VALUE idCall = id_call;
|
|
||||||
if (blk != Qnil) {
|
|
||||||
rb_funcall(blk, idCall, 1, output);
|
|
||||||
}
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
@ -1,6 +0,0 @@
|
|||||||
require "yaml"
|
|
||||||
|
|
||||||
sources = `git ls-files -z ../..`.split("\x0")
|
|
||||||
paths = YAML.load_file("../../.github/workflows/bindings-ruby.yml")[true]["push"]["paths"]
|
|
||||||
paths.delete "bindings/ruby/**"
|
|
||||||
EXTSOURCES = (Dir.glob(paths, base: "../..").collect {|path| "../../#{path}"} << "../../LICENSE") & sources
|
|
@ -1,170 +0,0 @@
|
|||||||
require "uri"
|
|
||||||
require "net/http"
|
|
||||||
require "time"
|
|
||||||
require "pathname"
|
|
||||||
require "io/console/size"
|
|
||||||
|
|
||||||
module Whisper
|
|
||||||
class Model
|
|
||||||
class URI
|
|
||||||
def initialize(uri)
|
|
||||||
@uri = URI(uri)
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_path
|
|
||||||
cache
|
|
||||||
cache_path.to_path
|
|
||||||
end
|
|
||||||
|
|
||||||
def clear_cache
|
|
||||||
path = cache_path
|
|
||||||
path.delete if path.exist?
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def cache_path
|
|
||||||
base_cache_dir/@uri.host/@uri.path[1..]
|
|
||||||
end
|
|
||||||
|
|
||||||
def base_cache_dir
|
|
||||||
base = case RUBY_PLATFORM
|
|
||||||
when /mswin|mingw/
|
|
||||||
ENV.key?("LOCALAPPDATA") ? Pathname(ENV["LOCALAPPDATA"]) : Pathname(Dir.home)/"AppData/Local"
|
|
||||||
when /darwin/
|
|
||||||
Pathname(Dir.home)/"Library/Caches"
|
|
||||||
else
|
|
||||||
ENV.key?("XDG_CACHE_HOME") ? ENV["XDG_CACHE_HOME"] : Pathname(Dir.home)/".cache"
|
|
||||||
end
|
|
||||||
base/"whisper.cpp"
|
|
||||||
end
|
|
||||||
|
|
||||||
def cache
|
|
||||||
path = cache_path
|
|
||||||
headers = {}
|
|
||||||
headers["if-modified-since"] = path.mtime.httpdate if path.exist?
|
|
||||||
request @uri, headers
|
|
||||||
path
|
|
||||||
end
|
|
||||||
|
|
||||||
def request(uri, headers)
|
|
||||||
Net::HTTP.start uri.host, uri.port, use_ssl: uri.scheme == "https" do |http|
|
|
||||||
request = Net::HTTP::Get.new(uri, headers)
|
|
||||||
http.request request do |response|
|
|
||||||
case response
|
|
||||||
when Net::HTTPNotModified
|
|
||||||
# noop
|
|
||||||
when Net::HTTPOK
|
|
||||||
download response
|
|
||||||
when Net::HTTPRedirection
|
|
||||||
request URI(response["location"]), headers
|
|
||||||
else
|
|
||||||
return if headers.key?("if-modified-since") # Use cache file
|
|
||||||
|
|
||||||
raise "#{response.code} #{response.message}\n#{response.body}"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
rescue => err
|
|
||||||
if cache_path.exist?
|
|
||||||
warn err
|
|
||||||
# Use cache file
|
|
||||||
else
|
|
||||||
raise
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def download(response)
|
|
||||||
path = cache_path
|
|
||||||
path.dirname.mkpath unless path.dirname.exist?
|
|
||||||
downloading_path = Pathname("#{path}.downloading")
|
|
||||||
size = response.content_length
|
|
||||||
downloading_path.open "wb" do |file|
|
|
||||||
downloaded = 0
|
|
||||||
response.read_body do |chunk|
|
|
||||||
file << chunk
|
|
||||||
downloaded += chunk.bytesize
|
|
||||||
show_progress downloaded, size
|
|
||||||
end
|
|
||||||
$stderr.puts
|
|
||||||
end
|
|
||||||
downloading_path.rename path
|
|
||||||
end
|
|
||||||
|
|
||||||
def show_progress(current, size)
|
|
||||||
progress_rate_available = size && $stderr.tty?
|
|
||||||
|
|
||||||
unless @prev
|
|
||||||
@prev = Time.now
|
|
||||||
$stderr.puts "Downloading #{@uri} to #{cache_path}"
|
|
||||||
end
|
|
||||||
|
|
||||||
now = Time.now
|
|
||||||
|
|
||||||
if progress_rate_available
|
|
||||||
return if now - @prev < 1 && current < size
|
|
||||||
|
|
||||||
progress_width = 20
|
|
||||||
progress = current.to_f / size
|
|
||||||
arrow_length = progress * progress_width
|
|
||||||
arrow = "=" * (arrow_length - 1) + ">" + " " * (progress_width - arrow_length)
|
|
||||||
line = "[#{arrow}] (#{format_bytesize(current)} / #{format_bytesize(size)})"
|
|
||||||
padding = ' ' * ($stderr.winsize[1] - line.size)
|
|
||||||
$stderr.print "\r#{line}#{padding}"
|
|
||||||
else
|
|
||||||
return if now - @prev < 1
|
|
||||||
|
|
||||||
$stderr.print "."
|
|
||||||
end
|
|
||||||
@prev = now
|
|
||||||
end
|
|
||||||
|
|
||||||
def format_bytesize(bytesize)
|
|
||||||
return "0.0 B" if bytesize.zero?
|
|
||||||
|
|
||||||
units = %w[B KiB MiB GiB TiB]
|
|
||||||
exp = (Math.log(bytesize) / Math.log(1024)).to_i
|
|
||||||
format("%.1f %s", bytesize.to_f / 1024 ** exp, units[exp])
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@pre_converted_models = %w[
|
|
||||||
tiny
|
|
||||||
tiny.en
|
|
||||||
tiny-q5_1
|
|
||||||
tiny.en-q5_1
|
|
||||||
tiny-q8_0
|
|
||||||
base
|
|
||||||
base.en
|
|
||||||
base-q5_1
|
|
||||||
base.en-q5_1
|
|
||||||
base-q8_0
|
|
||||||
small
|
|
||||||
small.en
|
|
||||||
small.en-tdrz
|
|
||||||
small-q5_1
|
|
||||||
small.en-q5_1
|
|
||||||
small-q8_0
|
|
||||||
medium
|
|
||||||
medium.en
|
|
||||||
medium-q5_0
|
|
||||||
medium.en-q5_0
|
|
||||||
medium-q8_0
|
|
||||||
large-v1
|
|
||||||
large-v2
|
|
||||||
large-v2-q5_0
|
|
||||||
large-v2-q8_0
|
|
||||||
large-v3
|
|
||||||
large-v3-q5_0
|
|
||||||
large-v3-turbo
|
|
||||||
large-v3-turbo-q5_0
|
|
||||||
large-v3-turbo-q8_0
|
|
||||||
].each_with_object({}) {|name, models|
|
|
||||||
models[name] = URI.new("https://huggingface.co/ggerganov/whisper.cpp/resolve/main/ggml-#{name}.bin")
|
|
||||||
}
|
|
||||||
|
|
||||||
class << self
|
|
||||||
attr_reader :pre_converted_models
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,189 +0,0 @@
|
|||||||
module Whisper
|
|
||||||
interface _Samples
|
|
||||||
def length: () -> Integer
|
|
||||||
def each: { (Float) -> void } -> void
|
|
||||||
end
|
|
||||||
|
|
||||||
type log_callback = ^(Integer level, String message, Object user_data) -> void
|
|
||||||
type new_segment_callback = ^(Whisper::Context, void, Integer n_new, Object user_data) -> void
|
|
||||||
type progress_callback = ^(Whisper::Context, void, Integer progress, Object user_data) -> void
|
|
||||||
type abort_callback = ^(Whisper::Context, void, Object user_data) -> boolish
|
|
||||||
|
|
||||||
LOG_LEVEL_NONE: Integer
|
|
||||||
LOG_LEVEL_INFO: Integer
|
|
||||||
LOG_LEVEL_WARN: Integer
|
|
||||||
LOG_LEVEL_ERROR: Integer
|
|
||||||
LOG_LEVEL_DEBUG: Integer
|
|
||||||
LOG_LEVEL_CONT: Integer
|
|
||||||
|
|
||||||
def self.lang_max_id: () -> Integer
|
|
||||||
def self.lang_id: (string name) -> Integer
|
|
||||||
def self.lang_str: (Integer id) -> String
|
|
||||||
def self.lang_str_full: (Integer id) -> String
|
|
||||||
def self.log_set: (log_callback, Object? user_data) -> log_callback
|
|
||||||
|
|
||||||
class Context
|
|
||||||
def self.new: (string | _ToPath | ::URI::HTTP) -> instance
|
|
||||||
def transcribe: (string, Params) -> self
|
|
||||||
| (string, Params) { (String) -> void } -> self
|
|
||||||
def model_n_vocab: () -> Integer
|
|
||||||
def model_n_audio_ctx: () -> Integer
|
|
||||||
def model_n_audio_state: () -> Integer
|
|
||||||
def model_n_text_head: () -> Integer
|
|
||||||
def model_n_text_layer: () -> Integer
|
|
||||||
def model_n_mels: () -> Integer
|
|
||||||
def model_ftype: () -> Integer
|
|
||||||
def model_type: () -> String
|
|
||||||
def each_segment: { (Segment) -> void } -> void
|
|
||||||
| () -> Enumerator[Segment]
|
|
||||||
def model: () -> Model
|
|
||||||
def full_get_segment: (Integer nth) -> Segment
|
|
||||||
def full_n_segments: () -> Integer
|
|
||||||
def full_lang_id: () -> Integer
|
|
||||||
def full_get_segment_t0: (Integer) -> Integer
|
|
||||||
def full_get_segment_t1: (Integer) -> Integer
|
|
||||||
def full_get_segment_speaker_turn_next: (Integer) -> (true | false)
|
|
||||||
def full_get_segment_text: (Integer) -> String
|
|
||||||
def full_get_segment_no_speech_prob: (Integer) -> Float
|
|
||||||
def full: (Params, Array[Float] samples, ?Integer n_samples) -> self
|
|
||||||
| (Params, _Samples, ?Integer n_samples) -> self
|
|
||||||
def full_parallel: (Params, Array[Float], ?Integer n_samples) -> self
|
|
||||||
| (Params, _Samples, ?Integer n_samples) -> self
|
|
||||||
| (Params, _Samples, ?Integer? n_samples, Integer n_processors) -> self
|
|
||||||
end
|
|
||||||
|
|
||||||
class Params
|
|
||||||
def self.new: (
|
|
||||||
?language: string,
|
|
||||||
?translate: boolish,
|
|
||||||
?no_context: boolish,
|
|
||||||
?single_segment: boolish,
|
|
||||||
?print_special: boolish,
|
|
||||||
?print_progress: boolish,
|
|
||||||
?print_realtime: boolish,
|
|
||||||
?print_timestamps: boolish,
|
|
||||||
?suppress_blank: boolish,
|
|
||||||
?suppress_nst: boolish,
|
|
||||||
?token_timestamps: boolish,
|
|
||||||
?split_on_word: boolish,
|
|
||||||
?initial_prompt: string | nil,
|
|
||||||
?diarize: boolish,
|
|
||||||
?offset: Integer,
|
|
||||||
?duration: Integer,
|
|
||||||
?max_text_tokens: Integer,
|
|
||||||
?temperature: Float,
|
|
||||||
?max_initial_ts: Float,
|
|
||||||
?length_penalty: Float,
|
|
||||||
?temperature_inc: Float,
|
|
||||||
?entropy_thold: Float,
|
|
||||||
?logprob_thold: Float,
|
|
||||||
?no_speech_thold: Float,
|
|
||||||
?new_segment_callback: new_segment_callback,
|
|
||||||
?new_segment_callback_user_data: Object,
|
|
||||||
?progress_callback: progress_callback,
|
|
||||||
?progress_callback_user_data: Object,
|
|
||||||
?abort_callback: abort_callback,
|
|
||||||
?abort_callback_user_data: Object
|
|
||||||
) -> instance
|
|
||||||
def language=: (String) -> String # TODO: Enumerate lang names
|
|
||||||
def language: () -> String
|
|
||||||
def translate=: (boolish) -> boolish
|
|
||||||
def translate: () -> (true | false)
|
|
||||||
def no_context=: (boolish) -> boolish
|
|
||||||
def no_context: () -> (true | false)
|
|
||||||
def single_segment=: (boolish) -> boolish
|
|
||||||
def single_segment: () -> (true | false)
|
|
||||||
def print_special=: (boolish) -> boolish
|
|
||||||
def print_special: () -> (true | false)
|
|
||||||
def print_progress=: (boolish) -> boolish
|
|
||||||
def print_progress: () -> (true | false)
|
|
||||||
def print_realtime=: (boolish) -> boolish
|
|
||||||
def print_realtime: () -> (true | false)
|
|
||||||
def print_timestamps=: (boolish) -> boolish
|
|
||||||
def print_timestamps: () -> (true | false)
|
|
||||||
def suppress_blank=: (boolish) -> boolish
|
|
||||||
def suppress_blank: () -> (true | false)
|
|
||||||
def suppress_nst=: (boolish) -> boolish
|
|
||||||
def suppress_nst: () -> (true | false)
|
|
||||||
def token_timestamps=: (boolish) -> boolish
|
|
||||||
def token_timestamps: () -> (true | false)
|
|
||||||
def split_on_word=: (boolish) -> boolish
|
|
||||||
def split_on_word: () -> (true | false)
|
|
||||||
def initial_prompt=: (_ToS) -> _ToS
|
|
||||||
def initial_prompt: () -> (String | nil)
|
|
||||||
def diarize=: (boolish) -> boolish
|
|
||||||
def diarize: () -> (true | false)
|
|
||||||
def offset=: (Integer) -> Integer
|
|
||||||
def offset: () -> Integer
|
|
||||||
def duration=: (Integer) -> Integer
|
|
||||||
def duration: () -> Integer
|
|
||||||
def max_text_tokens=: (Integer) -> Integer
|
|
||||||
def max_text_tokens: () -> Integer
|
|
||||||
def temperature=: (Float) -> Float
|
|
||||||
def temperature: () -> Float
|
|
||||||
def max_initial_ts=: (Float) -> Float
|
|
||||||
def max_initial_ts: () -> Float
|
|
||||||
def length_penalty=: (Float) -> Float
|
|
||||||
def length_penalty: () -> Float
|
|
||||||
def temperature_inc=: (Float) -> Float
|
|
||||||
def temperature_inc: () -> Float
|
|
||||||
def entropy_thold=: (Float) -> Float
|
|
||||||
def entropy_thold: () -> Float
|
|
||||||
def logprob_thold=: (Float) -> Float
|
|
||||||
def logprob_thold: () -> Float
|
|
||||||
def no_speech_thold=: (Float) -> Float
|
|
||||||
def no_speech_thold: () -> Float
|
|
||||||
def new_segment_callback=: (new_segment_callback) -> new_segment_callback
|
|
||||||
def new_segment_callback: () -> (new_segment_callback | nil)
|
|
||||||
def new_segment_callback_user_data=: (Object) -> Object
|
|
||||||
def new_segment_callback_user_data: () -> Object
|
|
||||||
def progress_callback=: (progress_callback) -> progress_callback
|
|
||||||
def progress_callback: () -> (progress_callback | nil)
|
|
||||||
def progress_callback_user_data=: (Object) -> Object
|
|
||||||
def progress_callback_user_data: () -> Object
|
|
||||||
def abort_callback=: (abort_callback) -> abort_callback
|
|
||||||
def abort_callback: () -> (abort_callback | nil)
|
|
||||||
def abort_callback_user_data=: (Object) -> Object
|
|
||||||
def abort_callback_user_data: () -> Object
|
|
||||||
def on_new_segment: { (Segment) -> void } -> void
|
|
||||||
def on_progress: { (Integer progress) -> void } -> void
|
|
||||||
def abort_on: { (Object user_data) -> boolish } -> void
|
|
||||||
end
|
|
||||||
|
|
||||||
class Model
|
|
||||||
def self.pre_converted_models: () -> Hash[String, Model::URI]
|
|
||||||
def self.new: () -> instance
|
|
||||||
def n_vocab: () -> Integer
|
|
||||||
def n_audio_ctx: () -> Integer
|
|
||||||
def n_audio_state: () -> Integer
|
|
||||||
def n_audio_head: () -> Integer
|
|
||||||
def n_audio_layer: () -> Integer
|
|
||||||
def n_text_ctx: () -> Integer
|
|
||||||
def n_text_state: () -> Integer
|
|
||||||
def n_text_head: () -> Integer
|
|
||||||
def n_text_layer: () -> Integer
|
|
||||||
def n_mels: () -> Integer
|
|
||||||
def ftype: () -> Integer
|
|
||||||
def type: () -> String
|
|
||||||
|
|
||||||
class URI
|
|
||||||
def self.new: (string | ::URI::HTTP) -> self
|
|
||||||
def to_path: -> String
|
|
||||||
def clear_cache: -> void
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class Segment
|
|
||||||
def start_time: () -> Integer
|
|
||||||
def end_time: () -> Integer
|
|
||||||
def speaker_next_turn?: () -> (true | false)
|
|
||||||
def text: () -> String
|
|
||||||
def no_speech_prob: () -> Float
|
|
||||||
end
|
|
||||||
|
|
||||||
class Error < StandardError
|
|
||||||
attr_reader code: Integer
|
|
||||||
|
|
||||||
def self.new: (Integer code) -> instance
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,24 +0,0 @@
|
|||||||
require "test/unit"
|
|
||||||
require "whisper"
|
|
||||||
require_relative "jfk_reader/jfk_reader"
|
|
||||||
|
|
||||||
class TestBase < Test::Unit::TestCase
|
|
||||||
AUDIO = File.join(__dir__, "..", "..", "..", "samples", "jfk.wav")
|
|
||||||
|
|
||||||
class << self
|
|
||||||
attr_reader :whisper
|
|
||||||
|
|
||||||
def startup
|
|
||||||
@whisper = Whisper::Context.new("base.en")
|
|
||||||
params = Whisper::Params.new
|
|
||||||
params.print_timestamps = false
|
|
||||||
@whisper.transcribe(TestBase::AUDIO, params)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def whisper
|
|
||||||
self.class.whisper
|
|
||||||
end
|
|
||||||
end
|
|
5
bindings/ruby/tests/jfk_reader/.gitignore
vendored
5
bindings/ruby/tests/jfk_reader/.gitignore
vendored
@ -1,5 +0,0 @@
|
|||||||
Makefile
|
|
||||||
jfk_reader.o
|
|
||||||
jfk_reader.so
|
|
||||||
jfk_reader.bundle
|
|
||||||
jfk_reader.dll
|
|
@ -1,3 +0,0 @@
|
|||||||
require "mkmf"
|
|
||||||
|
|
||||||
create_makefile("jfk_reader")
|
|
@ -1,68 +0,0 @@
|
|||||||
#include <ruby.h>
|
|
||||||
#include <ruby/memory_view.h>
|
|
||||||
#include <ruby/encoding.h>
|
|
||||||
|
|
||||||
static VALUE
|
|
||||||
jfk_reader_initialize(VALUE self, VALUE audio_path)
|
|
||||||
{
|
|
||||||
rb_iv_set(self, "audio_path", audio_path);
|
|
||||||
return Qnil;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
jfk_reader_get_memory_view(const VALUE obj, rb_memory_view_t *view, int flags)
|
|
||||||
{
|
|
||||||
VALUE audio_path = rb_iv_get(obj, "audio_path");
|
|
||||||
const char *audio_path_str = StringValueCStr(audio_path);
|
|
||||||
const int n_samples = 176000;
|
|
||||||
float *data = (float *)malloc(n_samples * sizeof(float));
|
|
||||||
short *samples = (short *)malloc(n_samples * sizeof(short));
|
|
||||||
FILE *file = fopen(audio_path_str, "rb");
|
|
||||||
|
|
||||||
fseek(file, 78, SEEK_SET);
|
|
||||||
fread(samples, sizeof(short), n_samples, file);
|
|
||||||
fclose(file);
|
|
||||||
for (int i = 0; i < n_samples; i++) {
|
|
||||||
data[i] = samples[i]/32768.0;
|
|
||||||
}
|
|
||||||
|
|
||||||
view->obj = obj;
|
|
||||||
view->data = (void *)data;
|
|
||||||
view->byte_size = sizeof(float) * n_samples;
|
|
||||||
view->readonly = true;
|
|
||||||
view->format = "f";
|
|
||||||
view->item_size = sizeof(float);
|
|
||||||
view->item_desc.components = NULL;
|
|
||||||
view->item_desc.length = 0;
|
|
||||||
view->ndim = 1;
|
|
||||||
view->shape = NULL;
|
|
||||||
view->sub_offsets = NULL;
|
|
||||||
view->private_data = NULL;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
jfk_reader_release_memory_view(const VALUE obj, rb_memory_view_t *view)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
jfk_reader_memory_view_available_p(const VALUE obj)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const rb_memory_view_entry_t jfk_reader_view_entry = {
|
|
||||||
jfk_reader_get_memory_view,
|
|
||||||
jfk_reader_release_memory_view,
|
|
||||||
jfk_reader_memory_view_available_p
|
|
||||||
};
|
|
||||||
|
|
||||||
void Init_jfk_reader(void)
|
|
||||||
{
|
|
||||||
VALUE cJFKReader = rb_define_class("JFKReader", rb_cObject);
|
|
||||||
rb_memory_view_register(cJFKReader, &jfk_reader_view_entry);
|
|
||||||
rb_define_method(cJFKReader, "initialize", jfk_reader_initialize, 1);
|
|
||||||
}
|
|
@ -1,160 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
|
|
||||||
class TestCallback < TestBase
|
|
||||||
def setup
|
|
||||||
GC.start
|
|
||||||
@params = Whisper::Params.new
|
|
||||||
@whisper = Whisper::Context.new("base.en")
|
|
||||||
@audio = File.join(AUDIO)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_segment_callback
|
|
||||||
@params.new_segment_callback = ->(context, state, n_new, user_data) {
|
|
||||||
assert_kind_of Integer, n_new
|
|
||||||
assert n_new > 0
|
|
||||||
assert_same @whisper, context
|
|
||||||
|
|
||||||
n_segments = context.full_n_segments
|
|
||||||
n_new.times do |i|
|
|
||||||
i_segment = n_segments - 1 + i
|
|
||||||
start_time = context.full_get_segment_t0(i_segment) * 10
|
|
||||||
end_time = context.full_get_segment_t1(i_segment) * 10
|
|
||||||
text = context.full_get_segment_text(i_segment)
|
|
||||||
|
|
||||||
assert_kind_of Integer, start_time
|
|
||||||
assert start_time >= 0
|
|
||||||
assert_kind_of Integer, end_time
|
|
||||||
assert end_time > 0
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, text if i_segment == 0
|
|
||||||
end
|
|
||||||
}
|
|
||||||
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_segment_callback_closure
|
|
||||||
search_word = "what"
|
|
||||||
@params.new_segment_callback = ->(context, state, n_new, user_data) {
|
|
||||||
n_segments = context.full_n_segments
|
|
||||||
n_new.times do |i|
|
|
||||||
i_segment = n_segments - 1 + i
|
|
||||||
text = context.full_get_segment_text(i_segment)
|
|
||||||
if text.include?(search_word)
|
|
||||||
t0 = context.full_get_segment_t0(i_segment)
|
|
||||||
t1 = context.full_get_segment_t1(i_segment)
|
|
||||||
raise "search word '#{search_word}' found at between #{t0} and #{t1}"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_raise RuntimeError do
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_segment_callback_user_data
|
|
||||||
udata = Object.new
|
|
||||||
@params.new_segment_callback_user_data = udata
|
|
||||||
@params.new_segment_callback = ->(context, state, n_new, user_data) {
|
|
||||||
assert_same udata, user_data
|
|
||||||
}
|
|
||||||
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_segment_callback_user_data_gc
|
|
||||||
@params.new_segment_callback_user_data = "My user data"
|
|
||||||
@params.new_segment_callback = ->(context, state, n_new, user_data) {
|
|
||||||
assert_equal "My user data", user_data
|
|
||||||
}
|
|
||||||
GC.start
|
|
||||||
|
|
||||||
assert_same @whisper, @whisper.transcribe(@audio, @params)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_progress_callback
|
|
||||||
first = nil
|
|
||||||
last = nil
|
|
||||||
@params.progress_callback = ->(context, state, progress, user_data) {
|
|
||||||
assert_kind_of Integer, progress
|
|
||||||
assert 0 <= progress && progress <= 100
|
|
||||||
assert_same @whisper, context
|
|
||||||
first = progress if first.nil?
|
|
||||||
last = progress
|
|
||||||
}
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert_equal 0, first
|
|
||||||
assert_equal 100, last
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_progress_callback_user_data
|
|
||||||
udata = Object.new
|
|
||||||
@params.progress_callback_user_data = udata
|
|
||||||
@params.progress_callback = ->(context, state, n_new, user_data) {
|
|
||||||
assert_same udata, user_data
|
|
||||||
}
|
|
||||||
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_on_progress
|
|
||||||
first = nil
|
|
||||||
last = nil
|
|
||||||
@params.on_progress do |progress|
|
|
||||||
assert_kind_of Integer, progress
|
|
||||||
assert 0 <= progress && progress <= 100
|
|
||||||
first = progress if first.nil?
|
|
||||||
last = progress
|
|
||||||
end
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert_equal 0, first
|
|
||||||
assert_equal 100, last
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_abort_callback
|
|
||||||
i = 0
|
|
||||||
@params.abort_callback = ->(user_data) {
|
|
||||||
assert_nil user_data
|
|
||||||
i += 1
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert i > 0
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_abort_callback_abort
|
|
||||||
i = 0
|
|
||||||
@params.abort_callback = ->(user_data) {
|
|
||||||
i += 1
|
|
||||||
return i == 3
|
|
||||||
}
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert_equal 3, i
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_abort_callback_user_data
|
|
||||||
udata = Object.new
|
|
||||||
@params.abort_callback_user_data = udata
|
|
||||||
yielded = nil
|
|
||||||
@params.abort_callback = ->(user_data) {
|
|
||||||
yielded = user_data
|
|
||||||
}
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert_same udata, yielded
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_abort_on
|
|
||||||
do_abort = false
|
|
||||||
aborted_from_callback = false
|
|
||||||
@params.on_new_segment do |segment|
|
|
||||||
do_abort = true if segment.text.match? /ask/
|
|
||||||
end
|
|
||||||
i = 0
|
|
||||||
@params.abort_on do
|
|
||||||
i += 1
|
|
||||||
do_abort
|
|
||||||
end
|
|
||||||
@whisper.transcribe(@audio, @params)
|
|
||||||
assert i > 0
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,20 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
|
|
||||||
class TestError < TestBase
|
|
||||||
def test_error
|
|
||||||
error = Whisper::Error.new(-2)
|
|
||||||
assert_equal "failed to compute log mel spectrogram", error.message
|
|
||||||
assert_equal -2, error.code
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_unknown_error
|
|
||||||
error = Whisper::Error.new(-20)
|
|
||||||
assert_equal "unknown error", error.message
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_non_int_code
|
|
||||||
assert_raise TypeError do
|
|
||||||
error = Whisper::Error.new("non int")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,109 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
require "pathname"
|
|
||||||
|
|
||||||
class TestModel < TestBase
|
|
||||||
def test_model
|
|
||||||
whisper = Whisper::Context.new("base.en")
|
|
||||||
assert_instance_of Whisper::Model, whisper.model
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_attributes
|
|
||||||
whisper = Whisper::Context.new("base.en")
|
|
||||||
model = whisper.model
|
|
||||||
|
|
||||||
assert_equal 51864, model.n_vocab
|
|
||||||
assert_equal 1500, model.n_audio_ctx
|
|
||||||
assert_equal 512, model.n_audio_state
|
|
||||||
assert_equal 8, model.n_audio_head
|
|
||||||
assert_equal 6, model.n_audio_layer
|
|
||||||
assert_equal 448, model.n_text_ctx
|
|
||||||
assert_equal 512, model.n_text_state
|
|
||||||
assert_equal 8, model.n_text_head
|
|
||||||
assert_equal 6, model.n_text_layer
|
|
||||||
assert_equal 80, model.n_mels
|
|
||||||
assert_equal 1, model.ftype
|
|
||||||
assert_equal "base", model.type
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_gc
|
|
||||||
model = Whisper::Context.new("base.en").model
|
|
||||||
GC.start
|
|
||||||
|
|
||||||
assert_equal 51864, model.n_vocab
|
|
||||||
assert_equal 1500, model.n_audio_ctx
|
|
||||||
assert_equal 512, model.n_audio_state
|
|
||||||
assert_equal 8, model.n_audio_head
|
|
||||||
assert_equal 6, model.n_audio_layer
|
|
||||||
assert_equal 448, model.n_text_ctx
|
|
||||||
assert_equal 512, model.n_text_state
|
|
||||||
assert_equal 8, model.n_text_head
|
|
||||||
assert_equal 6, model.n_text_layer
|
|
||||||
assert_equal 80, model.n_mels
|
|
||||||
assert_equal 1, model.ftype
|
|
||||||
assert_equal "base", model.type
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_pathname
|
|
||||||
path = Pathname(Whisper::Model.pre_converted_models["base.en"].to_path)
|
|
||||||
whisper = Whisper::Context.new(path)
|
|
||||||
model = whisper.model
|
|
||||||
|
|
||||||
assert_equal 51864, model.n_vocab
|
|
||||||
assert_equal 1500, model.n_audio_ctx
|
|
||||||
assert_equal 512, model.n_audio_state
|
|
||||||
assert_equal 8, model.n_audio_head
|
|
||||||
assert_equal 6, model.n_audio_layer
|
|
||||||
assert_equal 448, model.n_text_ctx
|
|
||||||
assert_equal 512, model.n_text_state
|
|
||||||
assert_equal 8, model.n_text_head
|
|
||||||
assert_equal 6, model.n_text_layer
|
|
||||||
assert_equal 80, model.n_mels
|
|
||||||
assert_equal 1, model.ftype
|
|
||||||
assert_equal "base", model.type
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_auto_download
|
|
||||||
path = Whisper::Model.pre_converted_models["base.en"].to_path
|
|
||||||
|
|
||||||
assert_path_exist path
|
|
||||||
assert_equal 147964211, File.size(path)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_uri_string
|
|
||||||
path = "https://huggingface.co/ggerganov/whisper.cpp/resolve/main/ggml-base.en.bin"
|
|
||||||
whisper = Whisper::Context.new(path)
|
|
||||||
model = whisper.model
|
|
||||||
|
|
||||||
assert_equal 51864, model.n_vocab
|
|
||||||
assert_equal 1500, model.n_audio_ctx
|
|
||||||
assert_equal 512, model.n_audio_state
|
|
||||||
assert_equal 8, model.n_audio_head
|
|
||||||
assert_equal 6, model.n_audio_layer
|
|
||||||
assert_equal 448, model.n_text_ctx
|
|
||||||
assert_equal 512, model.n_text_state
|
|
||||||
assert_equal 8, model.n_text_head
|
|
||||||
assert_equal 6, model.n_text_layer
|
|
||||||
assert_equal 80, model.n_mels
|
|
||||||
assert_equal 1, model.ftype
|
|
||||||
assert_equal "base", model.type
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_uri
|
|
||||||
path = URI("https://huggingface.co/ggerganov/whisper.cpp/resolve/main/ggml-base.en.bin")
|
|
||||||
whisper = Whisper::Context.new(path)
|
|
||||||
model = whisper.model
|
|
||||||
|
|
||||||
assert_equal 51864, model.n_vocab
|
|
||||||
assert_equal 1500, model.n_audio_ctx
|
|
||||||
assert_equal 512, model.n_audio_state
|
|
||||||
assert_equal 8, model.n_audio_head
|
|
||||||
assert_equal 6, model.n_audio_layer
|
|
||||||
assert_equal 448, model.n_text_ctx
|
|
||||||
assert_equal 512, model.n_text_state
|
|
||||||
assert_equal 8, model.n_text_head
|
|
||||||
assert_equal 6, model.n_text_layer
|
|
||||||
assert_equal 80, model.n_mels
|
|
||||||
assert_equal 1, model.ftype
|
|
||||||
assert_equal "base", model.type
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,31 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
require 'tempfile'
|
|
||||||
require 'tmpdir'
|
|
||||||
require 'shellwords'
|
|
||||||
|
|
||||||
class TestPackage < TestBase
|
|
||||||
def test_build
|
|
||||||
Tempfile.create do |file|
|
|
||||||
assert system("gem", "build", "whispercpp.gemspec", "--output", file.to_path.shellescape, exception: true)
|
|
||||||
assert file.size > 0
|
|
||||||
assert_path_exist file.to_path
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
sub_test_case "Building binary on installation" do
|
|
||||||
def setup
|
|
||||||
system "rake", "build", exception: true
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_install
|
|
||||||
match_data = `rake -Tbuild`.match(/(whispercpp-(.+)\.gem)/)
|
|
||||||
filename = match_data[1]
|
|
||||||
version = match_data[2]
|
|
||||||
basename = "whisper.#{RbConfig::CONFIG["DLEXT"]}"
|
|
||||||
Dir.mktmpdir do |dir|
|
|
||||||
system "gem", "install", "--install-dir", dir.shellescape, "--no-document", "pkg/#{filename.shellescape}", exception: true
|
|
||||||
assert_path_exist File.join(dir, "gems/whispercpp-#{version}/lib", basename)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,246 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
|
|
||||||
class TestParams < TestBase
|
|
||||||
PARAM_NAMES = [
|
|
||||||
:language,
|
|
||||||
:translate,
|
|
||||||
:no_context,
|
|
||||||
:single_segment,
|
|
||||||
:print_special,
|
|
||||||
:print_progress,
|
|
||||||
:print_realtime,
|
|
||||||
:print_timestamps,
|
|
||||||
:suppress_blank,
|
|
||||||
:suppress_nst,
|
|
||||||
:token_timestamps,
|
|
||||||
:split_on_word,
|
|
||||||
:initial_prompt,
|
|
||||||
:diarize,
|
|
||||||
:offset,
|
|
||||||
:duration,
|
|
||||||
:max_text_tokens,
|
|
||||||
:temperature,
|
|
||||||
:max_initial_ts,
|
|
||||||
:length_penalty,
|
|
||||||
:temperature_inc,
|
|
||||||
:entropy_thold,
|
|
||||||
:logprob_thold,
|
|
||||||
:no_speech_thold,
|
|
||||||
:new_segment_callback,
|
|
||||||
:new_segment_callback_user_data,
|
|
||||||
:progress_callback,
|
|
||||||
:progress_callback_user_data,
|
|
||||||
:abort_callback,
|
|
||||||
:abort_callback_user_data,
|
|
||||||
]
|
|
||||||
|
|
||||||
def setup
|
|
||||||
@params = Whisper::Params.new
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_language
|
|
||||||
@params.language = "en"
|
|
||||||
assert_equal @params.language, "en"
|
|
||||||
@params.language = "auto"
|
|
||||||
assert_equal @params.language, "auto"
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_offset
|
|
||||||
@params.offset = 10_000
|
|
||||||
assert_equal @params.offset, 10_000
|
|
||||||
@params.offset = 0
|
|
||||||
assert_equal @params.offset, 0
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_duration
|
|
||||||
@params.duration = 60_000
|
|
||||||
assert_equal @params.duration, 60_000
|
|
||||||
@params.duration = 0
|
|
||||||
assert_equal @params.duration, 0
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_max_text_tokens
|
|
||||||
@params.max_text_tokens = 300
|
|
||||||
assert_equal @params.max_text_tokens, 300
|
|
||||||
@params.max_text_tokens = 0
|
|
||||||
assert_equal @params.max_text_tokens, 0
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_translate
|
|
||||||
@params.translate = true
|
|
||||||
assert @params.translate
|
|
||||||
@params.translate = false
|
|
||||||
assert !@params.translate
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_no_context
|
|
||||||
@params.no_context = true
|
|
||||||
assert @params.no_context
|
|
||||||
@params.no_context = false
|
|
||||||
assert !@params.no_context
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_single_segment
|
|
||||||
@params.single_segment = true
|
|
||||||
assert @params.single_segment
|
|
||||||
@params.single_segment = false
|
|
||||||
assert !@params.single_segment
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_print_special
|
|
||||||
@params.print_special = true
|
|
||||||
assert @params.print_special
|
|
||||||
@params.print_special = false
|
|
||||||
assert !@params.print_special
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_print_progress
|
|
||||||
@params.print_progress = true
|
|
||||||
assert @params.print_progress
|
|
||||||
@params.print_progress = false
|
|
||||||
assert !@params.print_progress
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_print_realtime
|
|
||||||
@params.print_realtime = true
|
|
||||||
assert @params.print_realtime
|
|
||||||
@params.print_realtime = false
|
|
||||||
assert !@params.print_realtime
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_print_timestamps
|
|
||||||
@params.print_timestamps = true
|
|
||||||
assert @params.print_timestamps
|
|
||||||
@params.print_timestamps = false
|
|
||||||
assert !@params.print_timestamps
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_suppress_blank
|
|
||||||
@params.suppress_blank = true
|
|
||||||
assert @params.suppress_blank
|
|
||||||
@params.suppress_blank = false
|
|
||||||
assert !@params.suppress_blank
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_suppress_nst
|
|
||||||
@params.suppress_nst = true
|
|
||||||
assert @params.suppress_nst
|
|
||||||
@params.suppress_nst = false
|
|
||||||
assert !@params.suppress_nst
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_token_timestamps
|
|
||||||
@params.token_timestamps = true
|
|
||||||
assert @params.token_timestamps
|
|
||||||
@params.token_timestamps = false
|
|
||||||
assert !@params.token_timestamps
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_split_on_word
|
|
||||||
@params.split_on_word = true
|
|
||||||
assert @params.split_on_word
|
|
||||||
@params.split_on_word = false
|
|
||||||
assert !@params.split_on_word
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_initial_prompt
|
|
||||||
assert_nil @params.initial_prompt
|
|
||||||
@params.initial_prompt = "You are a polite person."
|
|
||||||
assert_equal "You are a polite person.", @params.initial_prompt
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_temperature
|
|
||||||
assert_equal 0.0, @params.temperature
|
|
||||||
@params.temperature = 0.5
|
|
||||||
assert_equal 0.5, @params.temperature
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_max_initial_ts
|
|
||||||
assert_equal 1.0, @params.max_initial_ts
|
|
||||||
@params.max_initial_ts = 600.0
|
|
||||||
assert_equal 600.0, @params.max_initial_ts
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_length_penalty
|
|
||||||
assert_equal -1.0, @params.length_penalty
|
|
||||||
@params.length_penalty = 0.5
|
|
||||||
assert_equal 0.5, @params.length_penalty
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_temperature_inc
|
|
||||||
assert_in_delta 0.2, @params.temperature_inc
|
|
||||||
@params.temperature_inc = 0.5
|
|
||||||
assert_in_delta 0.5, @params.temperature_inc
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_entropy_thold
|
|
||||||
assert_in_delta 2.4, @params.entropy_thold
|
|
||||||
@params.entropy_thold = 3.0
|
|
||||||
assert_in_delta 3.0, @params.entropy_thold
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_logprob_thold
|
|
||||||
assert_in_delta -1.0, @params.logprob_thold
|
|
||||||
@params.logprob_thold = -0.5
|
|
||||||
assert_in_delta -0.5, @params.logprob_thold
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_no_speech_thold
|
|
||||||
assert_in_delta 0.6, @params.no_speech_thold
|
|
||||||
@params.no_speech_thold = 0.2
|
|
||||||
assert_in_delta 0.2, @params.no_speech_thold
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_with_kw_args
|
|
||||||
params = Whisper::Params.new(language: "es")
|
|
||||||
assert_equal "es", params.language
|
|
||||||
assert_equal 1.0, params.max_initial_ts
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_with_kw_args_non_existent
|
|
||||||
assert_raise ArgumentError do
|
|
||||||
Whisper::Params.new(non_existent: "value")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_with_kw_args_wrong_type
|
|
||||||
assert_raise TypeError do
|
|
||||||
Whisper::Params.new(language: 3)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
data(PARAM_NAMES.collect {|param| [param, param]}.to_h)
|
|
||||||
def test_new_with_kw_args_default_values(param)
|
|
||||||
default_value = @params.send(param)
|
|
||||||
value = case [param, default_value]
|
|
||||||
in [*, true | false]
|
|
||||||
!default_value
|
|
||||||
in [*, Integer | Float]
|
|
||||||
default_value + 1
|
|
||||||
in [:language, *]
|
|
||||||
"es"
|
|
||||||
in [:initial_prompt, *]
|
|
||||||
"Initial prompt"
|
|
||||||
in [/_callback\Z/, *]
|
|
||||||
proc {}
|
|
||||||
in [/_user_data\Z/, *]
|
|
||||||
Object.new
|
|
||||||
end
|
|
||||||
params = Whisper::Params.new(param => value)
|
|
||||||
if Float === value
|
|
||||||
assert_in_delta value, params.send(param)
|
|
||||||
else
|
|
||||||
assert_equal value, params.send(param)
|
|
||||||
end
|
|
||||||
|
|
||||||
PARAM_NAMES.reject {|name| name == param}.each do |name|
|
|
||||||
expected = @params.send(name)
|
|
||||||
actual = params.send(name)
|
|
||||||
if Float === expected
|
|
||||||
assert_in_delta expected, actual
|
|
||||||
else
|
|
||||||
assert_equal expected, actual
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,74 +0,0 @@
|
|||||||
require_relative "helper"
|
|
||||||
|
|
||||||
class TestSegment < TestBase
|
|
||||||
def test_iteration
|
|
||||||
whisper.each_segment do |segment|
|
|
||||||
assert_instance_of Whisper::Segment, segment
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_enumerator
|
|
||||||
enum = whisper.each_segment
|
|
||||||
assert_instance_of Enumerator, enum
|
|
||||||
enum.to_a.each_with_index do |segment, index|
|
|
||||||
assert_instance_of Whisper::Segment, segment
|
|
||||||
assert_kind_of Integer, index
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_start_time
|
|
||||||
i = 0
|
|
||||||
whisper.each_segment do |segment|
|
|
||||||
assert_equal 0, segment.start_time if i == 0
|
|
||||||
i += 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_end_time
|
|
||||||
i = 0
|
|
||||||
whisper.each_segment do |segment|
|
|
||||||
assert_equal whisper.full_get_segment_t1(i) * 10, segment.end_time
|
|
||||||
i += 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_no_speech_prob
|
|
||||||
no_speech_prob = nil
|
|
||||||
whisper.each_segment do |segment|
|
|
||||||
no_speech_prob = segment.no_speech_prob
|
|
||||||
end
|
|
||||||
assert no_speech_prob > 0.0
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_on_new_segment
|
|
||||||
params = Whisper::Params.new
|
|
||||||
seg = nil
|
|
||||||
index = 0
|
|
||||||
params.on_new_segment do |segment|
|
|
||||||
assert_instance_of Whisper::Segment, segment
|
|
||||||
if index == 0
|
|
||||||
seg = segment
|
|
||||||
assert_equal 0, segment.start_time
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, segment.text
|
|
||||||
end
|
|
||||||
index += 1
|
|
||||||
end
|
|
||||||
whisper.transcribe(AUDIO, params)
|
|
||||||
assert_equal 0, seg.start_time
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, seg.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_on_new_segment_twice
|
|
||||||
params = Whisper::Params.new
|
|
||||||
seg = nil
|
|
||||||
params.on_new_segment do |segment|
|
|
||||||
seg = segment
|
|
||||||
return
|
|
||||||
end
|
|
||||||
params.on_new_segment do |segment|
|
|
||||||
assert_same seg, segment
|
|
||||||
return
|
|
||||||
end
|
|
||||||
whisper.transcribe(AUDIO, params)
|
|
||||||
end
|
|
||||||
end
|
|
@ -1,223 +1,131 @@
|
|||||||
require_relative "helper"
|
TOPDIR = File.expand_path(File.join(File.dirname(__FILE__), '..'))
|
||||||
require "stringio"
|
EXTDIR = File.join(TOPDIR, 'ext')
|
||||||
require "etc"
|
#$LIBDIR = File.join(TOPDIR, 'lib')
|
||||||
|
#$:.unshift(LIBDIR)
|
||||||
|
$:.unshift(EXTDIR)
|
||||||
|
|
||||||
# Exists to detect memory-related bug
|
require 'whisper'
|
||||||
Whisper.log_set ->(level, buffer, user_data) {}, nil
|
require 'test/unit'
|
||||||
|
|
||||||
class TestWhisper < TestBase
|
class TestWhisper < Test::Unit::TestCase
|
||||||
def setup
|
def setup
|
||||||
@params = Whisper::Params.new
|
@params = Whisper::Params.new
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_language
|
||||||
|
@params.language = "en"
|
||||||
|
assert_equal @params.language, "en"
|
||||||
|
@params.language = "auto"
|
||||||
|
assert_equal @params.language, "auto"
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_offset
|
||||||
|
@params.offset = 10_000
|
||||||
|
assert_equal @params.offset, 10_000
|
||||||
|
@params.offset = 0
|
||||||
|
assert_equal @params.offset, 0
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_duration
|
||||||
|
@params.duration = 60_000
|
||||||
|
assert_equal @params.duration, 60_000
|
||||||
|
@params.duration = 0
|
||||||
|
assert_equal @params.duration, 0
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_max_text_tokens
|
||||||
|
@params.max_text_tokens = 300
|
||||||
|
assert_equal @params.max_text_tokens, 300
|
||||||
|
@params.max_text_tokens = 0
|
||||||
|
assert_equal @params.max_text_tokens, 0
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_translate
|
||||||
|
@params.translate = true
|
||||||
|
assert @params.translate
|
||||||
|
@params.translate = false
|
||||||
|
assert !@params.translate
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_no_context
|
||||||
|
@params.no_context = true
|
||||||
|
assert @params.no_context
|
||||||
|
@params.no_context = false
|
||||||
|
assert !@params.no_context
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_single_segment
|
||||||
|
@params.single_segment = true
|
||||||
|
assert @params.single_segment
|
||||||
|
@params.single_segment = false
|
||||||
|
assert !@params.single_segment
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_print_special
|
||||||
|
@params.print_special = true
|
||||||
|
assert @params.print_special
|
||||||
|
@params.print_special = false
|
||||||
|
assert !@params.print_special
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_print_progress
|
||||||
|
@params.print_progress = true
|
||||||
|
assert @params.print_progress
|
||||||
|
@params.print_progress = false
|
||||||
|
assert !@params.print_progress
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_print_realtime
|
||||||
|
@params.print_realtime = true
|
||||||
|
assert @params.print_realtime
|
||||||
|
@params.print_realtime = false
|
||||||
|
assert !@params.print_realtime
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_print_timestamps
|
||||||
|
@params.print_timestamps = true
|
||||||
|
assert @params.print_timestamps
|
||||||
|
@params.print_timestamps = false
|
||||||
|
assert !@params.print_timestamps
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_suppress_blank
|
||||||
|
@params.suppress_blank = true
|
||||||
|
assert @params.suppress_blank
|
||||||
|
@params.suppress_blank = false
|
||||||
|
assert !@params.suppress_blank
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_suppress_non_speech_tokens
|
||||||
|
@params.suppress_non_speech_tokens = true
|
||||||
|
assert @params.suppress_non_speech_tokens
|
||||||
|
@params.suppress_non_speech_tokens = false
|
||||||
|
assert !@params.suppress_non_speech_tokens
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_token_timestamps
|
||||||
|
@params.token_timestamps = true
|
||||||
|
assert @params.token_timestamps
|
||||||
|
@params.token_timestamps = false
|
||||||
|
assert !@params.token_timestamps
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_split_on_word
|
||||||
|
@params.split_on_word = true
|
||||||
|
assert @params.split_on_word
|
||||||
|
@params.split_on_word = false
|
||||||
|
assert !@params.split_on_word
|
||||||
|
end
|
||||||
|
|
||||||
def test_whisper
|
def test_whisper
|
||||||
@whisper = Whisper::Context.new("base.en")
|
@whisper = Whisper::Context.new(File.join(TOPDIR, '..', '..', 'models', 'ggml-base.en.bin'))
|
||||||
params = Whisper::Params.new
|
params = Whisper::Params.new
|
||||||
params.print_timestamps = false
|
params.print_timestamps = false
|
||||||
|
|
||||||
@whisper.transcribe(AUDIO, params) {|text|
|
jfk = File.join(TOPDIR, '..', '..', 'samples', 'jfk.wav')
|
||||||
|
@whisper.transcribe(jfk, params) {|text|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, text
|
assert_match /ask not what your country can do for you, ask what you can do for your country/, text
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
sub_test_case "After transcription" do
|
|
||||||
def test_full_n_segments
|
|
||||||
assert_equal 1, whisper.full_n_segments
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_lang_id
|
|
||||||
assert_equal 0, whisper.full_lang_id
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment
|
|
||||||
segment = whisper.full_get_segment(0)
|
|
||||||
assert_equal 0, segment.start_time
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, segment.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment_t0
|
|
||||||
assert_equal 0, whisper.full_get_segment_t0(0)
|
|
||||||
assert_raise IndexError do
|
|
||||||
whisper.full_get_segment_t0(whisper.full_n_segments)
|
|
||||||
end
|
|
||||||
assert_raise IndexError do
|
|
||||||
whisper.full_get_segment_t0(-1)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment_t1
|
|
||||||
t1 = whisper.full_get_segment_t1(0)
|
|
||||||
assert_kind_of Integer, t1
|
|
||||||
assert t1 > 0
|
|
||||||
assert_raise IndexError do
|
|
||||||
whisper.full_get_segment_t1(whisper.full_n_segments)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment_speaker_turn_next
|
|
||||||
assert_false whisper.full_get_segment_speaker_turn_next(0)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment_text
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, whisper.full_get_segment_text(0)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_get_segment_no_speech_prob
|
|
||||||
prob = whisper.full_get_segment_no_speech_prob(0)
|
|
||||||
assert prob > 0.0
|
|
||||||
assert prob < 1.0
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lang_max_id
|
|
||||||
assert_kind_of Integer, Whisper.lang_max_id
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lang_id
|
|
||||||
assert_equal 0, Whisper.lang_id("en")
|
|
||||||
assert_raise ArgumentError do
|
|
||||||
Whisper.lang_id("non existing language")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lang_str
|
|
||||||
assert_equal "en", Whisper.lang_str(0)
|
|
||||||
assert_raise IndexError do
|
|
||||||
Whisper.lang_str(Whisper.lang_max_id + 1)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lang_str_full
|
|
||||||
assert_equal "english", Whisper.lang_str_full(0)
|
|
||||||
assert_raise IndexError do
|
|
||||||
Whisper.lang_str_full(Whisper.lang_max_id + 1)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_log_set
|
|
||||||
user_data = Object.new
|
|
||||||
logs = []
|
|
||||||
log_callback = ->(level, buffer, udata) {
|
|
||||||
logs << [level, buffer, udata]
|
|
||||||
}
|
|
||||||
Whisper.log_set log_callback, user_data
|
|
||||||
Whisper::Context.new("base.en")
|
|
||||||
|
|
||||||
assert logs.length > 30
|
|
||||||
logs.each do |log|
|
|
||||||
assert_include [Whisper::LOG_LEVEL_DEBUG, Whisper::LOG_LEVEL_INFO, Whisper::LOG_LEVEL_WARN], log[0]
|
|
||||||
assert_same user_data, log[2]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_log_suppress
|
|
||||||
stderr = $stderr
|
|
||||||
Whisper.log_set ->(level, buffer, user_data) {
|
|
||||||
# do nothing
|
|
||||||
}, nil
|
|
||||||
dev = StringIO.new("")
|
|
||||||
$stderr = dev
|
|
||||||
Whisper::Context.new("base.en")
|
|
||||||
assert_empty dev.string
|
|
||||||
ensure
|
|
||||||
$stderr = stderr
|
|
||||||
end
|
|
||||||
|
|
||||||
sub_test_case "full" do
|
|
||||||
def setup
|
|
||||||
super
|
|
||||||
@whisper = Whisper::Context.new("base.en")
|
|
||||||
@samples = File.read(AUDIO, nil, 78).unpack("s<*").collect {|i| i.to_f / 2**15}
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full
|
|
||||||
@whisper.full(@params, @samples, @samples.length)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, @whisper.each_segment.first.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_without_length
|
|
||||||
@whisper.full(@params, @samples)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, @whisper.each_segment.first.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_enumerator
|
|
||||||
samples = @samples.each
|
|
||||||
@whisper.full(@params, samples, @samples.length)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, @whisper.each_segment.first.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_enumerator_without_length
|
|
||||||
samples = @samples.each
|
|
||||||
assert_raise ArgumentError do
|
|
||||||
@whisper.full(@params, samples)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_enumerator_with_too_large_length
|
|
||||||
samples = @samples.each.take(10).to_enum
|
|
||||||
assert_raise StopIteration do
|
|
||||||
@whisper.full(@params, samples, 11)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_with_memory_view
|
|
||||||
samples = JFKReader.new(AUDIO)
|
|
||||||
@whisper.full(@params, samples)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
assert_match /ask not what your country can do for you, ask what you can do for your country/, @whisper.each_segment.first.text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_parallel
|
|
||||||
@whisper.full_parallel(@params, @samples, @samples.length, Etc.nprocessors)
|
|
||||||
|
|
||||||
assert_equal Etc.nprocessors, @whisper.full_n_segments
|
|
||||||
text = @whisper.each_segment.collect(&:text).join
|
|
||||||
assert_match /ask what you can do/i, text
|
|
||||||
assert_match /for your country/i, text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_parallel_with_memory_view
|
|
||||||
samples = JFKReader.new(AUDIO)
|
|
||||||
@whisper.full_parallel(@params, samples, nil, Etc.nprocessors)
|
|
||||||
|
|
||||||
assert_equal Etc.nprocessors, @whisper.full_n_segments
|
|
||||||
text = @whisper.each_segment.collect(&:text).join
|
|
||||||
assert_match /ask what you can do/i, text
|
|
||||||
assert_match /for your country/i, text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_parallel_without_length_and_n_processors
|
|
||||||
@whisper.full_parallel(@params, @samples)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
text = @whisper.each_segment.collect(&:text).join
|
|
||||||
assert_match /ask what you can do/i, text
|
|
||||||
assert_match /for your country/i, text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_parallel_without_length
|
|
||||||
@whisper.full_parallel(@params, @samples, nil, Etc.nprocessors)
|
|
||||||
|
|
||||||
assert_equal Etc.nprocessors, @whisper.full_n_segments
|
|
||||||
text = @whisper.each_segment.collect(&:text).join
|
|
||||||
assert_match /ask what you can do/i, text
|
|
||||||
assert_match /for your country/i, text
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_full_parallel_without_n_processors
|
|
||||||
@whisper.full_parallel(@params, @samples, @samples.length)
|
|
||||||
|
|
||||||
assert_equal 1, @whisper.full_n_segments
|
|
||||||
text = @whisper.each_segment.collect(&:text).join
|
|
||||||
assert_match /ask what you can do/i, text
|
|
||||||
assert_match /for your country/i, text
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
@ -1,36 +1,28 @@
|
|||||||
require_relative "extsources"
|
|
||||||
|
|
||||||
Gem::Specification.new do |s|
|
Gem::Specification.new do |s|
|
||||||
s.name = "whispercpp"
|
s.name = "whispercpp"
|
||||||
s.authors = ["Georgi Gerganov", "Todd A. Fisher"]
|
s.authors = ["Georgi Gerganov", "Todd A. Fisher"]
|
||||||
s.version = '1.3.1'
|
s.version = '1.3.0'
|
||||||
s.date = '2024-12-19'
|
s.date = '2024-05-14'
|
||||||
s.description = %q{High-performance inference of OpenAI's Whisper automatic speech recognition (ASR) model via Ruby}
|
s.description = %q{High-performance inference of OpenAI's Whisper automatic speech recognition (ASR) model via Ruby}
|
||||||
s.email = 'todd.fisher@gmail.com'
|
s.email = 'todd.fisher@gmail.com'
|
||||||
s.extra_rdoc_files = ['LICENSE', 'README.md']
|
s.extra_rdoc_files = ['LICENSE', 'README.md']
|
||||||
|
|
||||||
|
s.files = ["LICENSE", "README.md", "Rakefile", "ext/extconf.rb", "ext/ggml.c", "ext/ruby_whisper.cpp", "ext/whisper.cpp", "ext/dr_wav.h", "ext/ggml.h", "ext/ruby_whisper.h", "ext/whisper.h"]
|
||||||
|
|
||||||
s.files = `git ls-files . -z`.split("\x0") +
|
#### Load-time details
|
||||||
EXTSOURCES.collect {|file|
|
s.require_paths = ['lib','ext']
|
||||||
basename = File.basename(file)
|
|
||||||
if s.extra_rdoc_files.include?(basename)
|
|
||||||
basename
|
|
||||||
else
|
|
||||||
file.sub("../..", "ext")
|
|
||||||
end
|
|
||||||
}
|
|
||||||
|
|
||||||
s.summary = %q{Ruby whisper.cpp bindings}
|
s.summary = %q{Ruby whisper.cpp bindings}
|
||||||
s.test_files = s.files.select {|file| file.start_with? "tests/"}
|
s.test_files = ["tests/test_whisper.rb"]
|
||||||
|
|
||||||
s.extensions << 'ext/extconf.rb'
|
s.extensions << 'ext/extconf.rb'
|
||||||
s.required_ruby_version = '>= 3.1.0'
|
|
||||||
|
|
||||||
#### Documentation and testing.
|
#### Documentation and testing.
|
||||||
s.homepage = 'https://github.com/ggerganov/whisper.cpp'
|
s.homepage = 'https://github.com/ggerganov/whisper.cpp'
|
||||||
s.rdoc_options = ['--main', 'README.md']
|
s.rdoc_options = ['--main', '../../README.md']
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
s.platform = Gem::Platform::RUBY
|
s.platform = Gem::Platform::RUBY
|
||||||
|
|
||||||
s.licenses = ['MIT']
|
s.licenses = ['MIT']
|
||||||
end
|
end
|
||||||
|
@ -1,28 +0,0 @@
|
|||||||
name: Close inactive issues
|
|
||||||
on:
|
|
||||||
schedule:
|
|
||||||
- cron: "42 0 * * *"
|
|
||||||
|
|
||||||
# Fine-grant permission
|
|
||||||
# https://docs.github.com/en/actions/security-for-github-actions/security-guides/automatic-token-authentication#modifying-the-permissions-for-the-github_token
|
|
||||||
permissions:
|
|
||||||
issues: write
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
close-issues:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
issues: write
|
|
||||||
pull-requests: write
|
|
||||||
steps:
|
|
||||||
- uses: actions/stale@v5
|
|
||||||
with:
|
|
||||||
exempt-issue-labels: "refactor,help wanted,good first issue,research,bug,roadmap"
|
|
||||||
days-before-issue-stale: 30
|
|
||||||
days-before-issue-close: 14
|
|
||||||
stale-issue-label: "stale"
|
|
||||||
close-issue-message: "This issue was closed because it has been inactive for 14 days since being marked as stale."
|
|
||||||
days-before-pr-stale: -1
|
|
||||||
days-before-pr-close: -1
|
|
||||||
operations-per-run: 10000
|
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
|
@ -13,4 +13,5 @@ set_target_properties(${TARGET}
|
|||||||
PROPERTIES
|
PROPERTIES
|
||||||
EXPORT_COMPILE_COMMANDS ON
|
EXPORT_COMPILE_COMMANDS ON
|
||||||
RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/bin"
|
RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/bin"
|
||||||
|
INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/lib"
|
||||||
)
|
)
|
||||||
|
@ -36,7 +36,7 @@ include(FindPackageHandleStandardArgs)
|
|||||||
|
|
||||||
# The default components were taken from a survey over other FindFFMPEG.cmake files
|
# The default components were taken from a survey over other FindFFMPEG.cmake files
|
||||||
if (NOT FFmpeg_FIND_COMPONENTS)
|
if (NOT FFmpeg_FIND_COMPONENTS)
|
||||||
set(FFmpeg_FIND_COMPONENTS AVFORMAT AVCODEC AVUTIL SWRESAMPLE)
|
set(FFmpeg_FIND_COMPONENTS AVFORMAT AVCODEC AVUTIL SWRESAMPLE)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
#
|
#
|
||||||
@ -84,7 +84,7 @@ macro(find_component _component _pkgconfig _library _header)
|
|||||||
|
|
||||||
# CMake's default is to search first for shared libraries and then for static libraries.
|
# CMake's default is to search first for shared libraries and then for static libraries.
|
||||||
# Todo later: add option to prefer static libs over dynamic:
|
# Todo later: add option to prefer static libs over dynamic:
|
||||||
find_library(${_component}_LIBRARIES NAMES ${_library} lib${_library}.a
|
find_library(${_component}_LIBRARIES NAMES ${_library} lib${_library}.a
|
||||||
HINTS
|
HINTS
|
||||||
${PC_${_component}_LIBDIR}
|
${PC_${_component}_LIBDIR}
|
||||||
${PC_${_component}_LIBRARY_DIRS}
|
${PC_${_component}_LIBRARY_DIRS}
|
||||||
|
@ -42,8 +42,6 @@ endif()
|
|||||||
if(MSVC)
|
if(MSVC)
|
||||||
set(BUILD_COMPILER "${CMAKE_C_COMPILER_ID} ${CMAKE_C_COMPILER_VERSION}")
|
set(BUILD_COMPILER "${CMAKE_C_COMPILER_ID} ${CMAKE_C_COMPILER_VERSION}")
|
||||||
set(BUILD_TARGET ${CMAKE_VS_PLATFORM_NAME})
|
set(BUILD_TARGET ${CMAKE_VS_PLATFORM_NAME})
|
||||||
add_compile_options("$<$<COMPILE_LANGUAGE:C>:/utf-8>")
|
|
||||||
add_compile_options("$<$<COMPILE_LANGUAGE:CXX>:/utf-8>")
|
|
||||||
else()
|
else()
|
||||||
execute_process(
|
execute_process(
|
||||||
COMMAND sh -c "$@ --version | head -1" _ ${CMAKE_C_COMPILER}
|
COMMAND sh -c "$@ --version | head -1" _ ${CMAKE_C_COMPILER}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
set(WHISPER_VERSION @WHISPER_INSTALL_VERSION@)
|
set(LLAMA_VERSION @LLAMA_INSTALL_VERSION@)
|
||||||
set(WHISPER_BUILD_COMMIT @WHISPER_BUILD_COMMIT@)
|
set(LLAMA_BUILD_COMMIT @LLAMA_BUILD_COMMIT@)
|
||||||
set(WHISPER_BUILD_NUMBER @WHISPER_BUILD_NUMBER@)
|
set(LLAMA_BUILD_NUMBER @LLAMA_BUILD_NUMBER@)
|
||||||
set(WHISPER_SHARED_LIB @BUILD_SHARED_LIBS@)
|
set(LLAMA_SHARED_LIB @BUILD_SHARED_LIBS@)
|
||||||
|
|
||||||
set(GGML_BLAS @GGML_BLAS@)
|
set(GGML_BLAS @GGML_BLAS@)
|
||||||
set(GGML_CUDA @GGML_CUDA@)
|
set(GGML_CUDA @GGML_CUDA@)
|
||||||
@ -11,9 +11,9 @@ set(GGML_ACCELERATE @GGML_ACCELERATE@)
|
|||||||
|
|
||||||
@PACKAGE_INIT@
|
@PACKAGE_INIT@
|
||||||
|
|
||||||
set_and_check(WHISPER_INCLUDE_DIR "@PACKAGE_WHISPER_INCLUDE_INSTALL_DIR@")
|
set_and_check(LLAMA_INCLUDE_DIR "@PACKAGE_LLAMA_INCLUDE_INSTALL_DIR@")
|
||||||
set_and_check(WHISPER_LIB_DIR "@PACKAGE_WHISPER_LIB_INSTALL_DIR@")
|
set_and_check(LLAMA_LIB_DIR "@PACKAGE_LLAMA_LIB_INSTALL_DIR@")
|
||||||
set_and_check(WHISPER_BIN_DIR "@PACKAGE_WHISPER_BIN_INSTALL_DIR@")
|
set_and_check(LLAMA_BIN_DIR "@PACKAGE_LLAMA_BIN_INSTALL_DIR@")
|
||||||
|
|
||||||
# Ensure transient dependencies satisfied
|
# Ensure transient dependencies satisfied
|
||||||
|
|
||||||
@ -43,23 +43,23 @@ if (GGML_HIPBLAS)
|
|||||||
find_package(rocblas REQUIRED)
|
find_package(rocblas REQUIRED)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
find_library(whisper_LIBRARY whisper
|
find_library(llama_LIBRARY llama
|
||||||
REQUIRED
|
REQUIRED
|
||||||
HINTS ${WHISPER_LIB_DIR})
|
HINTS ${LLAMA_LIB_DIR})
|
||||||
|
|
||||||
set(_whisper_link_deps "Threads::Threads" "@WHISPER_EXTRA_LIBS@")
|
set(_llama_link_deps "Threads::Threads" "@LLAMA_EXTRA_LIBS@")
|
||||||
set(_whisper_transient_defines "@WHISPER_TRANSIENT_DEFINES@")
|
set(_llama_transient_defines "@LLAMA_TRANSIENT_DEFINES@")
|
||||||
|
|
||||||
add_library(whisper UNKNOWN IMPORTED)
|
add_library(llama UNKNOWN IMPORTED)
|
||||||
|
|
||||||
set_target_properties(whisper
|
set_target_properties(llama
|
||||||
PROPERTIES
|
PROPERTIES
|
||||||
INTERFACE_INCLUDE_DIRECTORIES "${WHISPER_INCLUDE_DIR}"
|
INTERFACE_INCLUDE_DIRECTORIES "${LLAMA_INCLUDE_DIR}"
|
||||||
INTERFACE_LINK_LIBRARIES "${_whisper_link_deps}"
|
INTERFACE_LINK_LIBRARIES "${_llama_link_deps}"
|
||||||
INTERFACE_COMPILE_DEFINITIONS "${_whisper_transient_defines}"
|
INTERFACE_COMPILE_DEFINITIONS "${_llama_transient_defines}"
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES "CXX"
|
IMPORTED_LINK_INTERFACE_LANGUAGES "CXX"
|
||||||
IMPORTED_LOCATION "${whisper_LIBRARY}"
|
IMPORTED_LOCATION "${llama_LIBRARY}"
|
||||||
INTERFACE_COMPILE_FEATURES cxx_std_11
|
INTERFACE_COMPILE_FEATURES cxx_std_11
|
||||||
POSITION_INDEPENDENT_CODE ON )
|
POSITION_INDEPENDENT_CODE ON )
|
||||||
|
|
||||||
check_required_components(whisper)
|
check_required_components(Llama)
|
||||||
|
@ -6,5 +6,5 @@ includedir=${prefix}/include
|
|||||||
Name: whisper
|
Name: whisper
|
||||||
Description: Port of OpenAI's Whisper model in C/C++
|
Description: Port of OpenAI's Whisper model in C/C++
|
||||||
Version: @PROJECT_VERSION@
|
Version: @PROJECT_VERSION@
|
||||||
Libs: -L${libdir} -lggml -lggml-base -lwhisper
|
Libs: -L${libdir} -lwhisper
|
||||||
Cflags: -I${includedir}
|
Cflags: -I${includedir}
|
||||||
|
@ -40,7 +40,7 @@ if (WHISPER_FFMPEG)
|
|||||||
message(STATUS "Found ffmpeg libs: ${FFMPEG_LIBRARIES}")
|
message(STATUS "Found ffmpeg libs: ${FFMPEG_LIBRARIES}")
|
||||||
message(STATUS "Found ffmpeg headers in: ${FFMPEG_INCLUDE_DIRS}")
|
message(STATUS "Found ffmpeg headers in: ${FFMPEG_INCLUDE_DIRS}")
|
||||||
message(STATUS "ffmpeg definitions: ${FFMPEG_DEFINITIONS}")
|
message(STATUS "ffmpeg definitions: ${FFMPEG_DEFINITIONS}")
|
||||||
message(STATUS "Found avformat ${AVFORMAT_VERSION}")
|
message(STATUS "Found avformat ${AVFORMAT_VERSION}")
|
||||||
|
|
||||||
include_directories(${FFMPEG_INCLUDE_DIRS})
|
include_directories(${FFMPEG_INCLUDE_DIRS})
|
||||||
add_compile_definitions(WHISPER_FFMPEG)
|
add_compile_definitions(WHISPER_FFMPEG)
|
||||||
@ -97,29 +97,50 @@ include_directories(${CMAKE_CURRENT_SOURCE_DIR})
|
|||||||
|
|
||||||
if (EMSCRIPTEN)
|
if (EMSCRIPTEN)
|
||||||
add_subdirectory(whisper.wasm)
|
add_subdirectory(whisper.wasm)
|
||||||
|
set_target_properties(libmain PROPERTIES FOLDER "libs")
|
||||||
add_subdirectory(stream.wasm)
|
add_subdirectory(stream.wasm)
|
||||||
|
set_target_properties(libstream PROPERTIES FOLDER "libs")
|
||||||
add_subdirectory(command.wasm)
|
add_subdirectory(command.wasm)
|
||||||
|
set_target_properties(libcommand PROPERTIES FOLDER "libs")
|
||||||
|
add_subdirectory(talk.wasm)
|
||||||
|
set_target_properties(libtalk PROPERTIES FOLDER "libs")
|
||||||
add_subdirectory(bench.wasm)
|
add_subdirectory(bench.wasm)
|
||||||
|
set_target_properties(libbench PROPERTIES FOLDER "libs")
|
||||||
elseif(CMAKE_JS_VERSION)
|
elseif(CMAKE_JS_VERSION)
|
||||||
add_subdirectory(addon.node)
|
add_subdirectory(addon.node)
|
||||||
|
set_target_properties(addon.node PROPERTIES FOLDER "examples")
|
||||||
else()
|
else()
|
||||||
add_subdirectory(cli)
|
add_subdirectory(main)
|
||||||
add_subdirectory(bench)
|
set_target_properties(main PROPERTIES FOLDER "examples")
|
||||||
|
if (WHISPER_SDL2)
|
||||||
|
add_subdirectory(stream)
|
||||||
|
set_target_properties(stream PROPERTIES FOLDER "examples")
|
||||||
|
endif (WHISPER_SDL2)
|
||||||
add_subdirectory(server)
|
add_subdirectory(server)
|
||||||
|
set_target_properties(server PROPERTIES FOLDER "examples")
|
||||||
|
if (WHISPER_SDL2)
|
||||||
|
add_subdirectory(command)
|
||||||
|
set_target_properties(command PROPERTIES FOLDER "examples")
|
||||||
|
endif (WHISPER_SDL2)
|
||||||
|
add_subdirectory(bench)
|
||||||
|
set_target_properties(bench PROPERTIES FOLDER "examples")
|
||||||
add_subdirectory(quantize)
|
add_subdirectory(quantize)
|
||||||
if (WHISPER_SDL2)
|
set_target_properties(quantize PROPERTIES FOLDER "examples")
|
||||||
add_subdirectory(stream)
|
if (WHISPER_SDL2)
|
||||||
add_subdirectory(command)
|
add_subdirectory(talk)
|
||||||
add_subdirectory(talk-llama)
|
set_target_properties(talk PROPERTIES FOLDER "examples")
|
||||||
add_subdirectory(lsp)
|
add_subdirectory(talk-llama)
|
||||||
if (GGML_SYCL)
|
set_target_properties(talk-llama PROPERTIES FOLDER "examples")
|
||||||
add_subdirectory(sycl)
|
add_subdirectory(lsp)
|
||||||
endif()
|
set_target_properties(lsp PROPERTIES FOLDER "examples")
|
||||||
endif (WHISPER_SDL2)
|
if (GGML_SYCL)
|
||||||
|
add_subdirectory(sycl)
|
||||||
add_subdirectory(deprecation-warning)
|
set_target_properties(sycl PROPERTIES FOLDER "examples")
|
||||||
|
endif()
|
||||||
|
endif (WHISPER_SDL2)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (WHISPER_SDL2)
|
if (WHISPER_SDL2)
|
||||||
add_subdirectory(wchess)
|
add_subdirectory(wchess)
|
||||||
|
set_target_properties(wchess PROPERTIES FOLDER "examples")
|
||||||
endif (WHISPER_SDL2)
|
endif (WHISPER_SDL2)
|
||||||
|
@ -330,7 +330,6 @@ Napi::Value whisper(const Napi::CallbackInfo& info) {
|
|||||||
bool no_timestamps = whisper_params.Get("no_timestamps").As<Napi::Boolean>();
|
bool no_timestamps = whisper_params.Get("no_timestamps").As<Napi::Boolean>();
|
||||||
int32_t audio_ctx = whisper_params.Get("audio_ctx").As<Napi::Number>();
|
int32_t audio_ctx = whisper_params.Get("audio_ctx").As<Napi::Number>();
|
||||||
bool comma_in_time = whisper_params.Get("comma_in_time").As<Napi::Boolean>();
|
bool comma_in_time = whisper_params.Get("comma_in_time").As<Napi::Boolean>();
|
||||||
int32_t max_len = whisper_params.Get("max_len").As<Napi::Number>();
|
|
||||||
|
|
||||||
Napi::Value pcmf32Value = whisper_params.Get("pcmf32");
|
Napi::Value pcmf32Value = whisper_params.Get("pcmf32");
|
||||||
std::vector<float> pcmf32_vec;
|
std::vector<float> pcmf32_vec;
|
||||||
@ -353,7 +352,6 @@ Napi::Value whisper(const Napi::CallbackInfo& info) {
|
|||||||
params.audio_ctx = audio_ctx;
|
params.audio_ctx = audio_ctx;
|
||||||
params.pcmf32 = pcmf32_vec;
|
params.pcmf32 = pcmf32_vec;
|
||||||
params.comma_in_time = comma_in_time;
|
params.comma_in_time = comma_in_time;
|
||||||
params.max_len = max_len;
|
|
||||||
|
|
||||||
Napi::Function callback = info[1].As<Napi::Function>();
|
Napi::Function callback = info[1].As<Napi::Function>();
|
||||||
Worker* worker = new Worker(callback, params);
|
Worker* worker = new Worker(callback, params);
|
||||||
|
@ -18,7 +18,6 @@ const whisperParams = {
|
|||||||
translate: true,
|
translate: true,
|
||||||
no_timestamps: false,
|
no_timestamps: false,
|
||||||
audio_ctx: 0,
|
audio_ctx: 0,
|
||||||
max_len: 0,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const arguments = process.argv.slice(2);
|
const arguments = process.argv.slice(2);
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
set(TARGET whisper-bench)
|
set(TARGET bench)
|
||||||
add_executable(${TARGET} bench.cpp)
|
add_executable(${TARGET} bench.cpp)
|
||||||
|
|
||||||
include(DefaultTargetOptions)
|
include(DefaultTargetOptions)
|
||||||
|
|
||||||
target_link_libraries(${TARGET} PRIVATE whisper ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(${TARGET} PRIVATE whisper ${CMAKE_THREAD_LIBS_INIT})
|
||||||
|
|
||||||
install(TARGETS ${TARGET} RUNTIME)
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# whisper.cpp/examples/bench
|
# bench
|
||||||
|
|
||||||
A very basic tool for benchmarking the inference performance on your device. The tool simply runs the Encoder part of
|
A very basic tool for benchmarking the inference performance on your device. The tool simply runs the Encoder part of
|
||||||
the transformer on some random audio data and records the execution time. This way we can have an objective comparison
|
the transformer on some random audio data and records the execution time. This way we can have an objective comparison
|
||||||
@ -7,8 +7,11 @@ of the performance of the model for various setups.
|
|||||||
Benchmark results are tracked in the following Github issue: https://github.com/ggerganov/whisper.cpp/issues/89
|
Benchmark results are tracked in the following Github issue: https://github.com/ggerganov/whisper.cpp/issues/89
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# run the bench too on the small.en model using 4 threads
|
# build the bench tool
|
||||||
$ ./build/bin/whisper-bench -m ./models/ggml-small.en.bin -t 4
|
$ make bench
|
||||||
|
|
||||||
|
# run it on the small.en model using 4 threads
|
||||||
|
$ ./bench -m ./models/ggml-small.en.bin -t 4
|
||||||
|
|
||||||
whisper_model_load: loading model from './models/ggml-small.en.bin'
|
whisper_model_load: loading model from './models/ggml-small.en.bin'
|
||||||
whisper_model_load: n_vocab = 51864
|
whisper_model_load: n_vocab = 51864
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
if (WHISPER_SDL2)
|
if (WHISPER_SDL2)
|
||||||
set(TARGET whisper-command)
|
# command
|
||||||
|
set(TARGET command)
|
||||||
add_executable(${TARGET} command.cpp)
|
add_executable(${TARGET} command.cpp)
|
||||||
|
|
||||||
include(DefaultTargetOptions)
|
include(DefaultTargetOptions)
|
||||||
|
|
||||||
target_link_libraries(${TARGET} PRIVATE common common-sdl whisper ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(${TARGET} PRIVATE common common-sdl whisper ${CMAKE_THREAD_LIBS_INIT})
|
||||||
|
|
||||||
install(TARGETS ${TARGET} RUNTIME)
|
|
||||||
endif ()
|
endif ()
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
# whisper.cpp/examples/command
|
# command
|
||||||
|
|
||||||
This is a basic Voice Assistant example that accepts voice commands from the microphone.
|
This is a basic Voice Assistant example that accepts voice commands from the microphone.
|
||||||
More info is available in [issue #171](https://github.com/ggerganov/whisper.cpp/issues/171).
|
More info is available in [issue #171](https://github.com/ggerganov/whisper.cpp/issues/171).
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Run with default arguments and small model
|
# Run with default arguments and small model
|
||||||
./whisper-command -m ./models/ggml-small.en.bin -t 8
|
./command -m ./models/ggml-small.en.bin -t 8
|
||||||
|
|
||||||
# On Raspberry Pi, use tiny or base models + "-ac 768" for better performance
|
# On Raspberry Pi, use tiny or base models + "-ac 768" for better performance
|
||||||
./whisper-command -m ./models/ggml-tiny.en.bin -ac 768 -t 3 -c 0
|
./command -m ./models/ggml-tiny.en.bin -ac 768 -t 3 -c 0
|
||||||
```
|
```
|
||||||
|
|
||||||
https://user-images.githubusercontent.com/1991296/204038393-2f846eae-c255-4099-a76d-5735c25c49da.mp4
|
https://user-images.githubusercontent.com/1991296/204038393-2f846eae-c255-4099-a76d-5735c25c49da.mp4
|
||||||
@ -23,10 +23,10 @@ Initial tests show that this approach might be extremely efficient in terms of p
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Run in guided mode, the list of allowed commands is in commands.txt
|
# Run in guided mode, the list of allowed commands is in commands.txt
|
||||||
./whisper-command -m ./models/ggml-base.en.bin -cmd ./examples/command/commands.txt
|
./command -m ./models/ggml-base.en.bin -cmd ./examples/command/commands.txt
|
||||||
|
|
||||||
# On Raspberry Pi, in guided mode you can use "-ac 128" for extra performance
|
# On Raspberry Pi, in guided mode you can use "-ac 128" for extra performance
|
||||||
./whisper-command -m ./models/ggml-tiny.en.bin -cmd ./examples/command/commands.txt -ac 128 -t 3 -c 0
|
./command -m ./models/ggml-tiny.en.bin -cmd ./examples/command/commands.txt -ac 128 -t 3 -c 0
|
||||||
```
|
```
|
||||||
|
|
||||||
https://user-images.githubusercontent.com/1991296/207435352-8fc4ed3f-bde5-4555-9b8b-aeeb76bee969.mp4
|
https://user-images.githubusercontent.com/1991296/207435352-8fc4ed3f-bde5-4555-9b8b-aeeb76bee969.mp4
|
||||||
@ -34,7 +34,7 @@ https://user-images.githubusercontent.com/1991296/207435352-8fc4ed3f-bde5-4555-9
|
|||||||
|
|
||||||
## Building
|
## Building
|
||||||
|
|
||||||
The `whisper-command` tool depends on SDL2 library to capture audio from the microphone. You can build it like this:
|
The `command` tool depends on SDL2 library to capture audio from the microphone. You can build it like this:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Install SDL2
|
# Install SDL2
|
||||||
@ -47,6 +47,5 @@ sudo dnf install SDL2 SDL2-devel
|
|||||||
# Install SDL2 on Mac OS
|
# Install SDL2 on Mac OS
|
||||||
brew install sdl2
|
brew install sdl2
|
||||||
|
|
||||||
cmake -B build -DWHISPER_SDL2=ON
|
make command
|
||||||
cmake --build build --config Release
|
|
||||||
```
|
```
|
||||||
|
@ -21,12 +21,6 @@
|
|||||||
#include <thread>
|
#include <thread>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <chrono>
|
|
||||||
|
|
||||||
#if defined(_WIN32)
|
|
||||||
#define NOMINMAX
|
|
||||||
#include <windows.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// command-line parameters
|
// command-line parameters
|
||||||
struct whisper_params {
|
struct whisper_params {
|
||||||
@ -685,10 +679,6 @@ static int process_general_transcription(struct whisper_context * ctx, audio_asy
|
|||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char ** argv) {
|
int main(int argc, char ** argv) {
|
||||||
#if defined(_WIN32)
|
|
||||||
SetConsoleOutputCP(CP_UTF8);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
whisper_params params;
|
whisper_params params;
|
||||||
|
|
||||||
if (whisper_params_parse(argc, argv, params) == false) {
|
if (whisper_params_parse(argc, argv, params) == false) {
|
||||||
|
@ -209,8 +209,6 @@ bool ggml_common_quantize_0(
|
|||||||
case GGML_TYPE_IQ4_XS:
|
case GGML_TYPE_IQ4_XS:
|
||||||
case GGML_TYPE_IQ1_M:
|
case GGML_TYPE_IQ1_M:
|
||||||
case GGML_TYPE_BF16:
|
case GGML_TYPE_BF16:
|
||||||
case GGML_TYPE_TQ1_0:
|
|
||||||
case GGML_TYPE_TQ2_0:
|
|
||||||
case GGML_TYPE_COUNT:
|
case GGML_TYPE_COUNT:
|
||||||
{
|
{
|
||||||
fprintf(stderr, "%s: unsupported quantization type %d (%s)\n", __func__, ttype, ggml_type_name((ggml_type) ttype));
|
fprintf(stderr, "%s: unsupported quantization type %d (%s)\n", __func__, ttype, ggml_type_name((ggml_type) ttype));
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
#include "common-sdl.h"
|
#include "common-sdl.h"
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
|
|
||||||
audio_async::audio_async(int len_ms) {
|
audio_async::audio_async(int len_ms) {
|
||||||
m_len_ms = len_ms;
|
m_len_ms = len_ms;
|
||||||
|
|
||||||
|
@ -147,6 +147,7 @@ std::string gpt_random_prompt(std::mt19937 & rng) {
|
|||||||
case 7: return "He";
|
case 7: return "He";
|
||||||
case 8: return "She";
|
case 8: return "She";
|
||||||
case 9: return "They";
|
case 9: return "They";
|
||||||
|
default: return "To";
|
||||||
}
|
}
|
||||||
|
|
||||||
return "The";
|
return "The";
|
||||||
|
@ -9,7 +9,6 @@
|
|||||||
#include <thread>
|
#include <thread>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <sstream>
|
|
||||||
|
|
||||||
#define COMMON_SAMPLE_RATE 16000
|
#define COMMON_SAMPLE_RATE 16000
|
||||||
|
|
||||||
@ -287,43 +286,12 @@ void sam_print_usage(int argc, char ** argv, const sam_params & params);
|
|||||||
// Terminal utils
|
// Terminal utils
|
||||||
//
|
//
|
||||||
|
|
||||||
#define SQR(X) ((X) * (X))
|
|
||||||
#define UNCUBE(x) x < 48 ? 0 : x < 115 ? 1 : (x - 35) / 40
|
|
||||||
|
|
||||||
/**
|
// Terminal color map. 10 colors grouped in ranges [0.0, 0.1, ..., 0.9]
|
||||||
* Quantizes 24-bit RGB to xterm256 code range [16,256).
|
// Lowest is red, middle is yellow, highest is green.
|
||||||
*/
|
|
||||||
static int rgb2xterm256(int r, int g, int b) {
|
|
||||||
unsigned char cube[] = {0, 0137, 0207, 0257, 0327, 0377};
|
|
||||||
int av, ir, ig, ib, il, qr, qg, qb, ql;
|
|
||||||
av = r * .299 + g * .587 + b * .114 + .5;
|
|
||||||
ql = (il = av > 238 ? 23 : (av - 3) / 10) * 10 + 8;
|
|
||||||
qr = cube[(ir = UNCUBE(r))];
|
|
||||||
qg = cube[(ig = UNCUBE(g))];
|
|
||||||
qb = cube[(ib = UNCUBE(b))];
|
|
||||||
if (SQR(qr - r) + SQR(qg - g) + SQR(qb - b) <=
|
|
||||||
SQR(ql - r) + SQR(ql - g) + SQR(ql - b))
|
|
||||||
return ir * 36 + ig * 6 + ib + 020;
|
|
||||||
return il + 0350;
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::string set_xterm256_foreground(int r, int g, int b) {
|
|
||||||
int x = rgb2xterm256(r, g, b);
|
|
||||||
std::ostringstream oss;
|
|
||||||
oss << "\033[38;5;" << x << "m";
|
|
||||||
return oss.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lowest is red, middle is yellow, highest is green. Color scheme from
|
|
||||||
// Paul Tol; it is colorblind friendly https://personal.sron.nl/~pault/
|
|
||||||
const std::vector<std::string> k_colors = {
|
const std::vector<std::string> k_colors = {
|
||||||
set_xterm256_foreground(220, 5, 12),
|
"\033[38;5;196m", "\033[38;5;202m", "\033[38;5;208m", "\033[38;5;214m", "\033[38;5;220m",
|
||||||
set_xterm256_foreground(232, 96, 28),
|
"\033[38;5;226m", "\033[38;5;190m", "\033[38;5;154m", "\033[38;5;118m", "\033[38;5;82m",
|
||||||
set_xterm256_foreground(241, 147, 45),
|
|
||||||
set_xterm256_foreground(246, 193, 65),
|
|
||||||
set_xterm256_foreground(247, 240, 86),
|
|
||||||
set_xterm256_foreground(144, 201, 135),
|
|
||||||
set_xterm256_foreground( 78, 178, 101),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -1,4 +0,0 @@
|
|||||||
add_executable(main ./deprecation-warning.cpp)
|
|
||||||
add_executable(bench ./deprecation-warning.cpp)
|
|
||||||
add_executable(stream ./deprecation-warning.cpp)
|
|
||||||
add_executable(command ./deprecation-warning.cpp)
|
|
@ -1,17 +0,0 @@
|
|||||||
# Migration notice for binary filenames
|
|
||||||
|
|
||||||
> [!IMPORTANT]
|
|
||||||
[2024 Dec 20] Binaries have been renamed w/ a `whisper-` prefix. `main` is now `whisper-cli`, `server` is `whisper-server`, etc (https://github.com/ggerganov/whisper.cpp/pull/2648)
|
|
||||||
|
|
||||||
This migration was important, but it is a breaking change that may not always be immediately obvious to users.
|
|
||||||
|
|
||||||
Please update all scripts and workflows to use the new binary names.
|
|
||||||
|
|
||||||
| Old Filename | New Filename |
|
|
||||||
| ---- | ---- |
|
|
||||||
| main | whisper-cli |
|
|
||||||
| bench | whisper-bench |
|
|
||||||
| stream | whisper-stream |
|
|
||||||
| command | whisper-command |
|
|
||||||
| server | whisper-server |
|
|
||||||
| talk-llama | whisper-talk-llama |
|
|
@ -1,38 +0,0 @@
|
|||||||
// Warns users that this filename was deprecated, and provides a link for more information.
|
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
// Main
|
|
||||||
int main(int argc, char** argv) {
|
|
||||||
std::string filename = "main";
|
|
||||||
if (argc >= 1) {
|
|
||||||
filename = argv[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get only the program name from the full path
|
|
||||||
size_t pos = filename.find_last_of("/\\");
|
|
||||||
if (pos != std::string::npos) {
|
|
||||||
filename = filename.substr(pos+1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Append "whisper-" to the beginning of filename to get the replacemnt filename
|
|
||||||
std::string replacement_filename = "whisper-" + filename;
|
|
||||||
|
|
||||||
// The exception is if the filename is "main", then our replacement filename is "whisper-cli"
|
|
||||||
if (filename == "main") {
|
|
||||||
replacement_filename = "whisper-cli";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filename == "main.exe") {
|
|
||||||
replacement_filename = "whisper-cli.exe";
|
|
||||||
}
|
|
||||||
|
|
||||||
fprintf(stdout, "\n");
|
|
||||||
fprintf(stdout, "WARNING: The binary '%s' is deprecated.\n", filename.c_str());
|
|
||||||
fprintf(stdout, " Please use '%s' instead.\n", replacement_filename.c_str());
|
|
||||||
fprintf(stdout, " See https://github.com/ggerganov/whisper.cpp/tree/master/examples/deprecation-warning/README.md for more information.\n");
|
|
||||||
fprintf(stdout, "\n");
|
|
||||||
|
|
||||||
return EXIT_FAILURE;
|
|
||||||
}
|
|
4429
examples/dr_wav.h
4429
examples/dr_wav.h
File diff suppressed because it is too large
Load Diff
@ -204,6 +204,8 @@ static int decode_audio(struct audio_buffer *audio_buf, s16 **data, int *size)
|
|||||||
const size_t errbuffsize = 1024;
|
const size_t errbuffsize = 1024;
|
||||||
char errbuff[errbuffsize];
|
char errbuff[errbuffsize];
|
||||||
|
|
||||||
|
av_register_all(); // from avformat. Still a must-have call for ffmpeg v3! (can be skipped for later versions)
|
||||||
|
|
||||||
fmt_ctx = avformat_alloc_context();
|
fmt_ctx = avformat_alloc_context();
|
||||||
avio_ctx_buffer = (u8*)av_malloc(AVIO_CTX_BUF_SZ);
|
avio_ctx_buffer = (u8*)av_malloc(AVIO_CTX_BUF_SZ);
|
||||||
LOG("Creating an avio context: AVIO_CTX_BUF_SZ=%d\n", AVIO_CTX_BUF_SZ);
|
LOG("Creating an avio context: AVIO_CTX_BUF_SZ=%d\n", AVIO_CTX_BUF_SZ);
|
||||||
@ -319,7 +321,7 @@ int ffmpeg_decode_audio(const std::string &ifname, std::vector<uint8_t>& owav_da
|
|||||||
LOG("Couldn't map input file %s\n", ifname.c_str());
|
LOG("Couldn't map input file %s\n", ifname.c_str());
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
LOG("Mapped input file: %s size: %d\n", ibuf, (int) ibuf_size);
|
LOG("Mapped input file: %x size: %d\n", ibuf, ibuf_size);
|
||||||
struct audio_buffer inaudio_buf;
|
struct audio_buffer inaudio_buf;
|
||||||
inaudio_buf.ptr = ibuf;
|
inaudio_buf.ptr = ibuf;
|
||||||
inaudio_buf.size = ibuf_size;
|
inaudio_buf.size = ibuf_size;
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
# Press Ctrl+C to stop recording
|
# Press Ctrl+C to stop recording
|
||||||
#
|
#
|
||||||
|
|
||||||
executable="./build/bin/whisper-cli"
|
executable="./main"
|
||||||
model="base.en"
|
model="base.en"
|
||||||
model_path="models/ggml-$model.bin"
|
model_path="models/ggml-$model.bin"
|
||||||
|
|
||||||
@ -46,7 +46,7 @@ ffmpeg -y -i ./rec.wav -ar 16000 -ac 1 -c:a pcm_s16le ./rec16.wav > /dev/null 2>
|
|||||||
|
|
||||||
# run Whisper
|
# run Whisper
|
||||||
echo "Processing ..."
|
echo "Processing ..."
|
||||||
${executable} -m models/ggml-base.en.bin rec16.wav -owts > /dev/null 2>&1
|
./main -m models/ggml-base.en.bin rec16.wav -owts > /dev/null 2>&1
|
||||||
|
|
||||||
# generate Karaoke video
|
# generate Karaoke video
|
||||||
echo "Generating video ..."
|
echo "Generating video ..."
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user