Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/segmentio/asm
Go library providing algorithms optimized to leverage the characteristics of modern CPUs
https://github.com/segmentio/asm
arm assembler assembly avo branch-prediction go golang simd x86
Last synced: 3 days ago
JSON representation
Go library providing algorithms optimized to leverage the characteristics of modern CPUs
- Host: GitHub
- URL: https://github.com/segmentio/asm
- Owner: segmentio
- License: mit-0
- Created: 2021-04-13T21:57:03.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2023-11-07T21:27:52.000Z (about 1 year ago)
- Last Synced: 2024-10-29T15:11:57.389Z (3 months ago)
- Topics: arm, assembler, assembly, avo, branch-prediction, go, golang, simd, x86
- Language: Go
- Homepage:
- Size: 435 KB
- Stars: 870
- Watchers: 12
- Forks: 36
- Open Issues: 9
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
- my-awesome - segmentio/asm - prediction,go,golang,simd,x86 pushed_at:2023-11 star:0.9k fork:0.0k Go library providing algorithms optimized to leverage the characteristics of modern CPUs (Go)
- awesome-golang-repositories - asm
README
# asm ![build status](https://github.com/segmentio/asm/actions/workflows/go.yml/badge.svg) [![GoDoc](https://godoc.org/github.com/segmentio/asm?status.svg)](https://godoc.org/github.com/segmentio/asm)
Go library providing algorithms that use the full power of modern CPUs to get
the best performance.## Motivation
The cloud makes it easier than ever to access large scale compute capacity,
and it's become common to run distributed systems deployed across dozens or
sometimes hundreds of CPUs. Because projects run on so many cores now, program
performance and efficiency matters more today than it has ever before.Modern CPUs are complex machines with performance characteristics that may
vary by orders of magnitude depending on how they are used. Features like
branch prediction, instruction reordering, pipelining, or caching are all
input variables that determine the compute throughput that a CPU can achieve.
While compilers keep being improved, and often employ micro-optimizations that
would be counter-productive for human developers to be responsible for, there
are limitations to what they can do, and Assembly still has a role to play in
optimizing algorithms on hot code paths of large scale applications.SIMD instruction sets offer interesting opportunities for software engineers.
Taking advantage of these instructions often requires rethinking how the program
represents and manipulates data, which is beyond the realm of optimizations that
can be implemented by a compiler. When renting CPU time from a Cloud provider,
programs that fail to leverage the full sets of instructions available are
therefore paying for features they do not use.This package aims to provide such algorithms, optimized to leverage advanced
instruction sets of modern CPUs to maximize throughput and take the best
advantage of the available compute power. Users of the package will find
functions that have often been designed to work on **arrays of values**,
which is where SIMD and branchless algorithms shine.The functions in this library have been used in high throughput production
environments at Segment, we hope that they will be useful to other developers
using Go in performance-sensitive software.## Usage
The library is composed of multiple Go packages intended to act as logical
groups of functions sharing similar properties:| Package | Purpose |
| ------- | ------- |
| [ascii](ascii) | library of functions designed to work on ASCII inputs |
| [base64](base64) | standard library compatible base64 encodings |
| [bswap](bswap) | byte swapping algorithms working on arrays of fixed-size items |
| [cpu](cpu) | definition of the ABI used to detect CPU features |
| [mem](mem) | functions operating on byte arrays |
| [qsort](qsort) | quick-sort implementations for arrays of fixed-size items |
| [slices](slices) | functions performing computations on pairs of slices |
| [sortedset](sortedset) | functions working on sorted arrays of fixed-size items |When no assembly version of a function is available for the target platform,
the package provides a generic implementation in Go which is automatically
picked up by the compiler.## Showcase
The purpose of this library being to improve the runtime efficiency of Go
programs, we compiled a few snapshots of benchmark runs to showcase the
kind of improvements that these code paths can expect from leveraging
SIMD and branchless optimizations:```
goos: darwin
goarch: amd64
cpu: Intel(R) Core(TM) i9-8950HK CPU @ 2.90GHz
``````
pkg: github.com/segmentio/asm/ascii
name old time/op new time/op delta
EqualFoldString/0512 276ns ± 1% 21ns ± 2% -92.50% (p=0.008 n=5+5)name old speed new speed delta
EqualFoldString/0512 3.71GB/s ± 1% 49.44GB/s ± 2% +1232.79% (p=0.008 n=5+5)
``````
pkg: github.com/segmentio/asm/bswap
name old time/op new time/op delta
Swap64 11.2µs ± 1% 0.9µs ± 9% -92.06% (p=0.008 n=5+5)name old speed new speed delta
Swap64 5.83GB/s ± 1% 73.67GB/s ± 9% +1162.98% (p=0.008 n=5+5)
``````
pkg: github.com/segmentio/asm/qsort
name old time/op new time/op delta
Sort16/1000000 269ms ± 2% 46ms ± 3% -83.08% (p=0.008 n=5+5)name old speed new speed delta
Sort16/1000000 59.4MB/s ± 2% 351.2MB/s ± 3% +491.24% (p=0.008 n=5+5)
```## Maintenance
The assembly code is generated with [AVO](https://github.com/mmcloughlin/avo),
and orchestrated by a Makefile which helps maintainers rebuild the assembly
source code when the AVO files are modified.The repository contains two Go modules; the main module is declared as
`github.com/segmentio/asm` at the root of the repository, and the second
module is found in the `build` subdirectory.The `build` module is used to isolate build dependencies from programs that
import the main module. Through this mechanism, AVO does not become a
dependency of programs using `github.com/segmentio/asm`, keeping the
dependency management overhead minimal for the users, and allowing
maintainers to make modifications to the `build` package.Versioning of the two modules is managed independently; while we aim to provide
stable APIs on the main package, breaking changes may be introduced on the
`build` package more often, as it is intended to be ground for more experimental
constructs in the project.### Requirements
Some libraries have custom purpose code for both amd64 and arm64. Others (qsort)
have only amd64. Search for a `.s` file matching your architecture to be sure
you are using the assembler optimized library instructions.The Go code requires Go 1.17 or above. These versions contain significant
performance improvements compared to previous Go versions.`asm` version v1.1.5 and earlier maintain compatibility with Go 1.16.
### purego
Programs in the `build` module should add the following declaration:
```go
func init() {
ConstraintExpr("!purego")
}
```It instructs AVO to inject the `!purego` tag in the generated files, allowing
the libraries to be compiled without any assembly optimizations with a build
command such as:```
go build -tags purego ...
```This is mainly useful to compare the impact of using the assembly optimized
versions instead of the simpler Go-only implementations.