From b869c757a929714d84b4b06cd6055df45e807806 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 09:56:41 +0300 Subject: [PATCH 01/18] docs/FAQ.md: add an answer for `What is the difference between single-node and cluster versions of VictoriaMetrics?` --- docs/FAQ.md | 29 +++++++++++++++++++++++++++++ 1 file changed, 29 insertions(+) diff --git a/docs/FAQ.md b/docs/FAQ.md index dd4951d3d..520e533e6 100644 --- a/docs/FAQ.md +++ b/docs/FAQ.md @@ -235,6 +235,30 @@ It supports [high cardinality data](https://medium.com/@valyala/high-cardinality perfectly [scales up on a single node](https://medium.com/@valyala/measuring-vertical-scalability-for-time-series-databases-in-google-cloud-92550d78d8ae) and scales horizontally to multiple nodes. +## What is the difference between single-node and cluster versions of VictoriaMetrics? + +Both [single-node](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html) and +[cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html) versions of VictoriaMetrics +share the core source code, so they have many common features. They have the following differences though: + +* [Single-node VictoriaMetrics](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html) runs on a single host, + while [cluster version of VictoriaMetrics](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html) can scale to many hosts. + Single-node VictoriaMetrics scales vertically though, e.g. its capacity and performance scales almost linearly when increasing + available CPU, RAM, disk IO and disk space. See [an article about vertical scalability of a single-node VictoriaMetrics](https://valyala.medium.com/measuring-vertical-scalability-for-time-series-databases-in-google-cloud-92550d78d8ae). + +* Cluster version of VictoriaMetrics supports [multitenancy](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#multitenancy), + while single-node VictoriaMetrics doesn't support it. + +* Cluster version of VictoriaMetrics supports data replication, while single-node VictoriaMetrics relies on the durability + of the persistent storage pointed by `-storageDataPath` command-line flag. + See [these docs](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#replication-and-data-safety) for details. + +* Single-node VictoriaMetrics provides higher capacity and performance comparing to cluster version of VictoriaMetrics + when running on the same hardware with the same amounts of CPU and RAM, since it has no overhead on data transfer + between cluster components over the network. + +See also [which type of VictoriaMetrics is recommended to use](#which-victoriametrics-type-is-recommended-for-use-in-production---single-node-or-cluster). + ## Where can I ask questions about VictoriaMetrics? Questions about VictoriaMetrics can be asked via the following channels: @@ -303,6 +327,11 @@ VictoriaMetrics maintains in-memory cache for mapping of [active time series](#w See [this article](https://valyala.medium.com/how-to-optimize-promql-and-metricsql-queries-85a1b75bf986). +VictoriaMetrics also provides [query tracer](https://docs.victoriametrics.com/#query-tracing) and [cardinality explorer](https://docs.victoriametrics.com/#cardinality-explorer), +which can help during query optimization. + +See also [troubleshooting slow queries](https://docs.victoriametrics.com/Troubleshooting.html#slow-queries). + ## Which VictoriaMetrics type is recommended for use in production - single-node or cluster? Both [single-node VictoriaMetrics](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html) and From a9629cc32dd5872bab4548362f1bb9eaa1a2418d Mon Sep 17 00:00:00 2001 From: Dmytro Kozlov Date: Thu, 15 Sep 2022 11:46:54 +0300 Subject: [PATCH 02/18] app/vmctl: add description about `influx-skip-database-label` flag (#3111) --- app/vmctl/README.md | 1 + docs/vmctl.md | 1 + 2 files changed, 2 insertions(+) diff --git a/app/vmctl/README.md b/app/vmctl/README.md index 35e3f5e70..014ee1cd9 100644 --- a/app/vmctl/README.md +++ b/app/vmctl/README.md @@ -242,6 +242,7 @@ Found 40000 timeseries to import. Continue? [Y/n] y Vmctl maps InfluxDB data the same way as VictoriaMetrics does by using the following rules: - `influx-database` arg is mapped into `db` label value unless `db` tag exists in the InfluxDB line. +If you want to skip this mapping just enable flag `influx-skip-database-label`. - Field names are mapped to time series names prefixed with {measurement}{separator} value, where {separator} equals to _ by default. It can be changed with `--influx-measurement-field-separator` command-line flag. diff --git a/docs/vmctl.md b/docs/vmctl.md index 5781696e8..e5e3631e7 100644 --- a/docs/vmctl.md +++ b/docs/vmctl.md @@ -246,6 +246,7 @@ Found 40000 timeseries to import. Continue? [Y/n] y Vmctl maps InfluxDB data the same way as VictoriaMetrics does by using the following rules: - `influx-database` arg is mapped into `db` label value unless `db` tag exists in the InfluxDB line. +If you want to skip this mapping just enable flag `influx-skip-database-label`. - Field names are mapped to time series names prefixed with {measurement}{separator} value, where {separator} equals to _ by default. It can be changed with `--influx-measurement-field-separator` command-line flag. From bbecd27557e2973eca35a2f5c96ab2b03ef5849a Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 12:53:56 +0300 Subject: [PATCH 03/18] docs: update docs with explicitly marked enterprise command-line flags for VictoriaMetrics and vmagent --- README.md | 12 ++++++------ app/vmagent/README.md | 22 +++++++++++----------- docs/README.md | 18 +++++++++--------- docs/Single-server-VictoriaMetrics.md | 18 +++++++++--------- docs/vmagent.md | 22 +++++++++++----------- 5 files changed, 46 insertions(+), 46 deletions(-) diff --git a/README.md b/README.md index be34a205e..d3d1dd515 100644 --- a/README.md +++ b/README.md @@ -2002,7 +2002,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -denyQueryTracing Whether to disable the ability to trace queries. See https://docs.victoriametrics.com/#query-tracing -downsampling.period array - Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details + Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -dryRun Whether to check only -promscrape.config and then exit. Unknown config entries aren't allowed in -promscrape.config by default. This can be changed with -promscrape.config.strictParse=false command-line flag @@ -2013,7 +2013,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -finalMergeDelay duration The delay before starting final merge for per-month partition after no new data is ingested into it. Final merge may require additional disk IO and CPU resources. Final merge may increase query speed and reduce disk space usage in some cases. Zero value disables final merge -flagsAuthKey string @@ -2196,7 +2196,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -promscrape.suppressScrapeErrorsDelay duration The delay for suppressing repeated scrape errors logging per each scrape targets. This may be used for reducing the number of log lines related to scrape errors. See also -promscrape.suppressScrapeErrors -promscrape.yandexcloudSDCheckInterval duration - Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. (default 30s) + Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#yandexcloud_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. @@ -2221,9 +2221,9 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.disableCache Whether to disable response caching. This may be useful during data backfilling -search.graphiteMaxPointsPerSeries int - The maximum number of points per series Graphite render API can return (default 1000000) + The maximum number of points per series Graphite render API can return. This flag is available only in enterprise version of VictoriaMetrics (default 1000000) -search.graphiteStorageStep duration - The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API (default 10s) + The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API. This flag is available only in enterprise version of VictoriaMetrics (default 10s) -search.latencyOffset duration The time when data points become visible in query results after the collection. Too small value can result in incomplete last points for query results (default 30s) -search.logSlowQueryDuration duration @@ -2237,7 +2237,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.maxFederateSeries int The maximum number of time series, which can be returned from /federate. This option allows limiting memory usage (default 1000000) -search.maxGraphiteSeries int - The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage (default 300000) + The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage . This flag is available only in enterprise version of VictoriaMetrics (default 300000) -search.maxLookback duration Synonym to -search.lookback-delta from Prometheus. The value is dynamically detected from interval between time series datapoints if not set. It can be overridden on per-query basis via max_lookback arg. See also '-search.maxStalenessInterval' flag, which has the same meaining due to historical reasons -search.maxPointsPerTimeseries int diff --git a/app/vmagent/README.md b/app/vmagent/README.md index 8e7fb2542..0db26b0d9 100644 --- a/app/vmagent/README.md +++ b/app/vmagent/README.md @@ -926,7 +926,7 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap @@ -977,30 +977,30 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -insert.maxQueueDuration duration The maximum duration for waiting in the queue for insert requests due to -maxConcurrentInserts (default 1m0s) -kafka.consumer.topic array - Kafka topic names for data consumption. + Kafka topic names for data consumption. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.basicAuth.password array - Optional basic auth password for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' + Optional basic auth password for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.basicAuth.username array - Optional basic auth username for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' + Optional basic auth username for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.brokers array - List of brokers to connect for given topic, e.g. -kafka.consumer.topic.broker=host-1:9092;host-2:9092 + List of brokers to connect for given topic, e.g. -kafka.consumer.topic.broker=host-1:9092;host-2:9092 . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.defaultFormat string - Expected data format in the topic if -kafka.consumer.topic.format is skipped. (default "promremotewrite") + Expected data format in the topic if -kafka.consumer.topic.format is skipped. This flag is available only in enterprise version of VictoriaMetrics (default "promremotewrite") -kafka.consumer.topic.format array - data format for corresponding kafka topic. Valid formats: influx, prometheus, promremotewrite, graphite, jsonline + data format for corresponding kafka topic. Valid formats: influx, prometheus, promremotewrite, graphite, jsonline . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.groupID array - Defines group.id for topic + Defines group.id for topic. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.isGzipped array - Enables gzip setting for topic messages payload. Only prometheus, jsonline and influx formats accept gzipped messages. + Enables gzip setting for topic messages payload. Only prometheus, jsonline and influx formats accept gzipped messages.This flag is available only in enterprise version of VictoriaMetrics Supports array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.options array - Optional key=value;key1=value2 settings for topic consumer. See full configuration options at https://github.com/edenhill/librdkafka/blob/master/CONFIGURATION.md. + Optional key=value;key1=value2 settings for topic consumer. See full configuration options at https://github.com/edenhill/librdkafka/blob/master/CONFIGURATION.md . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -loggerDisableTimestamps Whether to disable writing timestamps in logs @@ -1121,7 +1121,7 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -promscrape.suppressScrapeErrorsDelay duration The delay for suppressing repeated scrape errors logging per each scrape targets. This may be used for reducing the number of log lines related to scrape errors. See also -promscrape.suppressScrapeErrors -promscrape.yandexcloudSDCheckInterval duration - Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. (default 30s) + Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#yandexcloud_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. diff --git a/docs/README.md b/docs/README.md index c1c4eaa29..d3d1dd515 100644 --- a/docs/README.md +++ b/docs/README.md @@ -779,7 +779,7 @@ to your needs or when testing bugfixes. ### Development build -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics` binary and puts it into the `bin` folder. @@ -795,7 +795,7 @@ ARM build may run on Raspberry Pi or on [energy-efficient ARM servers](https://b ### Development ARM build -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics-linux-arm` or `make victoria-metrics-linux-arm64` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics-linux-arm` or `victoria-metrics-linux-arm64` binary respectively and puts it into the `bin` folder. @@ -809,7 +809,7 @@ ARM build may run on Raspberry Pi or on [energy-efficient ARM servers](https://b `Pure Go` mode builds only Go code without [cgo](https://golang.org/cmd/cgo/) dependencies. -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics-pure` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics-pure` binary and puts it into the `bin` folder. @@ -2002,7 +2002,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -denyQueryTracing Whether to disable the ability to trace queries. See https://docs.victoriametrics.com/#query-tracing -downsampling.period array - Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details + Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -dryRun Whether to check only -promscrape.config and then exit. Unknown config entries aren't allowed in -promscrape.config by default. This can be changed with -promscrape.config.strictParse=false command-line flag @@ -2013,7 +2013,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -finalMergeDelay duration The delay before starting final merge for per-month partition after no new data is ingested into it. Final merge may require additional disk IO and CPU resources. Final merge may increase query speed and reduce disk space usage in some cases. Zero value disables final merge -flagsAuthKey string @@ -2196,7 +2196,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -promscrape.suppressScrapeErrorsDelay duration The delay for suppressing repeated scrape errors logging per each scrape targets. This may be used for reducing the number of log lines related to scrape errors. See also -promscrape.suppressScrapeErrors -promscrape.yandexcloudSDCheckInterval duration - Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. (default 30s) + Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#yandexcloud_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. @@ -2221,9 +2221,9 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.disableCache Whether to disable response caching. This may be useful during data backfilling -search.graphiteMaxPointsPerSeries int - The maximum number of points per series Graphite render API can return (default 1000000) + The maximum number of points per series Graphite render API can return. This flag is available only in enterprise version of VictoriaMetrics (default 1000000) -search.graphiteStorageStep duration - The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API (default 10s) + The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API. This flag is available only in enterprise version of VictoriaMetrics (default 10s) -search.latencyOffset duration The time when data points become visible in query results after the collection. Too small value can result in incomplete last points for query results (default 30s) -search.logSlowQueryDuration duration @@ -2237,7 +2237,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.maxFederateSeries int The maximum number of time series, which can be returned from /federate. This option allows limiting memory usage (default 1000000) -search.maxGraphiteSeries int - The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage (default 300000) + The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage . This flag is available only in enterprise version of VictoriaMetrics (default 300000) -search.maxLookback duration Synonym to -search.lookback-delta from Prometheus. The value is dynamically detected from interval between time series datapoints if not set. It can be overridden on per-query basis via max_lookback arg. See also '-search.maxStalenessInterval' flag, which has the same meaining due to historical reasons -search.maxPointsPerTimeseries int diff --git a/docs/Single-server-VictoriaMetrics.md b/docs/Single-server-VictoriaMetrics.md index 203bc28da..68403c082 100644 --- a/docs/Single-server-VictoriaMetrics.md +++ b/docs/Single-server-VictoriaMetrics.md @@ -783,7 +783,7 @@ to your needs or when testing bugfixes. ### Development build -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics` binary and puts it into the `bin` folder. @@ -799,7 +799,7 @@ ARM build may run on Raspberry Pi or on [energy-efficient ARM servers](https://b ### Development ARM build -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics-linux-arm` or `make victoria-metrics-linux-arm64` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics-linux-arm` or `victoria-metrics-linux-arm64` binary respectively and puts it into the `bin` folder. @@ -813,7 +813,7 @@ ARM build may run on Raspberry Pi or on [energy-efficient ARM servers](https://b `Pure Go` mode builds only Go code without [cgo](https://golang.org/cmd/cgo/) dependencies. -1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.18. +1. [Install Go](https://golang.org/doc/install). The minimum supported version is Go 1.19.1 2. Run `make victoria-metrics-pure` from the root folder of [the repository](https://github.com/VictoriaMetrics/VictoriaMetrics). It builds `victoria-metrics-pure` binary and puts it into the `bin` folder. @@ -2006,7 +2006,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -denyQueryTracing Whether to disable the ability to trace queries. See https://docs.victoriametrics.com/#query-tracing -downsampling.period array - Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details + Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -dryRun Whether to check only -promscrape.config and then exit. Unknown config entries aren't allowed in -promscrape.config by default. This can be changed with -promscrape.config.strictParse=false command-line flag @@ -2017,7 +2017,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -finalMergeDelay duration The delay before starting final merge for per-month partition after no new data is ingested into it. Final merge may require additional disk IO and CPU resources. Final merge may increase query speed and reduce disk space usage in some cases. Zero value disables final merge -flagsAuthKey string @@ -2200,7 +2200,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -promscrape.suppressScrapeErrorsDelay duration The delay for suppressing repeated scrape errors logging per each scrape targets. This may be used for reducing the number of log lines related to scrape errors. See also -promscrape.suppressScrapeErrors -promscrape.yandexcloudSDCheckInterval duration - Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. (default 30s) + Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#yandexcloud_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. @@ -2225,9 +2225,9 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.disableCache Whether to disable response caching. This may be useful during data backfilling -search.graphiteMaxPointsPerSeries int - The maximum number of points per series Graphite render API can return (default 1000000) + The maximum number of points per series Graphite render API can return. This flag is available only in enterprise version of VictoriaMetrics (default 1000000) -search.graphiteStorageStep duration - The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API (default 10s) + The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API. This flag is available only in enterprise version of VictoriaMetrics (default 10s) -search.latencyOffset duration The time when data points become visible in query results after the collection. Too small value can result in incomplete last points for query results (default 30s) -search.logSlowQueryDuration duration @@ -2241,7 +2241,7 @@ Pass `-help` to VictoriaMetrics in order to see the list of supported command-li -search.maxFederateSeries int The maximum number of time series, which can be returned from /federate. This option allows limiting memory usage (default 1000000) -search.maxGraphiteSeries int - The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage (default 300000) + The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage . This flag is available only in enterprise version of VictoriaMetrics (default 300000) -search.maxLookback duration Synonym to -search.lookback-delta from Prometheus. The value is dynamically detected from interval between time series datapoints if not set. It can be overridden on per-query basis via max_lookback arg. See also '-search.maxStalenessInterval' flag, which has the same meaining due to historical reasons -search.maxPointsPerTimeseries int diff --git a/docs/vmagent.md b/docs/vmagent.md index 502d6080f..9a8e39a4d 100644 --- a/docs/vmagent.md +++ b/docs/vmagent.md @@ -930,7 +930,7 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap @@ -981,30 +981,30 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -insert.maxQueueDuration duration The maximum duration for waiting in the queue for insert requests due to -maxConcurrentInserts (default 1m0s) -kafka.consumer.topic array - Kafka topic names for data consumption. + Kafka topic names for data consumption. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.basicAuth.password array - Optional basic auth password for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' + Optional basic auth password for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.basicAuth.username array - Optional basic auth username for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' + Optional basic auth username for -kafka.consumer.topic. Must be used in conjunction with any supported auth methods for kafka client, specified by flag -kafka.consumer.topic.options='security.protocol=SASL_SSL;sasl.mechanisms=PLAIN' . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.brokers array - List of brokers to connect for given topic, e.g. -kafka.consumer.topic.broker=host-1:9092;host-2:9092 + List of brokers to connect for given topic, e.g. -kafka.consumer.topic.broker=host-1:9092;host-2:9092 . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.defaultFormat string - Expected data format in the topic if -kafka.consumer.topic.format is skipped. (default "promremotewrite") + Expected data format in the topic if -kafka.consumer.topic.format is skipped. This flag is available only in enterprise version of VictoriaMetrics (default "promremotewrite") -kafka.consumer.topic.format array - data format for corresponding kafka topic. Valid formats: influx, prometheus, promremotewrite, graphite, jsonline + data format for corresponding kafka topic. Valid formats: influx, prometheus, promremotewrite, graphite, jsonline . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.groupID array - Defines group.id for topic + Defines group.id for topic. This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.isGzipped array - Enables gzip setting for topic messages payload. Only prometheus, jsonline and influx formats accept gzipped messages. + Enables gzip setting for topic messages payload. Only prometheus, jsonline and influx formats accept gzipped messages.This flag is available only in enterprise version of VictoriaMetrics Supports array of values separated by comma or specified via multiple flags. -kafka.consumer.topic.options array - Optional key=value;key1=value2 settings for topic consumer. See full configuration options at https://github.com/edenhill/librdkafka/blob/master/CONFIGURATION.md. + Optional key=value;key1=value2 settings for topic consumer. See full configuration options at https://github.com/edenhill/librdkafka/blob/master/CONFIGURATION.md . This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -loggerDisableTimestamps Whether to disable writing timestamps in logs @@ -1125,7 +1125,7 @@ See the docs at https://docs.victoriametrics.com/vmagent.html . -promscrape.suppressScrapeErrorsDelay duration The delay for suppressing repeated scrape errors logging per each scrape targets. This may be used for reducing the number of log lines related to scrape errors. See also -promscrape.suppressScrapeErrors -promscrape.yandexcloudSDCheckInterval duration - Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. (default 30s) + Interval for checking for changes in Yandex Cloud API. This works only if yandexcloud_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#yandexcloud_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. From 3b2599c65952e6e5071a62ab17bf40637545e67b Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 13:00:36 +0300 Subject: [PATCH 04/18] docs/vmalert.md: update `-help` output after explicit marking of enterprise flags --- app/vmalert/README.md | 19 ++++++++++--------- app/vmalert/main.go | 2 +- docs/vmalert.md | 19 ++++++++++--------- 3 files changed, 21 insertions(+), 19 deletions(-) diff --git a/app/vmalert/README.md b/app/vmalert/README.md index 8253c2d71..7d1c4a150 100644 --- a/app/vmalert/README.md +++ b/app/vmalert/README.md @@ -678,7 +678,7 @@ The shortlist of configuration flags is the following: {% raw %} ``` -clusterMode - If clusterMode is enabled, then vmalert automatically adds the tenant specified in config groups to -datasource.url, -remoteWrite.url and -remoteRead.url. See https://docs.victoriametrics.com/vmalert.html#multitenancy + If clusterMode is enabled, then vmalert automatically adds the tenant specified in config groups to -datasource.url, -remoteWrite.url and -remoteRead.url. See https://docs.victoriametrics.com/vmalert.html#multitenancy . This flag is available only in enterprise version of VictoriaMetrics -configCheckInterval duration Interval for checking for changes in '-rule' or '-notifier.config' files. By default the checking is disabled. Send SIGHUP signal in order to force config check for changes. -datasource.appendTypePrefix @@ -732,12 +732,12 @@ The shortlist of configuration flags is the following: -datasource.url string Datasource compatible with Prometheus HTTP API. It can be single node VictoriaMetrics or vmselect URL. Required parameter. E.g. http://127.0.0.1:8428 . See also '-datasource.disablePathAppend', '-datasource.showURL'. -defaultTenant.graphite string - Default tenant for Graphite alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy + Default tenant for Graphite alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy .This flag is available only in enterprise version of VictoriaMetrics -defaultTenant.prometheus string - Default tenant for Prometheus alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy + Default tenant for Prometheus alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy . This flag is available only in enterprise version of VictoriaMetrics -disableAlertgroupLabel Whether to disable adding group's Name as label to generated alerts and time series. - -dryRun -rule + -dryRun Whether to check only config files without running vmalert. The rules file are validated. The -rule flag must be specified. -enableTCP6 Whether to enable IPv6 for listening and dialing. By default only IPv4 TCP and UDP is used @@ -746,12 +746,13 @@ The shortlist of configuration flags is the following: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -evaluationInterval duration How often to evaluate the rules (default 1m0s) -external.alert.source string - External Alert Source allows to override the Source link for alerts sent to AlertManager for cases where you want to build a custom link to Grafana, Prometheus or any other service. - Supports templating. For example, link to Grafana: 'explore?orgId=1&left=[\"now-1h\",\"now\",\"VictoriaMetrics\",{\"expr\": \"{{$expr|quotesEscape|crlfEscape|queryEscape}}\"},{\"mode\":\"Metrics\"},{\"ui\":[true,true,true,\"none\"]}]'. (default "{{.ExternalURL}}/vmalert/alert?group_id={{.GroupID}}&alert_id={{.AlertID}}") + External Alert Source allows to override the Source link for alerts sent to AlertManager for cases where you want to build a custom link to Grafana, Prometheus or any other service. + Supports templating. For example, link to Grafana: 'explore?orgId=1&left=[\"now-1h\",\"now\",\"VictoriaMetrics\",{\"expr\": \"{{$expr|quotesEscape|crlfEscape|queryEscape}}\"},{\"mode\":\"Metrics\"},{\"ui\":[true,true,true,\"none\"]}]'. + If empty 'vmalert/alert?group_id={{.GroupID}}&alert_id={{.AlertID}}' is used. -external.label array Optional label in the form 'Name=value' to add to all generated recording rules and alerts. Pass multiple -label flags in order to add multiple label sets. Supports an array of values separated by comma or specified via multiple flags. @@ -857,13 +858,13 @@ The shortlist of configuration flags is the following: -promscrape.consul.waitTime duration Wait time used by Consul service discovery. Default value is used if not set -promscrape.consulSDCheckInterval duration - Interval for checking for changes in Consul. This works only if consul_sd_configs is configured in '-promscrape.config' file. See https://prometheus.io/docs/prometheus/latest/configuration/configuration/#consul_sd_config for details (default 30s) + Interval for checking for changes in Consul. This works only if consul_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#consul_sd_configs for details (default 30s) -promscrape.discovery.concurrency int The maximum number of concurrent requests to Prometheus autodiscovery API (Consul, Kubernetes, etc.) (default 100) -promscrape.discovery.concurrentWaitTime duration The maximum duration for waiting to perform API requests if more than -promscrape.discovery.concurrency requests are simultaneously performed (default 1m0s) -promscrape.dnsSDCheckInterval duration - Interval for checking for changes in dns. This works only if dns_sd_configs is configured in '-promscrape.config' file. See https://prometheus.io/docs/prometheus/latest/configuration/configuration/#dns_sd_config for details (default 30s) + Interval for checking for changes in dns. This works only if dns_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#dns_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. diff --git a/app/vmalert/main.go b/app/vmalert/main.go index 2b8d7e8f1..01adb9972 100644 --- a/app/vmalert/main.go +++ b/app/vmalert/main.go @@ -71,7 +71,7 @@ If empty 'vmalert/alert?group_id={{.GroupID}}&alert_id={{.AlertID}}' is used.`) disableAlertGroupLabel = flag.Bool("disableAlertgroupLabel", false, "Whether to disable adding group's Name as label to generated alerts and time series.") - dryRun = flag.Bool("dryRun", false, "Whether to check only config files without running vmalert. The rules file are validated. The `-rule` flag must be specified.") + dryRun = flag.Bool("dryRun", false, "Whether to check only config files without running vmalert. The rules file are validated. The -rule flag must be specified.") ) var alertURLGeneratorFn notifier.AlertURLGenerator diff --git a/docs/vmalert.md b/docs/vmalert.md index 6603209e9..2f7e19fc5 100644 --- a/docs/vmalert.md +++ b/docs/vmalert.md @@ -682,7 +682,7 @@ The shortlist of configuration flags is the following: {% raw %} ``` -clusterMode - If clusterMode is enabled, then vmalert automatically adds the tenant specified in config groups to -datasource.url, -remoteWrite.url and -remoteRead.url. See https://docs.victoriametrics.com/vmalert.html#multitenancy + If clusterMode is enabled, then vmalert automatically adds the tenant specified in config groups to -datasource.url, -remoteWrite.url and -remoteRead.url. See https://docs.victoriametrics.com/vmalert.html#multitenancy . This flag is available only in enterprise version of VictoriaMetrics -configCheckInterval duration Interval for checking for changes in '-rule' or '-notifier.config' files. By default the checking is disabled. Send SIGHUP signal in order to force config check for changes. -datasource.appendTypePrefix @@ -736,12 +736,12 @@ The shortlist of configuration flags is the following: -datasource.url string Datasource compatible with Prometheus HTTP API. It can be single node VictoriaMetrics or vmselect URL. Required parameter. E.g. http://127.0.0.1:8428 . See also '-datasource.disablePathAppend', '-datasource.showURL'. -defaultTenant.graphite string - Default tenant for Graphite alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy + Default tenant for Graphite alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy .This flag is available only in enterprise version of VictoriaMetrics -defaultTenant.prometheus string - Default tenant for Prometheus alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy + Default tenant for Prometheus alerting groups. See https://docs.victoriametrics.com/vmalert.html#multitenancy . This flag is available only in enterprise version of VictoriaMetrics -disableAlertgroupLabel Whether to disable adding group's Name as label to generated alerts and time series. - -dryRun -rule + -dryRun Whether to check only config files without running vmalert. The rules file are validated. The -rule flag must be specified. -enableTCP6 Whether to enable IPv6 for listening and dialing. By default only IPv4 TCP and UDP is used @@ -750,12 +750,13 @@ The shortlist of configuration flags is the following: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -evaluationInterval duration How often to evaluate the rules (default 1m0s) -external.alert.source string - External Alert Source allows to override the Source link for alerts sent to AlertManager for cases where you want to build a custom link to Grafana, Prometheus or any other service. - Supports templating. For example, link to Grafana: 'explore?orgId=1&left=[\"now-1h\",\"now\",\"VictoriaMetrics\",{\"expr\": \"{{$expr|quotesEscape|crlfEscape|queryEscape}}\"},{\"mode\":\"Metrics\"},{\"ui\":[true,true,true,\"none\"]}]'. (default "{{.ExternalURL}}/vmalert/alert?group_id={{.GroupID}}&alert_id={{.AlertID}}") + External Alert Source allows to override the Source link for alerts sent to AlertManager for cases where you want to build a custom link to Grafana, Prometheus or any other service. + Supports templating. For example, link to Grafana: 'explore?orgId=1&left=[\"now-1h\",\"now\",\"VictoriaMetrics\",{\"expr\": \"{{$expr|quotesEscape|crlfEscape|queryEscape}}\"},{\"mode\":\"Metrics\"},{\"ui\":[true,true,true,\"none\"]}]'. + If empty 'vmalert/alert?group_id={{.GroupID}}&alert_id={{.AlertID}}' is used. -external.label array Optional label in the form 'Name=value' to add to all generated recording rules and alerts. Pass multiple -label flags in order to add multiple label sets. Supports an array of values separated by comma or specified via multiple flags. @@ -861,13 +862,13 @@ The shortlist of configuration flags is the following: -promscrape.consul.waitTime duration Wait time used by Consul service discovery. Default value is used if not set -promscrape.consulSDCheckInterval duration - Interval for checking for changes in Consul. This works only if consul_sd_configs is configured in '-promscrape.config' file. See https://prometheus.io/docs/prometheus/latest/configuration/configuration/#consul_sd_config for details (default 30s) + Interval for checking for changes in Consul. This works only if consul_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#consul_sd_configs for details (default 30s) -promscrape.discovery.concurrency int The maximum number of concurrent requests to Prometheus autodiscovery API (Consul, Kubernetes, etc.) (default 100) -promscrape.discovery.concurrentWaitTime duration The maximum duration for waiting to perform API requests if more than -promscrape.discovery.concurrency requests are simultaneously performed (default 1m0s) -promscrape.dnsSDCheckInterval duration - Interval for checking for changes in dns. This works only if dns_sd_configs is configured in '-promscrape.config' file. See https://prometheus.io/docs/prometheus/latest/configuration/configuration/#dns_sd_config for details (default 30s) + Interval for checking for changes in dns. This works only if dns_sd_configs is configured in '-promscrape.config' file. See https://docs.victoriametrics.com/sd_configs.html#dns_sd_configs for details (default 30s) -pushmetrics.extraLabel array Optional labels to add to metrics pushed to -pushmetrics.url . For example, -pushmetrics.extraLabel='instance="foo"' adds instance="foo" label to all the metrics pushed to -pushmetrics.url Supports an array of values separated by comma or specified via multiple flags. From 4193af45717abc939b2c8191eafc8c4f241fdbc9 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 13:02:08 +0300 Subject: [PATCH 05/18] docs/vmauth.md: update `-help` output after explicit marking of enterprise flags --- app/vmauth/README.md | 2 +- docs/vmauth.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/app/vmauth/README.md b/app/vmauth/README.md index c23c87032..654578347 100644 --- a/app/vmauth/README.md +++ b/app/vmauth/README.md @@ -239,7 +239,7 @@ See the docs at https://docs.victoriametrics.com/vmauth.html . -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/docs/vmauth.md b/docs/vmauth.md index a36ba220d..ae0d6d835 100644 --- a/docs/vmauth.md +++ b/docs/vmauth.md @@ -243,7 +243,7 @@ See the docs at https://docs.victoriametrics.com/vmauth.html . -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap From e7995375b54cf2f201a3e6c7e3903f628808e70a Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 13:05:50 +0300 Subject: [PATCH 06/18] docs: update `-help` output after explicit mentioning of enterprise flags --- app/vmbackup/README.md | 2 +- app/vmbackupmanager/README.md | 2 +- app/vmgateway/README.md | 8 ++++++-- app/vmrestore/README.md | 2 +- docs/vmbackup.md | 2 +- docs/vmbackupmanager.md | 2 +- docs/vmgateway.md | 8 ++++++-- docs/vmrestore.md | 2 +- 8 files changed, 18 insertions(+), 10 deletions(-) diff --git a/app/vmbackup/README.md b/app/vmbackup/README.md index 8dd3755e1..6eabbe34a 100644 --- a/app/vmbackup/README.md +++ b/app/vmbackup/README.md @@ -188,7 +188,7 @@ See [this article](https://medium.com/@valyala/speeding-up-backups-for-big-time- -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/app/vmbackupmanager/README.md b/app/vmbackupmanager/README.md index eb2b6193a..1bf2005ed 100644 --- a/app/vmbackupmanager/README.md +++ b/app/vmbackupmanager/README.md @@ -182,7 +182,7 @@ vmbackupmanager performs regular backups according to the provided configs. -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/app/vmgateway/README.md b/app/vmgateway/README.md index 8bbbecc98..74fa44f0b 100644 --- a/app/vmgateway/README.md +++ b/app/vmgateway/README.md @@ -192,6 +192,8 @@ The shortlist of configuration flags include the following: Optional path to bearer token file to use for -datasource.url. -datasource.disableKeepAlive Whether to disable long-lived connections to the datasource. If true, disables HTTP keep-alives and will only use the connection to the server for a single HTTP request. + -datasource.headers string + Optional HTTP extraHeaders to send with each request to the corresponding -datasource.url. For example, -datasource.headers='My-Auth:foobar' would send 'My-Auth: foobar' HTTP header with every request to the corresponding -datasource.url. Multiple headers must be delimited by '^^': -datasource.headers='header1:value1^^header2:value2' -datasource.lookback duration Lookback defines how far into the past to look when evaluating queries. For example, if the datasource.lookback=5m then param "time" with value now()-5m will be added to every query. -datasource.maxIdleConnections int @@ -212,6 +214,8 @@ The shortlist of configuration flags include the following: Whether to align "time" parameter with evaluation interval.Alignment supposed to produce deterministic results despite of number of vmalert replicas or time they were started. See more details here https://github.com/VictoriaMetrics/VictoriaMetrics/pull/1257 (default true) -datasource.roundDigits int Adds "round_digits" GET param to datasource requests. In VM "round_digits" limits the number of digits after the decimal point in response values. + -datasource.showURL + Whether to show -datasource.url in the exported metrics. It is hidden by default, since it can contain sensitive info such as auth key -datasource.tlsCAFile string Optional path to TLS CA file to use for verifying connections to -datasource.url. By default, system CA is used -datasource.tlsCertFile string @@ -223,7 +227,7 @@ The shortlist of configuration flags include the following: -datasource.tlsServerName string Optional TLS server name to use for connections to -datasource.url. By default, the server name from -datasource.url is used -datasource.url string - VictoriaMetrics or vmselect url. Required parameter. E.g. http://127.0.0.1:8428 . See also -remoteRead.disablePathAppend + Datasource compatible with Prometheus HTTP API. It can be single node VictoriaMetrics or vmselect URL. Required parameter. E.g. http://127.0.0.1:8428 . See also '-datasource.disablePathAppend', '-datasource.showURL'. -enable.auth enables auth with jwt token -enable.rateLimit @@ -235,7 +239,7 @@ The shortlist of configuration flags include the following: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/app/vmrestore/README.md b/app/vmrestore/README.md index 35bb1521b..4eecdd57d 100644 --- a/app/vmrestore/README.md +++ b/app/vmrestore/README.md @@ -92,7 +92,7 @@ i.e. the end result would be similar to [rsync --delete](https://askubuntu.com/q -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/docs/vmbackup.md b/docs/vmbackup.md index 3d8009f21..cb39e4e47 100644 --- a/docs/vmbackup.md +++ b/docs/vmbackup.md @@ -192,7 +192,7 @@ See [this article](https://medium.com/@valyala/speeding-up-backups-for-big-time- -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/docs/vmbackupmanager.md b/docs/vmbackupmanager.md index 04b92e935..256bef085 100644 --- a/docs/vmbackupmanager.md +++ b/docs/vmbackupmanager.md @@ -186,7 +186,7 @@ vmbackupmanager performs regular backups according to the provided configs. -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/docs/vmgateway.md b/docs/vmgateway.md index 240215037..ea16967c0 100644 --- a/docs/vmgateway.md +++ b/docs/vmgateway.md @@ -196,6 +196,8 @@ The shortlist of configuration flags include the following: Optional path to bearer token file to use for -datasource.url. -datasource.disableKeepAlive Whether to disable long-lived connections to the datasource. If true, disables HTTP keep-alives and will only use the connection to the server for a single HTTP request. + -datasource.headers string + Optional HTTP extraHeaders to send with each request to the corresponding -datasource.url. For example, -datasource.headers='My-Auth:foobar' would send 'My-Auth: foobar' HTTP header with every request to the corresponding -datasource.url. Multiple headers must be delimited by '^^': -datasource.headers='header1:value1^^header2:value2' -datasource.lookback duration Lookback defines how far into the past to look when evaluating queries. For example, if the datasource.lookback=5m then param "time" with value now()-5m will be added to every query. -datasource.maxIdleConnections int @@ -216,6 +218,8 @@ The shortlist of configuration flags include the following: Whether to align "time" parameter with evaluation interval.Alignment supposed to produce deterministic results despite of number of vmalert replicas or time they were started. See more details here https://github.com/VictoriaMetrics/VictoriaMetrics/pull/1257 (default true) -datasource.roundDigits int Adds "round_digits" GET param to datasource requests. In VM "round_digits" limits the number of digits after the decimal point in response values. + -datasource.showURL + Whether to show -datasource.url in the exported metrics. It is hidden by default, since it can contain sensitive info such as auth key -datasource.tlsCAFile string Optional path to TLS CA file to use for verifying connections to -datasource.url. By default, system CA is used -datasource.tlsCertFile string @@ -227,7 +231,7 @@ The shortlist of configuration flags include the following: -datasource.tlsServerName string Optional TLS server name to use for connections to -datasource.url. By default, the server name from -datasource.url is used -datasource.url string - VictoriaMetrics or vmselect url. Required parameter. E.g. http://127.0.0.1:8428 . See also -remoteRead.disablePathAppend + Datasource compatible with Prometheus HTTP API. It can be single node VictoriaMetrics or vmselect URL. Required parameter. E.g. http://127.0.0.1:8428 . See also '-datasource.disablePathAppend', '-datasource.showURL'. -enable.auth enables auth with jwt token -enable.rateLimit @@ -239,7 +243,7 @@ The shortlist of configuration flags include the following: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap diff --git a/docs/vmrestore.md b/docs/vmrestore.md index 8e4866b2b..43816bfd2 100644 --- a/docs/vmrestore.md +++ b/docs/vmrestore.md @@ -96,7 +96,7 @@ i.e. the end result would be similar to [rsync --delete](https://askubuntu.com/q -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap From 455002922e71fcd4fb0b7b5c6d32e8eb45f3f456 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Thu, 15 Sep 2022 13:15:51 +0300 Subject: [PATCH 07/18] docs/Cluster-VictoriaMetrics.md: update `-help` output after explicit marking of enterprise flags --- docs/Cluster-VictoriaMetrics.md | 44 ++++++++++++++++----------------- 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/docs/Cluster-VictoriaMetrics.md b/docs/Cluster-VictoriaMetrics.md index 591e31082..79c666f95 100644 --- a/docs/Cluster-VictoriaMetrics.md +++ b/docs/Cluster-VictoriaMetrics.md @@ -622,15 +622,15 @@ Below is the output for `/path/to/vminsert -help`: ``` -cluster.tls - Whether to use TLS for connections to -storageNode. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Whether to use TLS for connections to -storageNode. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection .This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCAFile string - Path to TLS CA file to use for verifying certificates provided by -storageNode if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to TLS CA file to use for verifying certificates provided by -storageNode if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection .This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCertFile string - Path to client-side TLS certificate file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to client-side TLS certificate file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsInsecureSkipVerify - Whether to skip verification of TLS certificates provided by -storageNode nodes if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security + Whether to skip verification of TLS certificates provided by -storageNode nodes if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security. This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsKeyFile string - Path to client-side TLS key file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to client-side TLS key file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -clusternativeListenAddr string TCP address to listen for data from other vminsert nodes in multi-level cluster setup. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#multi-level-cluster-setup . Usually :8400 must be set. Doesn't work if empty -csvTrimTimestamp duration @@ -651,7 +651,7 @@ Below is the output for `/path/to/vminsert -help`: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap @@ -788,15 +788,15 @@ Below is the output for `/path/to/vmselect -help`: -cacheDataPath string Path to directory for cache files. Cache isn't saved if empty -cluster.tls - Whether to use TLS for connections to -storageNode. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Whether to use TLS for connections to -storageNode. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection .This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCAFile string - Path to TLS CA file to use for verifying certificates provided by -storageNode if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to TLS CA file to use for verifying certificates provided by -storageNode if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection .This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCertFile string - Path to client-side TLS certificate file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to client-side TLS certificate file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsInsecureSkipVerify - Whether to skip verification of TLS certificates provided by -storageNode nodes if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security + Whether to skip verification of TLS certificates provided by -storageNode nodes if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security. This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsKeyFile string - Path to client-side TLS key file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to client-side TLS key file to use when connecting to -storageNode if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -clusternative.disableCompression Whether to disable compression of the data sent to vmselect via -clusternativeListenAddr. This reduces CPU usage at the cost of higher network bandwidth usage -clusternative.maxTagKeys int @@ -834,7 +834,7 @@ Below is the output for `/path/to/vmselect -help`: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -flagsAuthKey string Auth key for /flags endpoint. It must be passed via authKey query arg. It overrides httpAuth.* settings -fs.disableMmap @@ -899,9 +899,9 @@ Below is the output for `/path/to/vmselect -help`: -search.disableCache Whether to disable response caching. This may be useful during data backfilling -search.graphiteMaxPointsPerSeries int - The maximum number of points per series Graphite render API can return (default 1000000) + The maximum number of points per series Graphite render API can return. This flag is available only in enterprise version of VictoriaMetrics (default 1000000) -search.graphiteStorageStep duration - The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API (default 10s) + The interval between datapoints stored in the database. It is used at Graphite Render API handler for normalizing the interval between datapoints in case it isn't normalized. It can be overridden by sending 'storage_step' query arg to /render API or by sending the desired interval via 'Storage-Step' http header during querying /render API. This flag is available only in enterprise version of VictoriaMetrics (default 10s) -search.latencyOffset duration The time when data points become visible in query results after the collection. Too small value can result in incomplete last points for query results (default 30s) -search.logSlowQueryDuration duration @@ -915,7 +915,7 @@ Below is the output for `/path/to/vmselect -help`: -search.maxFederateSeries int The maximum number of time series, which can be returned from /federate. This option allows limiting memory usage (default 1000000) -search.maxGraphiteSeries int - The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage (default 300000) + The maximum number of time series, which can be scanned during queries to Graphite Render API. See https://docs.victoriametrics.com/#graphite-render-api-usage . This flag is available only in enterprise version of VictoriaMetrics (default 300000) -search.maxLookback duration Synonym to -search.lookback-delta from Prometheus. The value is dynamically detected from interval between time series datapoints if not set. It can be overridden on per-query basis via max_lookback arg. See also '-search.maxStalenessInterval' flag, which has the same meaining due to historical reasons -search.maxPointsPerTimeseries int @@ -992,18 +992,18 @@ Below is the output for `/path/to/vmstorage -help`: -bigMergeConcurrency int The maximum number of CPU cores to use for big merges. Default value is used if set to 0 -cluster.tls - Whether to use TLS when accepting connections from vminsert and vmselect. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Whether to use TLS when accepting connections from vminsert and vmselect. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCAFile string - Path to TLS CA file to use for verifying certificates provided by vminsert and vmselect if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to TLS CA file to use for verifying certificates provided by vminsert and vmselect if -cluster.tls flag is set. By default system CA is used. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCertFile string - Path to server-side TLS certificate file to use when accepting connections from vminsert and vmselect if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to server-side TLS certificate file to use when accepting connections from vminsert and vmselect if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsCipherSuites array - Optional list of TLS cipher suites used for connections from vminsert and vmselect if -cluster.tls flag is set. See the list of supported cipher suites at https://pkg.go.dev/crypto/tls#pkg-constants + Optional list of TLS cipher suites used for connections from vminsert and vmselect if -cluster.tls flag is set. See the list of supported cipher suites at https://pkg.go.dev/crypto/tls#pkg-constants .This flag is available only in enterprise version of VictoriaMetrics Supports an array of values separated by comma or specified via multiple flags. -cluster.tlsInsecureSkipVerify - Whether to skip verification of TLS certificates provided by vminsert and vmselect if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security + Whether to skip verification of TLS certificates provided by vminsert and vmselect if -cluster.tls flag is set. Note that disabled TLS certificate verification breaks security. This flag is available only in enterprise version of VictoriaMetrics -cluster.tlsKeyFile string - Path to server-side TLS key file to use when accepting connections from vminsert and vmselect if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection + Path to server-side TLS key file to use when accepting connections from vminsert and vmselect if -cluster.tls flag is set. See https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#mtls-protection . This flag is available only in enterprise version of VictoriaMetrics -dedup.minScrapeInterval duration Leave only the last sample in every time series per each discrete interval equal to -dedup.minScrapeInterval > 0. See https://docs.victoriametrics.com/#deduplication for details -denyQueriesOutsideRetention @@ -1020,7 +1020,7 @@ Below is the output for `/path/to/vmstorage -help`: -envflag.prefix string Prefix for environment variables if -envflag.enable is set -eula - By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf + By specifying this flag, you confirm that you have an enterprise license and accept the EULA https://victoriametrics.com/assets/VM_EULA.pdf . This flag is available only in enterprise version of VictoriaMetrics -finalMergeDelay duration The delay before starting final merge for per-month partition after no new data is ingested into it. Final merge may require additional disk IO and CPU resources. Final merge may increase query speed and reduce disk space usage in some cases. Zero value disables final merge -flagsAuthKey string From 9c95c81534305543af07ab6d4d8ab06a8b8658cf Mon Sep 17 00:00:00 2001 From: Roman Khavronenko Date: Thu, 15 Sep 2022 12:40:22 +0200 Subject: [PATCH 08/18] vmalert: print example of `curl` command for rule's state (#3112) The change adds an example of `curl` command to the Rule's page. The command is generated for each recorded state. It is supposed user can just copy&execute the command to see what was returned to vmalert. Signed-off-by: hagen1778 --- app/vmalert/alerting.go | 15 ++- app/vmalert/datasource/datasource.go | 11 +- app/vmalert/datasource/vm.go | 11 +- app/vmalert/datasource/vm_test.go | 11 +- app/vmalert/helpers_test.go | 11 +- app/vmalert/recording.go | 3 +- app/vmalert/rule.go | 3 + app/vmalert/tpl/header.qtpl | 9 ++ app/vmalert/tpl/header.qtpl.go | 111 +++++++++-------- app/vmalert/utils.go | 62 ++++++++++ app/vmalert/utils_test.go | 47 ++++++++ app/vmalert/web.qtpl | 16 ++- app/vmalert/web.qtpl.go | 172 ++++++++++++++------------- 13 files changed, 324 insertions(+), 158 deletions(-) create mode 100644 app/vmalert/utils_test.go diff --git a/app/vmalert/alerting.go b/app/vmalert/alerting.go index 8c9d98060..c35acdce4 100644 --- a/app/vmalert/alerting.go +++ b/app/vmalert/alerting.go @@ -264,13 +264,14 @@ const resolvedRetention = 15 * time.Minute // Based on the Querier results AlertingRule maintains notifier.Alerts func (ar *AlertingRule) Exec(ctx context.Context, ts time.Time, limit int) ([]prompbmarshal.TimeSeries, error) { start := time.Now() - qMetrics, err := ar.q.Query(ctx, ar.Expr, ts) + qMetrics, req, err := ar.q.Query(ctx, ar.Expr, ts) curState := ruleStateEntry{ time: start, at: ts, duration: time.Since(start), samples: len(qMetrics), err: err, + req: req, } defer func() { @@ -294,7 +295,10 @@ func (ar *AlertingRule) Exec(ctx context.Context, ts time.Time, limit int) ([]pr } } - qFn := func(query string) ([]datasource.Metric, error) { return ar.q.Query(ctx, query, ts) } + qFn := func(query string) ([]datasource.Metric, error) { + res, _, err := ar.q.Query(ctx, query, ts) + return res, err + } updated := make(map[uint64]struct{}) // update list of active alerts for _, m := range qMetrics { @@ -595,7 +599,10 @@ func (ar *AlertingRule) Restore(ctx context.Context, q datasource.Querier, lookb } ts := time.Now() - qFn := func(query string) ([]datasource.Metric, error) { return ar.q.Query(ctx, query, ts) } + qFn := func(query string) ([]datasource.Metric, error) { + res, _, err := ar.q.Query(ctx, query, ts) + return res, err + } // account for external labels in filter var labelsFilter string @@ -605,7 +612,7 @@ func (ar *AlertingRule) Restore(ctx context.Context, q datasource.Querier, lookb expr := fmt.Sprintf("last_over_time(%s{alertname=%q%s}[%ds])", alertForStateMetricName, ar.Name, labelsFilter, int(lookback.Seconds())) - qMetrics, err := q.Query(ctx, expr, ts) + qMetrics, _, err := q.Query(ctx, expr, ts) if err != nil { return err } diff --git a/app/vmalert/datasource/datasource.go b/app/vmalert/datasource/datasource.go index 8f8662125..898061fe7 100644 --- a/app/vmalert/datasource/datasource.go +++ b/app/vmalert/datasource/datasource.go @@ -2,18 +2,27 @@ package datasource import ( "context" + "net/http" "net/url" "time" ) // Querier interface wraps Query and QueryRange methods type Querier interface { - Query(ctx context.Context, query string, ts time.Time) ([]Metric, error) + // Query executes instant request with the given query at the given ts. + // It returns list of Metric in response, the http.Request used for sending query + // and error if any. Returned http.Request can't be reused and its body is already read. + // Query should stop once ctx is cancelled. + Query(ctx context.Context, query string, ts time.Time) ([]Metric, *http.Request, error) + // QueryRange executes range request with the given query on the given time range. + // It returns list of Metric in response and error if any. + // QueryRange should stop once ctx is cancelled. QueryRange(ctx context.Context, query string, from, to time.Time) ([]Metric, error) } // QuerierBuilder builds Querier with given params. type QuerierBuilder interface { + // BuildWithParams creates a new Querier object with the given params BuildWithParams(params QuerierParams) Querier } diff --git a/app/vmalert/datasource/vm.go b/app/vmalert/datasource/vm.go index 331a5827a..83fb5060d 100644 --- a/app/vmalert/datasource/vm.go +++ b/app/vmalert/datasource/vm.go @@ -98,10 +98,10 @@ func NewVMStorage(baseURL string, authCfg *promauth.Config, lookBack time.Durati } // Query executes the given query and returns parsed response -func (s *VMStorage) Query(ctx context.Context, query string, ts time.Time) ([]Metric, error) { +func (s *VMStorage) Query(ctx context.Context, query string, ts time.Time) ([]Metric, *http.Request, error) { req, err := s.newRequestPOST() if err != nil { - return nil, err + return nil, nil, err } switch s.dataSourceType { @@ -110,12 +110,12 @@ func (s *VMStorage) Query(ctx context.Context, query string, ts time.Time) ([]Me case datasourceGraphite: s.setGraphiteReqParams(req, query, ts) default: - return nil, fmt.Errorf("engine not found: %q", s.dataSourceType) + return nil, nil, fmt.Errorf("engine not found: %q", s.dataSourceType) } resp, err := s.do(ctx, req) if err != nil { - return nil, err + return nil, req, err } defer func() { _ = resp.Body.Close() @@ -125,7 +125,8 @@ func (s *VMStorage) Query(ctx context.Context, query string, ts time.Time) ([]Me if s.dataSourceType != datasourcePrometheus { parseFn = parseGraphiteResponse } - return parseFn(req, resp) + result, err := parseFn(req, resp) + return result, req, err } // QueryRange executes the given query on the given time range. diff --git a/app/vmalert/datasource/vm_test.go b/app/vmalert/datasource/vm_test.go index a63e7220d..c01c021e6 100644 --- a/app/vmalert/datasource/vm_test.go +++ b/app/vmalert/datasource/vm_test.go @@ -94,7 +94,7 @@ func TestVMInstantQuery(t *testing.T) { ts := time.Now() expErr := func(err string) { - if _, err := pq.Query(ctx, query, ts); err == nil { + if _, _, err := pq.Query(ctx, query, ts); err == nil { t.Fatalf("expected %q got nil", err) } } @@ -106,7 +106,7 @@ func TestVMInstantQuery(t *testing.T) { expErr("unknown status") // 4 expErr("non-vector resultType error") // 5 - m, err := pq.Query(ctx, query, ts) // 6 - vector + m, _, err := pq.Query(ctx, query, ts) // 6 - vector if err != nil { t.Fatalf("unexpected %s", err) } @@ -129,10 +129,13 @@ func TestVMInstantQuery(t *testing.T) { t.Fatalf("unexpected metric %+v want %+v", m, expected) } - m, err = pq.Query(ctx, query, ts) // 7 - scalar + m, req, err := pq.Query(ctx, query, ts) // 7 - scalar if err != nil { t.Fatalf("unexpected %s", err) } + if req == nil { + t.Fatalf("expected request to be non-nil") + } if len(m) != 1 { t.Fatalf("expected 1 metrics got %d in %+v", len(m), m) } @@ -148,7 +151,7 @@ func TestVMInstantQuery(t *testing.T) { gq := s.BuildWithParams(QuerierParams{DataSourceType: string(datasourceGraphite)}) - m, err = gq.Query(ctx, queryRender, ts) // 8 - graphite + m, _, err = gq.Query(ctx, queryRender, ts) // 8 - graphite if err != nil { t.Fatalf("unexpected %s", err) } diff --git a/app/vmalert/helpers_test.go b/app/vmalert/helpers_test.go index 1a7978cc1..5ee9bb945 100644 --- a/app/vmalert/helpers_test.go +++ b/app/vmalert/helpers_test.go @@ -3,6 +3,7 @@ package main import ( "context" "fmt" + "net/http" "reflect" "sort" "sync" @@ -44,18 +45,20 @@ func (fq *fakeQuerier) BuildWithParams(_ datasource.QuerierParams) datasource.Qu } func (fq *fakeQuerier) QueryRange(ctx context.Context, q string, _, _ time.Time) ([]datasource.Metric, error) { - return fq.Query(ctx, q, time.Now()) + req, _, err := fq.Query(ctx, q, time.Now()) + return req, err } -func (fq *fakeQuerier) Query(_ context.Context, _ string, _ time.Time) ([]datasource.Metric, error) { +func (fq *fakeQuerier) Query(_ context.Context, _ string, _ time.Time) ([]datasource.Metric, *http.Request, error) { fq.Lock() defer fq.Unlock() if fq.err != nil { - return nil, fq.err + return nil, nil, fq.err } cp := make([]datasource.Metric, len(fq.metrics)) copy(cp, fq.metrics) - return cp, nil + req, _ := http.NewRequest(http.MethodPost, "foo.com", nil) + return cp, req, nil } type fakeNotifier struct { diff --git a/app/vmalert/recording.go b/app/vmalert/recording.go index b15059395..f7bfcb09b 100644 --- a/app/vmalert/recording.go +++ b/app/vmalert/recording.go @@ -115,12 +115,13 @@ func (rr *RecordingRule) ExecRange(ctx context.Context, start, end time.Time) ([ // Exec executes RecordingRule expression via the given Querier. func (rr *RecordingRule) Exec(ctx context.Context, ts time.Time, limit int) ([]prompbmarshal.TimeSeries, error) { start := time.Now() - qMetrics, err := rr.q.Query(ctx, rr.Expr, ts) + qMetrics, req, err := rr.q.Query(ctx, rr.Expr, ts) curState := ruleStateEntry{ time: start, at: ts, duration: time.Since(start), samples: len(qMetrics), + req: req, } defer func() { diff --git a/app/vmalert/rule.go b/app/vmalert/rule.go index 55e6f7070..18734e5cf 100644 --- a/app/vmalert/rule.go +++ b/app/vmalert/rule.go @@ -3,6 +3,7 @@ package main import ( "context" "errors" + "net/http" "sync" "time" @@ -53,6 +54,8 @@ type ruleStateEntry struct { // stores the number of samples returned during // the last evaluation samples int + // stores the HTTP request used by datasource during rule.Exec + req *http.Request } const defaultStateEntriesLimit = 20 diff --git a/app/vmalert/tpl/header.qtpl b/app/vmalert/tpl/header.qtpl index acc98573d..42390fa1f 100644 --- a/app/vmalert/tpl/header.qtpl +++ b/app/vmalert/tpl/header.qtpl @@ -59,6 +59,15 @@ background-color: rgba(0,0,0,.5); -webkit-box-shadow: 0 0 1px rgba(255,255,255,.5); } + textarea.curl-area{ + width: 100%; + line-height: 1; + font-size: 12px; + border: none; + margin: 0; + padding: 0; + overflow: scroll; + } diff --git a/app/vmalert/tpl/header.qtpl.go b/app/vmalert/tpl/header.qtpl.go index 2caee4cf8..56c9002a1 100644 --- a/app/vmalert/tpl/header.qtpl.go +++ b/app/vmalert/tpl/header.qtpl.go @@ -101,143 +101,152 @@ func StreamHeader(qw422016 *qt422016.Writer, r *http.Request, navItems []NavItem background-color: rgba(0,0,0,.5); -webkit-box-shadow: 0 0 1px rgba(255,255,255,.5); } + textarea.curl-area{ + width: 100%; + line-height: 1; + font-size: 12px; + border: none; + margin: 0; + padding: 0; + overflow: scroll; + } `) -//line app/vmalert/tpl/header.qtpl:65 +//line app/vmalert/tpl/header.qtpl:74 streamprintNavItems(qw422016, r, title, navItems) -//line app/vmalert/tpl/header.qtpl:65 +//line app/vmalert/tpl/header.qtpl:74 qw422016.N().S(`
`) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 } -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 func WriteHeader(qq422016 qtio422016.Writer, r *http.Request, navItems []NavItem, title string) { -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 StreamHeader(qw422016, r, navItems, title) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 } -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 func Header(r *http.Request, navItems []NavItem, title string) string { -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 WriteHeader(qb422016, r, navItems, title) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 qs422016 := string(qb422016.B) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 return qs422016 -//line app/vmalert/tpl/header.qtpl:67 +//line app/vmalert/tpl/header.qtpl:76 } -//line app/vmalert/tpl/header.qtpl:71 +//line app/vmalert/tpl/header.qtpl:80 type NavItem struct { Name string Url string } -//line app/vmalert/tpl/header.qtpl:77 +//line app/vmalert/tpl/header.qtpl:86 func streamprintNavItems(qw422016 *qt422016.Writer, r *http.Request, current string, items []NavItem) { -//line app/vmalert/tpl/header.qtpl:77 +//line app/vmalert/tpl/header.qtpl:86 qw422016.N().S(` `) -//line app/vmalert/tpl/header.qtpl:79 +//line app/vmalert/tpl/header.qtpl:88 prefix := "/vmalert/" if strings.HasPrefix(r.URL.Path, prefix) { prefix = "" } -//line app/vmalert/tpl/header.qtpl:83 +//line app/vmalert/tpl/header.qtpl:92 qw422016.N().S(` `) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 } -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 func writeprintNavItems(qq422016 qtio422016.Writer, r *http.Request, current string, items []NavItem) { -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 streamprintNavItems(qw422016, r, current, items) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 } -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 func printNavItems(r *http.Request, current string, items []NavItem) string { -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 writeprintNavItems(qb422016, r, current, items) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 qs422016 := string(qb422016.B) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 return qs422016 -//line app/vmalert/tpl/header.qtpl:102 +//line app/vmalert/tpl/header.qtpl:111 } diff --git a/app/vmalert/utils.go b/app/vmalert/utils.go index 7b15e1879..b68708630 100644 --- a/app/vmalert/utils.go +++ b/app/vmalert/utils.go @@ -1,7 +1,10 @@ package main import ( + "fmt" + "net/http" "sort" + "strings" "time" "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" @@ -47,3 +50,62 @@ func newTimeSeriesPB(values []float64, timestamps []int64, labels []prompbmarsha ts.Labels = labels return ts } + +type curlWriter struct { + b strings.Builder +} + +func (cw *curlWriter) string() string { + res := "curl " + cw.b.String() + cw.b.Reset() + return strings.TrimSpace(res) +} + +func (cw *curlWriter) addWithEsc(str string) { + escStr := `'` + strings.Replace(str, `'`, `'\''`, -1) + `'` + cw.add(escStr) +} + +func (cw *curlWriter) add(str string) { + cw.b.WriteString(str) + cw.b.WriteString(" ") +} + +func requestToCurl(req *http.Request) string { + if req.URL == nil { + return "" + } + + cw := &curlWriter{} + + schema := req.URL.Scheme + requestURL := req.URL.String() + if schema == "" { + schema = "http" + if req.TLS != nil { + schema = "https" + } + requestURL = schema + "://" + req.Host + requestURL + } + + if schema == "https" { + cw.add("-k") + } + + cw.add("-X") + cw.add(req.Method) + + var keys []string + for k := range req.Header { + keys = append(keys, k) + } + sort.Strings(keys) + + for _, k := range keys { + cw.add("-H") + cw.addWithEsc(fmt.Sprintf("%s: %s", k, strings.Join(req.Header[k], " "))) + } + + cw.addWithEsc(requestURL) + return cw.string() +} diff --git a/app/vmalert/utils_test.go b/app/vmalert/utils_test.go new file mode 100644 index 000000000..048d08cb6 --- /dev/null +++ b/app/vmalert/utils_test.go @@ -0,0 +1,47 @@ +package main + +import ( + "net/http" + "testing" +) + +func TestRequestToCurl(t *testing.T) { + f := func(req *http.Request, exp string) { + got := requestToCurl(req) + if got != exp { + t.Fatalf("expected to have %q; got %q instead", exp, got) + } + } + + req, _ := http.NewRequest(http.MethodPost, "foo.com", nil) + f(req, "curl -X POST 'http://foo.com'") + + req, _ = http.NewRequest(http.MethodGet, "https://foo.com", nil) + f(req, "curl -k -X GET 'https://foo.com'") + + req, _ = http.NewRequest(http.MethodPost, "foo.com", nil) + req.Header.Set("foo", "bar") + req.Header.Set("baz", "qux") + f(req, "curl -X POST -H 'Baz: qux' -H 'Foo: bar' 'http://foo.com'") + + req, _ = http.NewRequest(http.MethodPost, "foo.com", nil) + params := req.URL.Query() + params.Add("query", "up") + params.Add("step", "10") + req.URL.RawQuery = params.Encode() + f(req, "curl -X POST 'http://foo.com?query=up&step=10'") + + req, _ = http.NewRequest(http.MethodPost, "http://foo.com", nil) + params = req.URL.Query() + params.Add("query", "up") + params.Add("step", "10") + req.URL.RawQuery = params.Encode() + f(req, "curl -X POST 'http://foo.com?query=up&step=10'") + + req, _ = http.NewRequest(http.MethodPost, "https://foo.com", nil) + params = req.URL.Query() + params.Add("query", "up") + params.Add("step", "10") + req.URL.RawQuery = params.Encode() + f(req, "curl -k -X POST 'https://foo.com?query=up&step=10'") +} diff --git a/app/vmalert/web.qtpl b/app/vmalert/web.qtpl index b743a36b5..2d35e5880 100644 --- a/app/vmalert/web.qtpl +++ b/app/vmalert/web.qtpl @@ -435,10 +435,11 @@ - - - - + + + + + @@ -448,14 +449,17 @@ - + + {% if u.err != nil %} - diff --git a/app/vmalert/web.qtpl.go b/app/vmalert/web.qtpl.go index 29f490065..b4648bbbd 100644 --- a/app/vmalert/web.qtpl.go +++ b/app/vmalert/web.qtpl.go @@ -1290,203 +1290,211 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
Updated atSamplesDurationExecuted atUpdated atSamplesDurationExecuted atcURL
{%s u.time.Format(time.RFC3339) %} {%d u.samples %}{%d u.samples %} {%f.3 u.duration.Seconds() %}s {%s u.at.Format(time.RFC3339) %} + +
+ {%v u.err %}
- - - - + + + + + `) -//line app/vmalert/web.qtpl:446 +//line app/vmalert/web.qtpl:447 for _, u := range rule.Updates { -//line app/vmalert/web.qtpl:446 +//line app/vmalert/web.qtpl:447 qw422016.N().S(` - + `) -//line app/vmalert/web.qtpl:456 +//line app/vmalert/web.qtpl:460 if u.err != nil { -//line app/vmalert/web.qtpl:456 +//line app/vmalert/web.qtpl:460 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:462 +//line app/vmalert/web.qtpl:466 } -//line app/vmalert/web.qtpl:462 +//line app/vmalert/web.qtpl:466 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:463 +//line app/vmalert/web.qtpl:467 } -//line app/vmalert/web.qtpl:463 +//line app/vmalert/web.qtpl:467 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:465 +//line app/vmalert/web.qtpl:469 tpl.StreamFooter(qw422016, r) -//line app/vmalert/web.qtpl:465 +//line app/vmalert/web.qtpl:469 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 } -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 func WriteRuleDetails(qq422016 qtio422016.Writer, r *http.Request, rule APIRule) { -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 StreamRuleDetails(qw422016, r, rule) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 } -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 func RuleDetails(r *http.Request, rule APIRule) string { -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 WriteRuleDetails(qb422016, r, rule) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 return qs422016 -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 } -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 func streambadgeState(qw422016 *qt422016.Writer, state string) { -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:472 +//line app/vmalert/web.qtpl:476 badgeClass := "bg-warning text-dark" if state == "firing" { badgeClass = "bg-danger" } -//line app/vmalert/web.qtpl:476 +//line app/vmalert/web.qtpl:480 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:477 +//line app/vmalert/web.qtpl:481 qw422016.E().S(state) -//line app/vmalert/web.qtpl:477 +//line app/vmalert/web.qtpl:481 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 } -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 func writebadgeState(qq422016 qtio422016.Writer, state string) { -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 streambadgeState(qw422016, state) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 } -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 func badgeState(state string) string { -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 writebadgeState(qb422016, state) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 return qs422016 -//line app/vmalert/web.qtpl:478 +//line app/vmalert/web.qtpl:482 } -//line app/vmalert/web.qtpl:480 +//line app/vmalert/web.qtpl:484 func streambadgeRestored(qw422016 *qt422016.Writer) { -//line app/vmalert/web.qtpl:480 +//line app/vmalert/web.qtpl:484 qw422016.N().S(` restored `) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 func writebadgeRestored(qq422016 qtio422016.Writer) { -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 streambadgeRestored(qw422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 func badgeRestored() string { -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 writebadgeRestored(qb422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 return qs422016 -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } From 606166ef6831dd8334e81552cb1b6b1fa4b91745 Mon Sep 17 00:00:00 2001 From: Dmytro Kozlov Date: Thu, 15 Sep 2022 17:02:31 +0300 Subject: [PATCH 09/18] docs/managed_victoriametrics: add how to restore password section (#3116) docs/managed_victoriametrics: add how to restore password section --- docs/managed_victoriametrics/quickstart.md | 41 +++++++++++++++++- .../restore-password-email.png | Bin 0 -> 50231 bytes .../restore-password-profile.png | Bin 0 -> 53997 bytes .../restore-password-save-password.png | Bin 0 -> 114528 bytes .../restore-password.png | Bin 0 -> 98174 bytes 5 files changed, 40 insertions(+), 1 deletion(-) create mode 100644 docs/managed_victoriametrics/restore-password-email.png create mode 100644 docs/managed_victoriametrics/restore-password-profile.png create mode 100644 docs/managed_victoriametrics/restore-password-save-password.png create mode 100644 docs/managed_victoriametrics/restore-password.png diff --git a/docs/managed_victoriametrics/quickstart.md b/docs/managed_victoriametrics/quickstart.md index 84326b377..6dbc02980 100644 --- a/docs/managed_victoriametrics/quickstart.md +++ b/docs/managed_victoriametrics/quickstart.md @@ -11,6 +11,45 @@ To start using the service, one should have already registered AWS account and visit [VictoriaMetrics product page](https://aws.amazon.com/marketplace/pp/prodview-4tbfq5icmbmyc). See more details [here](https://dbaas.victoriametrics.com/howToRegister). +## How to restore password + +If you forgot your password you can restore it the following steps: +1. Click on `Forgot your password?` link at https://dbaas.victoriametrics.com/signIn + +

+ +

+ +2. Enter your email in the field and press button `Send Email` + +

+ +

+ +Follow the instruction sent to the specified email address: +``` +Victoria Metrics Cloud password restore +Follow https://dbaas.victoriametrics.com/login_by_link/{id} the link in order to restore access to Victoria Metrics Cloud. +Access link expires once you login successfully or after 30min. +You can change your password after login https://dbaas.victoriametrics.com/profile profile +Please, ignore this email if you didn't init this action on Victoria Metrics Cloud.
+ +In case of questions contact our support support@victoriametrics.com +``` + +3. To change you password visit Profile page in the top right corner of the web page + +

+ +

+ +4. At the Profile page, enter new password and repeat it in the next field and press `Save` button. + +

+ +

+ + ## Creating instance Instances is a page where user can list and manage VictoriaMetrics single-node instances. @@ -62,4 +101,4 @@ Grafana or any other software. ## Modifying Remember, you always can add, remove or modify existing instances by changing their type or increasing the disk space. -However, such an update requires an instance restart and may result into a couple of minutes of downtime. \ No newline at end of file +However, such an update requires an instance restart and may result into a couple of minutes of downtime. diff --git a/docs/managed_victoriametrics/restore-password-email.png b/docs/managed_victoriametrics/restore-password-email.png new file mode 100644 index 0000000000000000000000000000000000000000..bac4045c9333d6ea8dbced4b800403a0e12d2af9 GIT binary patch literal 50231 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 T;2roI@N>r4{AB(K=%4=sy=1xI literal 0 HcmV?d00001 diff --git a/docs/managed_victoriametrics/restore-password-profile.png b/docs/managed_victoriametrics/restore-password-profile.png new file mode 100644 index 0000000000000000000000000000000000000000..8cbca58ceb33868ca3b6a3d90eb0e447eb389829 GIT binary patch literal 53997 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 literal 0 HcmV?d00001 diff --git a/docs/managed_victoriametrics/restore-password-save-password.png b/docs/managed_victoriametrics/restore-password-save-password.png new file mode 100644 index 0000000000000000000000000000000000000000..3e9b23c082ed4d9bfa86d9dfed5590e47d3d5cd8 GIT binary patch literal 114528 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®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 literal 0 HcmV?d00001 diff --git a/docs/managed_victoriametrics/restore-password.png b/docs/managed_victoriametrics/restore-password.png new file mode 100644 index 0000000000000000000000000000000000000000..70b4fd9565141c448659a77da4fab10e014dded8 GIT binary patch literal 98174 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শ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 Date: Thu, 15 Sep 2022 16:15:39 +0200 Subject: [PATCH 10/18] vmalert: add Troubleshooting section to docs (#3115) Signed-off-by: hagen1778 --- app/vmalert/README.md | 55 ++++++ app/vmalert/vmalert_state.png | Bin 0 -> 111865 bytes app/vmalert/vmalert_ts_data_delay.gif | Bin 0 -> 41876 bytes app/vmalert/vmalert_ts_normal.gif | Bin 0 -> 41884 bytes app/vmalert/web.qtpl | 4 + app/vmalert/web.qtpl.go | 252 ++++++++++++++------------ docs/vmalert.md | 55 ++++++ docs/vmalert_state.png | Bin 0 -> 111865 bytes docs/vmalert_ts_data_delay.gif | Bin 0 -> 41876 bytes docs/vmalert_ts_normal.gif | Bin 0 -> 41884 bytes 10 files changed, 250 insertions(+), 116 deletions(-) create mode 100644 app/vmalert/vmalert_state.png create mode 100644 app/vmalert/vmalert_ts_data_delay.gif create mode 100644 app/vmalert/vmalert_ts_normal.gif create mode 100644 docs/vmalert_state.png create mode 100644 docs/vmalert_ts_data_delay.gif create mode 100644 docs/vmalert_ts_normal.gif diff --git a/app/vmalert/README.md b/app/vmalert/README.md index 7d1c4a150..f268cf959 100644 --- a/app/vmalert/README.md +++ b/app/vmalert/README.md @@ -638,6 +638,61 @@ Use the official [Grafana dashboard](https://grafana.com/grafana/dashboards/1495 If you have suggestions for improvements or have found a bug - please open an issue on github or add a review to the dashboard. +## Troubleshooting + +vmalert executes configured rules within certain intervals. It is expected that at the moment when rule is executed, +the data is already present in configured `-datasource.url`: + +vmalert expected evaluation + +Usually, troubles start to appear when data in `-datasource.url` is delayed or absent. In such cases, evaluations +may get empty response from datasource and produce empty recording rules or reset alerts state: + +vmalert evaluation when data is delayed + +Try the following recommendations in such cases: + +* Always configure group's `evaluationInterval` to be bigger or equal to `scrape_interval` at which metrics +are delivered to the datasource; +* If you know in advance, that data in datasource is delayed - try changing vmalert's `-datasource.lookback` +command-line flag to add a time shift for evaluations; +* If time intervals between datapoints in datasource are irregular - try changing vmalert's `-datasource.queryStep` +command-line flag to specify how far search query can lookback for the recent datapoint. By default, this value +is equal to group's `evaluationInterval`. + +Sometimes, it is not clear why some specific alert fired or didn't fire. It is very important to remember, that +alerts with `for: 0` fire immediately when their expression becomes true. And alerts with `for > 0` will fire only +after multiple consecutive evaluations, and at each evaluation their expression must be true. If at least one evaluation +becomes false, then alert's state resets to the initial state. + +If `-remoteWrite.url` command-line flag is configured, vmalert will persist alert's state in form of time series +`ALERTS` and `ALERTS_FOR_STATE` to the specified destination. Such time series can be then queried via +[vmui](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html#vmui) or Grafana to track how alerts state +changed in time. + +vmalert also stores last N state updates for each rule. To check updates, click on `Details` link next to rule's name +on `/vmalert/groups` page and check the `Last updates` section: + +vmalert state + +Rows in the section represent ordered rule evaluations and their results. The column `curl` contains an example of +HTTP request sent by vmalert to the `-datasource.url` during evaluation. If specific state shows that there were +no samples returned and curl command returns data - then it is very likely there was no data in datasource on the +moment when rule was evaluated. + +vmalert also alows configuring more detailed logging for specific rule. Just set `debug: true` in rule's configuration +and vmalert will start printing additional log messages: +```terminal +2022-09-15T13:35:41.155Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:41+02:00: query returned 0 samples (elapsed: 5.896041ms) +2022-09-15T13:35:56.149Z DEBUG datasource request: executing POST request with params "denyPartialResponse=true&query=sum%28vm_tcplistener_conns%7Binstance%3D%22localhost%3A8429%22%7D%29+by%28instance%29+%3E+0&step=15s&time=1663248945" +2022-09-15T13:35:56.178Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:56+02:00: query returned 1 samples (elapsed: 28.368208ms) +2022-09-15T13:35:56.178Z DEBUG datasource request: executing POST request with params "denyPartialResponse=true&query=sum%28vm_tcplistener_conns%7Binstance%3D%22localhost%3A8429%22%7D%29&step=15s&time=1663248945" +2022-09-15T13:35:56.179Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:56+02:00: alert 10705778000901301787 {alertgroup="TestGroup",alertname="Conns",cluster="east-1",instance="localhost:8429",replica="a"} created in state PENDING +... +2022-09-15T13:36:56.153Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:36:56+02:00: alert 10705778000901301787 {alertgroup="TestGroup",alertname="Conns",cluster="east-1",instance="localhost:8429",replica="a"} PENDING => FIRING: 1m0s since becoming active at 2022-09-15 15:35:56.126006 +0200 CEST m=+39.384575417 +``` + + ## Profiling `vmalert` provides handlers for collecting the following [Go profiles](https://blog.golang.org/profiling-go-programs): diff --git a/app/vmalert/vmalert_state.png b/app/vmalert/vmalert_state.png new file mode 100644 index 0000000000000000000000000000000000000000..5bf656b01eda6ffee8e5a740fdd260c1598ea7ee GIT binary patch literal 111865 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

h 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

p*0H zzIH@~e^4f0!czMMUI!mn`L)X+#-&9#6c%)DhhBeFLw6fB-W8N)q3J+>^en9T!RVU0 zzr|BxZ>>{CjA^s~*-XLnou6DLY@am5id|Z|ew(RE_bmxnQig)t5?9steiOAAb_@EdxSZ ztaBwFow;0hoQ|Q*tDmHo;svsZV5DRU1m-Qm6?-V%V>9;KT0|@^rr+kaPJ`$UM_Pl{ z2~*@#IrOiiFZ+qh>ZUW6GBV>Vza&Etrn#w`umvF_=jOoS8ne)1mTt(!AW?5Eq`E)x zwvU1oR(O`F+?`P#1t}B8fjI^tP7)VS(+`4g0d60=lU}eky*~gGd_UhTV(h*+Qs7i- zBpjLsl+*M+5ALue7~qZYdj`P*jUaDqHQ&w4>3)1;;sCS`(gC?+RjcYhabi$o`x!mpoOF7fI_Ic)s`};OYF~wRGQCV%<5)VMJ$iK-ZDv&%$zyE^Uh1LGqA!wcj z3!h;)tX#JO9rlWn6npk9uj4Z4 zN&q8h??_w-@M07d;Yx!>Y(z@M!4*MG3Mf$NRmyp&w^ee##v-rF*}R9DVor_+&nfn8 z7t4z;UQ8E`x&5yOpxqi_$8Q*iw{)^|3_mtP#JD@-O$XTGAblHEYE#tL(XRPn!(@nU z!9FpgYPp-9b!U`0IHnOI@py`NB$P2)O6jdG@%08>-U@ojvLfTxasHq1&Bong&wjpN)^&nozFD6`rz+n+{>>2*P7W8iFeNHFzc_;Q} z9QMyd1|&%I1LbO(lBz_YBYgcmr<2XSEQM0T?Gtiz81Ms(yu?L3yAssQ;4ra`^(Or! z_!I{N$E>DuXjQQ)BpGA?ij+IUkjhc@C|HIo3#Uv_#$ciK>;4e0Z2_Tn+z&Lbo z;e~tjiOT$9=DSc9vGX^^>$ZZ5Ij#b8o>>kBjgYsNZk(Lh7*M}oswfE|&O9>jvu|+W zm4KTXi#)cNVuLcb2l6H0PT8XO5!1KD+)o&&n9Gn~(NwilOa_Sk!o6xzy6VvFEp32H z<(jXtST{54a%7F)1UH!xjpx^ut6)bTWTiOqrgdq}PEd3HzPL#4HK8l~Gb;Cn0)+;& zo=qxA$*)IPvWoVTL;JIW23S_#Gq5CZ>ND_-`0XpyD)kl}omJD#6T0+3`Q?q{)yc{f zVtv^L<=+AQeSn##(JIh_Yhe2)?A(S~N68>0svM-JvZ+eI4dkR5W9P@~Jq_jQ6HXUM z;1*d4ai^2*6U&9MU-_uu7B|3(I~1OWz;MZ;`V-%9sMGJZ`3kZ#6)~`v9f`A-v5drY z8C<@&~#SJXjTsFE%H^jN`in57Cr~IQsD;ZZJ_<7ZJ*U!uN%`m z!EKFu#)H23wn-x;)*SRblyqkaShcaF>YrXLT?bH?6T@|^To?qj;vx@%mQo*tRDe0`5qFGJlRf?)O zLqK%+>#u7Is*bXO0sjQiqJT94EPu;b!8AV6vI51hixe_Z89{f1(!GSkzzN_Gl(mPR zbhphbeslwiz&MjH_$Ocpq(_luQamvlLKc_kGPFX3Ow^Obvk6iN

MM*|%al;vA0 z1u$XZ>*BG0zD`jbWUJ|KTJ7HFnB{g$3r(M-uCxcn0*fJiF-(u!2p?h@)*w+T}yb zuO4y{pAjW7aB&qzo<&`9-W_0_kC(!^Xf;xbXVv4VJ2CkIc*_K7pF*Uyr`-!WNi*;K zIXQA+Mj-lOTPlJc;HfiRM`ZKhPx(I>u=E%A^< zKg=eAVYk9zVj-4XfSymgd<+=^HKh$ioQvSc@IdAN%xuw2=l1Ptp@6y~{mXA#YM%F; zX=WV}sb20Kn6jASHXBxWZv;vjtkSHymGd4`d_ERfW7CYHWaH#)J4Ao}!X_qdlsSokgY}ica%e@%sqyqyb4&KX7$73wfoVQ?ZG-oR)CeC%j2n4iju_n=r&a{-8g+z2cPJ4 zzJdh9}0!xM`^RqehjO3>R|_gg>pLO7lq; z{m+eCM^5mNc3!G1ou!C(!U+_0=-BmMzl`^#23N|0q$HXNIGEcCLWdOX;jDbJsq!*n zKbP;#mOQ&ntqgZVhXQNTQPGq-gOY8-P=lAqs64>+s_eHQ?rO%7yZL6SSOxKCRGyvf ztKy)1YrFmUAUds0XJ44-kGw73o;?*D{OLCOq%vhDTq!_q$fy5W?67O^b2CN3@JF1u zSd@eucs8s_#$aKS9p}LykMkYI8eoQMi4%(%&kxCcu8rUNu6hKxgh#IoF9!@&22M&l zHa63Q6gjdZlmA52J8drZZ@iF8;9%5boLJw|Aio>ysJj&3DFHFTIM@R_mR<4-pN4y$ zm4H@6(O=E}6iXbdw%h!?ErYG<>`fJhZj-0rS zY+gDBXYrg{9uHT>R~{=w`vtfp2%j*?Lm~BH)aL()J>o7Kjbfwzg9CT{VuQY4IhUnY z~;`uoBSA6I^PQe2`v zIGFl}1rcpoa>R{SVue#Ixjp0MaOMATBieB9J8hc*Z6Vx}YSuc?<10!)f9_b!(~5XO zYB{uRad*EfxcCsP(EL;TnjG~rej9_#n6~D128(!q)0vdF9>+k?y|qK7My6Wkmv7o+ z;R$rGP>+SkrVia}y_#;kL9WIJ+yZEJfxCNUA#(V1V>mdY>M^&Us~#-m$SpZvULRx&LrIv5^&4* z*K_dxYC3<>B$}1k`?{mF|0u_mB_<9jSyHdbC-KLTUSuIs7Z7CH+ydj zd{684*byHlG5k@7eCr-1_ghe>u(bForiUC;4VNCB=DdOlBRKier1D+gymTon_1b>4 zvJfMFrDd~!D=bEupCiOHfA`t-zFf7;p9Zx0P=5Eu&CHV31b!Rou6~gRkg`6&kaCQi+YS{iml($1`jLtaukZmw}#L}p%Rd05hGcZ%V zO42mr7L8b#sd{U9LM2+r#DapPc^^5^xJW<0mv!5XOu+tlm?fSyO_fWnTW>R5CVXI5 zQTwkLSfXLvzpfLKORqo(CD*I|*6lTi{N*5X8cw>>!ce|WsdpCVzq|%%N(T*F)2wU}P`Unu>kZ*aqS<%&?^LQ+ zp_j9Ay6AcT?vazprM*9Q7Qgg&-u{wYDw39app#pAR61px_jvN6!_4Hzfl#fsRlpHI z+Tv9GKKK6RL#23Lvxqs&ETpO#m9u^QyDzugM!dH9l2Fn_vpkawapwd~!v5St+x`UXHZM zj@-*vo7kvk$-3Qv&_>wr62)ehP|c8)jUL6!pQce)W(7|6Fm|fH^GjN7=!-K;UVKF} zR9K-2d@~}k-y>IAcU-bx=NPcmb(uoa3MLh=F+8wahNPlESN4SG;L~MjnTRIBncMtR^T#OwP>z z`4kJ?zM0oP#eDCV)*!L?D4SvstQ*(Hkscm`7a9nhz71=Wq(ZA#FkUfd{U%BCLrr5I}vnJ%^{+~GZ z`u{n7M=&AaYvbrYDf{1p7Svu0XsGseKuBTidf9(xxqlr+6bFxc5C3K|Q1jpGeeUvZ r==qnru76+u&!`2ZZpPJ^{qmRra;V>Q<>bL6Knr6-bAuYB+ui>GNQW~9 literal 0 HcmV?d00001 diff --git a/app/vmalert/vmalert_ts_data_delay.gif b/app/vmalert/vmalert_ts_data_delay.gif new file mode 100644 index 0000000000000000000000000000000000000000..2da024b469b19abe429b781697ec0357aff8e5b4 GIT binary patch literal 41876 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 z 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 z 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 Rs_Eaq_iqCKCh-57z+VY59LoRz literal 0 HcmV?d00001 diff --git a/app/vmalert/web.qtpl b/app/vmalert/web.qtpl index 2d35e5880..7346a2705 100644 --- a/app/vmalert/web.qtpl +++ b/app/vmalert/web.qtpl @@ -384,6 +384,7 @@ + {% if rule.Type == "alerting" %}
@@ -394,6 +395,7 @@
+ {% endif %}
@@ -406,6 +408,7 @@
+ {% if rule.Type == "alerting" %}
@@ -419,6 +422,7 @@
+ {% endif %}
diff --git a/app/vmalert/web.qtpl.go b/app/vmalert/web.qtpl.go index b4648bbbd..08baf5333 100644 --- a/app/vmalert/web.qtpl.go +++ b/app/vmalert/web.qtpl.go @@ -1187,6 +1187,11 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
+ `) +//line app/vmalert/web.qtpl:387 + if rule.Type == "alerting" { +//line app/vmalert/web.qtpl:387 + qw422016.N().S(`
@@ -1194,13 +1199,18 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
`) -//line app/vmalert/web.qtpl:393 - qw422016.E().V(rule.Duration) -//line app/vmalert/web.qtpl:393 - qw422016.N().S(` seconds +//line app/vmalert/web.qtpl:394 + qw422016.E().V(rule.Duration) +//line app/vmalert/web.qtpl:394 + qw422016.N().S(` seconds
+ `) +//line app/vmalert/web.qtpl:398 + } +//line app/vmalert/web.qtpl:398 + qw422016.N().S(`
@@ -1208,27 +1218,32 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
`) -//line app/vmalert/web.qtpl:403 +//line app/vmalert/web.qtpl:405 for _, k := range labelKeys { -//line app/vmalert/web.qtpl:403 +//line app/vmalert/web.qtpl:405 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:404 +//line app/vmalert/web.qtpl:406 qw422016.E().S(k) -//line app/vmalert/web.qtpl:404 +//line app/vmalert/web.qtpl:406 qw422016.N().S(`=`) -//line app/vmalert/web.qtpl:404 +//line app/vmalert/web.qtpl:406 qw422016.E().S(rule.Labels[k]) -//line app/vmalert/web.qtpl:404 +//line app/vmalert/web.qtpl:406 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:405 +//line app/vmalert/web.qtpl:407 } -//line app/vmalert/web.qtpl:405 +//line app/vmalert/web.qtpl:407 qw422016.N().S(`
+ `) +//line app/vmalert/web.qtpl:411 + if rule.Type == "alerting" { +//line app/vmalert/web.qtpl:411 + qw422016.N().S(`
+ `) +//line app/vmalert/web.qtpl:425 + } +//line app/vmalert/web.qtpl:425 + qw422016.N().S(`
@@ -1283,9 +1303,9 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
Last `) -//line app/vmalert/web.qtpl:434 +//line app/vmalert/web.qtpl:438 qw422016.N().D(len(rule.Updates)) -//line app/vmalert/web.qtpl:434 +//line app/vmalert/web.qtpl:438 qw422016.N().S(` updates:
@@ -1236,28 +1251,33 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule)
`) -//line app/vmalert/web.qtpl:415 - for _, k := range annotationKeys { -//line app/vmalert/web.qtpl:415 - qw422016.N().S(` +//line app/vmalert/web.qtpl:418 + for _, k := range annotationKeys { +//line app/vmalert/web.qtpl:418 + qw422016.N().S(` `) -//line app/vmalert/web.qtpl:416 - qw422016.E().S(k) -//line app/vmalert/web.qtpl:416 - qw422016.N().S(`:
+//line app/vmalert/web.qtpl:419 + qw422016.E().S(k) +//line app/vmalert/web.qtpl:419 + qw422016.N().S(`:

`) -//line app/vmalert/web.qtpl:417 - qw422016.E().S(rule.Annotations[k]) -//line app/vmalert/web.qtpl:417 - qw422016.N().S(`

+//line app/vmalert/web.qtpl:420 + qw422016.E().S(rule.Annotations[k]) +//line app/vmalert/web.qtpl:420 + qw422016.N().S(`

`) -//line app/vmalert/web.qtpl:418 - } -//line app/vmalert/web.qtpl:418 - qw422016.N().S(` +//line app/vmalert/web.qtpl:421 + } +//line app/vmalert/web.qtpl:421 + qw422016.N().S(`
Updated atSamplesDurationExecuted atUpdated atSamplesDurationExecuted atcURL
`) -//line app/vmalert/web.qtpl:449 +//line app/vmalert/web.qtpl:450 qw422016.E().S(u.time.Format(time.RFC3339)) -//line app/vmalert/web.qtpl:449 +//line app/vmalert/web.qtpl:450 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:451 + `) +//line app/vmalert/web.qtpl:452 qw422016.N().D(u.samples) -//line app/vmalert/web.qtpl:451 +//line app/vmalert/web.qtpl:452 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:452 +//line app/vmalert/web.qtpl:453 qw422016.N().FPrec(u.duration.Seconds(), 3) -//line app/vmalert/web.qtpl:452 +//line app/vmalert/web.qtpl:453 qw422016.N().S(`s `) -//line app/vmalert/web.qtpl:453 +//line app/vmalert/web.qtpl:454 qw422016.E().S(u.at.Format(time.RFC3339)) -//line app/vmalert/web.qtpl:453 +//line app/vmalert/web.qtpl:454 qw422016.N().S(` + +
`) -//line app/vmalert/web.qtpl:459 +//line app/vmalert/web.qtpl:463 qw422016.E().V(u.err) -//line app/vmalert/web.qtpl:459 +//line app/vmalert/web.qtpl:463 qw422016.N().S(`
@@ -1300,201 +1320,201 @@ func StreamRuleDetails(qw422016 *qt422016.Writer, r *http.Request, rule APIRule) `) -//line app/vmalert/web.qtpl:447 +//line app/vmalert/web.qtpl:451 for _, u := range rule.Updates { -//line app/vmalert/web.qtpl:447 +//line app/vmalert/web.qtpl:451 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:460 +//line app/vmalert/web.qtpl:464 if u.err != nil { -//line app/vmalert/web.qtpl:460 +//line app/vmalert/web.qtpl:464 qw422016.N().S(` - `) -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 } -//line app/vmalert/web.qtpl:466 +//line app/vmalert/web.qtpl:470 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:467 +//line app/vmalert/web.qtpl:471 } -//line app/vmalert/web.qtpl:467 +//line app/vmalert/web.qtpl:471 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:469 +//line app/vmalert/web.qtpl:473 tpl.StreamFooter(qw422016, r) -//line app/vmalert/web.qtpl:469 +//line app/vmalert/web.qtpl:473 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 } -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 func WriteRuleDetails(qq422016 qtio422016.Writer, r *http.Request, rule APIRule) { -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 StreamRuleDetails(qw422016, r, rule) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 } -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 func RuleDetails(r *http.Request, rule APIRule) string { -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 WriteRuleDetails(qb422016, r, rule) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 return qs422016 -//line app/vmalert/web.qtpl:470 +//line app/vmalert/web.qtpl:474 } -//line app/vmalert/web.qtpl:474 +//line app/vmalert/web.qtpl:478 func streambadgeState(qw422016 *qt422016.Writer, state string) { -//line app/vmalert/web.qtpl:474 +//line app/vmalert/web.qtpl:478 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:476 +//line app/vmalert/web.qtpl:480 badgeClass := "bg-warning text-dark" if state == "firing" { badgeClass = "bg-danger" } -//line app/vmalert/web.qtpl:480 +//line app/vmalert/web.qtpl:484 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:481 +//line app/vmalert/web.qtpl:485 qw422016.E().S(state) -//line app/vmalert/web.qtpl:481 +//line app/vmalert/web.qtpl:485 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 func writebadgeState(qq422016 qtio422016.Writer, state string) { -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 streambadgeState(qw422016, state) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 func badgeState(state string) string { -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 writebadgeState(qb422016, state) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 return qs422016 -//line app/vmalert/web.qtpl:482 +//line app/vmalert/web.qtpl:486 } -//line app/vmalert/web.qtpl:484 +//line app/vmalert/web.qtpl:488 func streambadgeRestored(qw422016 *qt422016.Writer) { -//line app/vmalert/web.qtpl:484 +//line app/vmalert/web.qtpl:488 qw422016.N().S(` restored `) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 } -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 func writebadgeRestored(qq422016 qtio422016.Writer) { -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 streambadgeRestored(qw422016) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 qt422016.ReleaseWriter(qw422016) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 } -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 func badgeRestored() string { -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 writebadgeRestored(qb422016) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 qs422016 := string(qb422016.B) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 return qs422016 -//line app/vmalert/web.qtpl:486 +//line app/vmalert/web.qtpl:490 } diff --git a/docs/vmalert.md b/docs/vmalert.md index 2f7e19fc5..8d6f18159 100644 --- a/docs/vmalert.md +++ b/docs/vmalert.md @@ -642,6 +642,61 @@ Use the official [Grafana dashboard](https://grafana.com/grafana/dashboards/1495 If you have suggestions for improvements or have found a bug - please open an issue on github or add a review to the dashboard. +## Troubleshooting + +vmalert executes configured rules within certain intervals. It is expected that at the moment when rule is executed, +the data is already present in configured `-datasource.url`: + +vmalert expected evaluation + +Usually, troubles start to appear when data in `-datasource.url` is delayed or absent. In such cases, evaluations +may get empty response from datasource and produce empty recording rules or reset alerts state: + +vmalert evaluation when data is delayed + +Try the following recommendations in such cases: + +* Always configure group's `evaluationInterval` to be bigger or equal to `scrape_interval` at which metrics +are delivered to the datasource; +* If you know in advance, that data in datasource is delayed - try changing vmalert's `-datasource.lookback` +command-line flag to add a time shift for evaluations; +* If time intervals between datapoints in datasource are irregular - try changing vmalert's `-datasource.queryStep` +command-line flag to specify how far search query can lookback for the recent datapoint. By default, this value +is equal to group's `evaluationInterval`. + +Sometimes, it is not clear why some specific alert fired or didn't fire. It is very important to remember, that +alerts with `for: 0` fire immediately when their expression becomes true. And alerts with `for > 0` will fire only +after multiple consecutive evaluations, and at each evaluation their expression must be true. If at least one evaluation +becomes false, then alert's state resets to the initial state. + +If `-remoteWrite.url` command-line flag is configured, vmalert will persist alert's state in form of time series +`ALERTS` and `ALERTS_FOR_STATE` to the specified destination. Such time series can be then queried via +[vmui](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html#vmui) or Grafana to track how alerts state +changed in time. + +vmalert also stores last N state updates for each rule. To check updates, click on `Details` link next to rule's name +on `/vmalert/groups` page and check the `Last updates` section: + +vmalert state + +Rows in the section represent ordered rule evaluations and their results. The column `curl` contains an example of +HTTP request sent by vmalert to the `-datasource.url` during evaluation. If specific state shows that there were +no samples returned and curl command returns data - then it is very likely there was no data in datasource on the +moment when rule was evaluated. + +vmalert also alows configuring more detailed logging for specific rule. Just set `debug: true` in rule's configuration +and vmalert will start printing additional log messages: +```terminal +2022-09-15T13:35:41.155Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:41+02:00: query returned 0 samples (elapsed: 5.896041ms) +2022-09-15T13:35:56.149Z DEBUG datasource request: executing POST request with params "denyPartialResponse=true&query=sum%28vm_tcplistener_conns%7Binstance%3D%22localhost%3A8429%22%7D%29+by%28instance%29+%3E+0&step=15s&time=1663248945" +2022-09-15T13:35:56.178Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:56+02:00: query returned 1 samples (elapsed: 28.368208ms) +2022-09-15T13:35:56.178Z DEBUG datasource request: executing POST request with params "denyPartialResponse=true&query=sum%28vm_tcplistener_conns%7Binstance%3D%22localhost%3A8429%22%7D%29&step=15s&time=1663248945" +2022-09-15T13:35:56.179Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:35:56+02:00: alert 10705778000901301787 {alertgroup="TestGroup",alertname="Conns",cluster="east-1",instance="localhost:8429",replica="a"} created in state PENDING +... +2022-09-15T13:36:56.153Z DEBUG rule "TestGroup":"Conns" (2601299393013563564) at 2022-09-15T15:36:56+02:00: alert 10705778000901301787 {alertgroup="TestGroup",alertname="Conns",cluster="east-1",instance="localhost:8429",replica="a"} PENDING => FIRING: 1m0s since becoming active at 2022-09-15 15:35:56.126006 +0200 CEST m=+39.384575417 +``` + + ## Profiling `vmalert` provides handlers for collecting the following [Go profiles](https://blog.golang.org/profiling-go-programs): diff --git a/docs/vmalert_state.png b/docs/vmalert_state.png new file mode 100644 index 0000000000000000000000000000000000000000..5bf656b01eda6ffee8e5a740fdd260c1598ea7ee GIT binary patch literal 111865 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

h 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

p*0H zzIH@~e^4f0!czMMUI!mn`L)X+#-&9#6c%)DhhBeFLw6fB-W8N)q3J+>^en9T!RVU0 zzr|BxZ>>{CjA^s~*-XLnou6DLY@am5id|Z|ew(RE_bmxnQig)t5?9steiOAAb_@EdxSZ ztaBwFow;0hoQ|Q*tDmHo;svsZV5DRU1m-Qm6?-V%V>9;KT0|@^rr+kaPJ`$UM_Pl{ z2~*@#IrOiiFZ+qh>ZUW6GBV>Vza&Etrn#w`umvF_=jOoS8ne)1mTt(!AW?5Eq`E)x zwvU1oR(O`F+?`P#1t}B8fjI^tP7)VS(+`4g0d60=lU}eky*~gGd_UhTV(h*+Qs7i- zBpjLsl+*M+5ALue7~qZYdj`P*jUaDqHQ&w4>3)1;;sCS`(gC?+RjcYhabi$o`x!mpoOF7fI_Ic)s`};OYF~wRGQCV%<5)VMJ$iK-ZDv&%$zyE^Uh1LGqA!wcj z3!h;)tX#JO9rlWn6npk9uj4Z4 zN&q8h??_w-@M07d;Yx!>Y(z@M!4*MG3Mf$NRmyp&w^ee##v-rF*}R9DVor_+&nfn8 z7t4z;UQ8E`x&5yOpxqi_$8Q*iw{)^|3_mtP#JD@-O$XTGAblHEYE#tL(XRPn!(@nU z!9FpgYPp-9b!U`0IHnOI@py`NB$P2)O6jdG@%08>-U@ojvLfTxasHq1&Bong&wjpN)^&nozFD6`rz+n+{>>2*P7W8iFeNHFzc_;Q} z9QMyd1|&%I1LbO(lBz_YBYgcmr<2XSEQM0T?Gtiz81Ms(yu?L3yAssQ;4ra`^(Or! z_!I{N$E>DuXjQQ)BpGA?ij+IUkjhc@C|HIo3#Uv_#$ciK>;4e0Z2_Tn+z&Lbo z;e~tjiOT$9=DSc9vGX^^>$ZZ5Ij#b8o>>kBjgYsNZk(Lh7*M}oswfE|&O9>jvu|+W zm4KTXi#)cNVuLcb2l6H0PT8XO5!1KD+)o&&n9Gn~(NwilOa_Sk!o6xzy6VvFEp32H z<(jXtST{54a%7F)1UH!xjpx^ut6)bTWTiOqrgdq}PEd3HzPL#4HK8l~Gb;Cn0)+;& zo=qxA$*)IPvWoVTL;JIW23S_#Gq5CZ>ND_-`0XpyD)kl}omJD#6T0+3`Q?q{)yc{f zVtv^L<=+AQeSn##(JIh_Yhe2)?A(S~N68>0svM-JvZ+eI4dkR5W9P@~Jq_jQ6HXUM z;1*d4ai^2*6U&9MU-_uu7B|3(I~1OWz;MZ;`V-%9sMGJZ`3kZ#6)~`v9f`A-v5drY z8C<@&~#SJXjTsFE%H^jN`in57Cr~IQsD;ZZJ_<7ZJ*U!uN%`m z!EKFu#)H23wn-x;)*SRblyqkaShcaF>YrXLT?bH?6T@|^To?qj;vx@%mQo*tRDe0`5qFGJlRf?)O zLqK%+>#u7Is*bXO0sjQiqJT94EPu;b!8AV6vI51hixe_Z89{f1(!GSkzzN_Gl(mPR zbhphbeslwiz&MjH_$Ocpq(_luQamvlLKc_kGPFX3Ow^Obvk6iN

MM*|%al;vA0 z1u$XZ>*BG0zD`jbWUJ|KTJ7HFnB{g$3r(M-uCxcn0*fJiF-(u!2p?h@)*w+T}yb zuO4y{pAjW7aB&qzo<&`9-W_0_kC(!^Xf;xbXVv4VJ2CkIc*_K7pF*Uyr`-!WNi*;K zIXQA+Mj-lOTPlJc;HfiRM`ZKhPx(I>u=E%A^< zKg=eAVYk9zVj-4XfSymgd<+=^HKh$ioQvSc@IdAN%xuw2=l1Ptp@6y~{mXA#YM%F; zX=WV}sb20Kn6jASHXBxWZv;vjtkSHymGd4`d_ERfW7CYHWaH#)J4Ao}!X_qdlsSokgY}ica%e@%sqyqyb4&KX7$73wfoVQ?ZG-oR)CeC%j2n4iju_n=r&a{-8g+z2cPJ4 zzJdh9}0!xM`^RqehjO3>R|_gg>pLO7lq; z{m+eCM^5mNc3!G1ou!C(!U+_0=-BmMzl`^#23N|0q$HXNIGEcCLWdOX;jDbJsq!*n zKbP;#mOQ&ntqgZVhXQNTQPGq-gOY8-P=lAqs64>+s_eHQ?rO%7yZL6SSOxKCRGyvf ztKy)1YrFmUAUds0XJ44-kGw73o;?*D{OLCOq%vhDTq!_q$fy5W?67O^b2CN3@JF1u zSd@eucs8s_#$aKS9p}LykMkYI8eoQMi4%(%&kxCcu8rUNu6hKxgh#IoF9!@&22M&l zHa63Q6gjdZlmA52J8drZZ@iF8;9%5boLJw|Aio>ysJj&3DFHFTIM@R_mR<4-pN4y$ zm4H@6(O=E}6iXbdw%h!?ErYG<>`fJhZj-0rS zY+gDBXYrg{9uHT>R~{=w`vtfp2%j*?Lm~BH)aL()J>o7Kjbfwzg9CT{VuQY4IhUnY z~;`uoBSA6I^PQe2`v zIGFl}1rcpoa>R{SVue#Ixjp0MaOMATBieB9J8hc*Z6Vx}YSuc?<10!)f9_b!(~5XO zYB{uRad*EfxcCsP(EL;TnjG~rej9_#n6~D128(!q)0vdF9>+k?y|qK7My6Wkmv7o+ z;R$rGP>+SkrVia}y_#;kL9WIJ+yZEJfxCNUA#(V1V>mdY>M^&Us~#-m$SpZvULRx&LrIv5^&4* z*K_dxYC3<>B$}1k`?{mF|0u_mB_<9jSyHdbC-KLTUSuIs7Z7CH+ydj zd{684*byHlG5k@7eCr-1_ghe>u(bForiUC;4VNCB=DdOlBRKier1D+gymTon_1b>4 zvJfMFrDd~!D=bEupCiOHfA`t-zFf7;p9Zx0P=5Eu&CHV31b!Rou6~gRkg`6&kaCQi+YS{iml($1`jLtaukZmw}#L}p%Rd05hGcZ%V zO42mr7L8b#sd{U9LM2+r#DapPc^^5^xJW<0mv!5XOu+tlm?fSyO_fWnTW>R5CVXI5 zQTwkLSfXLvzpfLKORqo(CD*I|*6lTi{N*5X8cw>>!ce|WsdpCVzq|%%N(T*F)2wU}P`Unu>kZ*aqS<%&?^LQ+ zp_j9Ay6AcT?vazprM*9Q7Qgg&-u{wYDw39app#pAR61px_jvN6!_4Hzfl#fsRlpHI z+Tv9GKKK6RL#23Lvxqs&ETpO#m9u^QyDzugM!dH9l2Fn_vpkawapwd~!v5St+x`UXHZM zj@-*vo7kvk$-3Qv&_>wr62)ehP|c8)jUL6!pQce)W(7|6Fm|fH^GjN7=!-K;UVKF} zR9K-2d@~}k-y>IAcU-bx=NPcmb(uoa3MLh=F+8wahNPlESN4SG;L~MjnTRIBncMtR^T#OwP>z z`4kJ?zM0oP#eDCV)*!L?D4SvstQ*(Hkscm`7a9nhz71=Wq(ZA#FkUfd{U%BCLrr5I}vnJ%^{+~GZ z`u{n7M=&AaYvbrYDf{1p7Svu0XsGseKuBTidf9(xxqlr+6bFxc5C3K|Q1jpGeeUvZ r==qnru76+u&!`2ZZpPJ^{qmRra;V>Q<>bL6Knr6-bAuYB+ui>GNQW~9 literal 0 HcmV?d00001 diff --git a/docs/vmalert_ts_data_delay.gif b/docs/vmalert_ts_data_delay.gif new file mode 100644 index 0000000000000000000000000000000000000000..2da024b469b19abe429b781697ec0357aff8e5b4 GIT binary patch literal 41876 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 z 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 z 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 Rs_Eaq_iqCKCh-57z+VY59LoRz literal 0 HcmV?d00001 From 622bbedbe11149ad0a7dbfef075a2c4e2b57e527 Mon Sep 17 00:00:00 2001 From: Roman Khavronenko Date: Fri, 16 Sep 2022 16:19:10 +0200 Subject: [PATCH 11/18] vmalert: always re-evaluate Annotations (#3119) * vmalert: always re-evaluate Annotations Previously, Annotations were evaluated only: 1. On alert creating. 2. On alert's value change. This is premature optimization. It was assumed that since annotations could contain only text with alert's labels or value - there is no need in spending resources to re-compile Annotations. Later, template function `query` was added, which can execute arbitrary queries and return different results on every evaluation. So if it was used in annotations, it would be executed only on init or value change. Another case when optimization caused an issue - annotations hot reload. In this case, annotations of the active alert won't change even if Rule's annotations were changed. This fix enables Annotations re-evaluation on each iteration to resolve issues above. It would have some impact on performance, but it is unlikely it will be noticeable. Signed-off-by: hagen1778 * vmalert: add tp Changelog Signed-off-by: hagen1778 Signed-off-by: hagen1778 --- app/vmalert/alerting.go | 15 +++++---------- docs/CHANGELOG.md | 1 + 2 files changed, 6 insertions(+), 10 deletions(-) diff --git a/app/vmalert/alerting.go b/app/vmalert/alerting.go index c35acdce4..89b35deb3 100644 --- a/app/vmalert/alerting.go +++ b/app/vmalert/alerting.go @@ -324,16 +324,11 @@ func (ar *AlertingRule) Exec(ctx context.Context, ts time.Time, limit int) ([]pr a.ActiveAt = ts ar.logDebugf(ts, a, "INACTIVE => PENDING") } - if a.Value != m.Values[0] { - // update Value field with the latest value - a.Value = m.Values[0] - // and re-exec template since Value can be used - // in annotations - a.Annotations, err = a.ExecTemplate(qFn, ls.origin, ar.Annotations) - if err != nil { - curState.err = err - return nil, curState.err - } + a.Value = m.Values[0] + // re-exec template since Value or query can be used in annotations + a.Annotations, err = a.ExecTemplate(qFn, ls.origin, ar.Annotations) + if err != nil { + return nil, err } continue } diff --git a/docs/CHANGELOG.md b/docs/CHANGELOG.md index 61e248680..8cb11f34a 100644 --- a/docs/CHANGELOG.md +++ b/docs/CHANGELOG.md @@ -31,6 +31,7 @@ The following tip changes can be tested by building VictoriaMetrics components f * BUGFIX: [VictoriaMetrics cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html): properly calculate query results at `vmselect`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3067). The issue has been introduced in [v1.81.0](https://docs.victoriametrics.com/CHANGELOG.html#v1810). * BUGFIX: [VictoriaMetrics cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html): log clear error when multiple identical `-storageNode` command-line flags are passed to `vmselect` or to `vminsert`. Previously these components were crashed with cryptic panic `metric ... is already registered` in this case. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3076). * BUGFIX: [vmui](https://docs.victoriametrics.com/#vmui): fix `RangeError: Maximum call stack size exceeded` error when the query returns too many data points at `Table` view. See [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3092/files). +* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): re-evaluate alert's annotations on each execution. Previously, annotations were evaluated only on alert's value change. See [this PR](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3119) for details. ## [v1.81.2](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.81.2) From b4410b1c630b0ce4e25ab7fdf8307aabcea45df5 Mon Sep 17 00:00:00 2001 From: Roman Khavronenko Date: Fri, 16 Sep 2022 21:24:32 +0200 Subject: [PATCH 12/18] Dashboards (#3120) * dashboards/cluster: few updates * apply consistent formatting across panels; * make resource usage panels per component more detailed; * add extra panels to vmselect for displaying `vm_rows_read_per_query`, `vm_rows_scanned_per_query`, `vm_rows_read_per_series` and `vm_series_read_per_query` metrics. Signed-off-by: hagen1778 * dashboards/single: few updates * apply consistent formatting across panels; * add extra panels to Performance for displaying `vm_rows_read_per_query`, `vm_rows_scanned_per_query`, `vm_rows_read_per_series` and `vm_series_read_per_query` metrics. Signed-off-by: hagen1778 * dashboards/vmagent: few updates * apply consistent formatting across panels; * add panels for showing number of samples ingested or scraped; * adapt resource usage panels for multiple selected jobs/instances; * add adhoc variable; * display vmagent's version in Stats. Signed-off-by: hagen1778 * dashboards/vmalert: few updates * apply consistent formatting across panels; * adapt resource usage panels for multiple selected jobs/instances; * show vmalert version in Stats section. Signed-off-by: hagen1778 --- dashboards/victoriametrics-cluster.json | 2861 ++++++++++++++--------- dashboards/victoriametrics.json | 572 ++++- dashboards/vmagent.json | 736 ++++-- dashboards/vmalert.json | 448 ++-- 4 files changed, 3103 insertions(+), 1514 deletions(-) diff --git a/dashboards/victoriametrics-cluster.json b/dashboards/victoriametrics-cluster.json index 35b1f8a56..a27c591a2 100644 --- a/dashboards/victoriametrics-cluster.json +++ b/dashboards/victoriametrics-cluster.json @@ -1,7 +1,8 @@ { "__inputs": [], "__elements": {}, - "__requires": [{ + "__requires": [ + { "type": "grafana", "id": "grafana", "name": "Grafana", @@ -33,32 +34,35 @@ } ], "annotations": { - "list": [{ - "builtIn": 1, - "datasource": { - "type": "datasource", - "uid": "grafana" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "target": { - "limit": 100, - "matchAny": false, - "tags": [], + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, "type": "dashboard" - }, - "type": "dashboard" - }] + } + ] }, - "description": "Overview for cluster VictoriaMetrics v1.77.0 or higher", + "description": "Overview for cluster VictoriaMetrics v1.79.0 or higher", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 1, "id": null, - "iteration": 1659947950093, - "links": [{ + "iteration": 1663336027743, + "links": [ + { "icon": "doc", "tags": [], "targetBlank": true, @@ -84,7 +88,8 @@ } ], "liveNow": false, - "panels": [{ + "panels": [ + { "collapsed": true, "datasource": { "type": "prometheus", @@ -97,7 +102,8 @@ "y": 0 }, "id": 137, - "panels": [{ + "panels": [ + { "datasource": { "uid": "$ds" }, @@ -110,9 +116,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "short" }, @@ -142,21 +150,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_rows{job=~\"$job_storage\", type!=\"indexdb\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_rows{job=~\"$job_storage\", type!=\"indexdb\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Total datapoints", "type": "stat" }, @@ -173,9 +183,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "bytes" }, @@ -205,21 +217,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "min(vm_free_disk_space_bytes{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "min(vm_free_disk_space_bytes{job=~\"$job\", instance=~\"$instance\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Min free disk space", "type": "stat" }, @@ -237,9 +251,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "bytes" }, @@ -269,21 +285,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", type!=\"indexdb\"}) / sum(vm_rows{job=~\"$job_storage\", type!=\"indexdb\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", type!=\"indexdb\"}) / sum(vm_rows{job=~\"$job_storage\", type!=\"indexdb\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Bytes per point", "type": "stat" }, @@ -300,9 +318,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "bytes" }, @@ -332,21 +352,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_allowed_memory_bytes{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_allowed_memory_bytes{job=~\"$job\", instance=~\"$instance\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Allowed memory", "type": "stat" }, @@ -363,9 +385,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "short" }, @@ -395,21 +419,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_rows{job=~\"$job_storage\", type=\"indexdb\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_rows{job=~\"$job_storage\", type=\"indexdb\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Index size", "type": "stat" }, @@ -427,9 +453,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "bytes" }, @@ -459,21 +487,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": false, - "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": false, + "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Disk space usage", "type": "stat" }, @@ -490,9 +520,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "short" }, @@ -522,21 +554,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_available_cpu_cores{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_available_cpu_cores{job=~\"$job\", instance=~\"$instance\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Available CPU", "type": "stat" }, @@ -554,9 +588,11 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ - "color": "green" - }] + "steps": [ + { + "color": "green" + } + ] }, "unit": "bytes" }, @@ -586,21 +622,23 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_available_memory_bytes{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "instant": true, - "interval": "", - "intervalFactor": 1, - "legendFormat": "", - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_available_memory_bytes{job=~\"$job\", instance=~\"$instance\"})", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], "title": "Available memory", "type": "stat" }, @@ -623,7 +661,8 @@ "mappings": [], "thresholds": { "mode": "absolute", - "steps": [{ + "steps": [ + { "color": "green" }, { @@ -633,25 +672,30 @@ ] } }, - "overrides": [{ + "overrides": [ + { "matcher": { "id": "byName", "options": "Time" }, - "properties": [{ - "id": "custom.hidden", - "value": true - }] + "properties": [ + { + "id": "custom.hidden", + "value": true + } + ] }, { "matcher": { "id": "byName", "options": "Value" }, - "properties": [{ - "id": "displayName", - "value": "Count" - }] + "properties": [ + { + "id": "displayName", + "value": "Count" + } + ] } ] }, @@ -672,20 +716,22 @@ }, "showHeader": true }, - "pluginVersion": "8.5.3", - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(vm_app_version{job=~\"$job\", instance=~\"$instance\"}) by(job, short_version)", - "format": "table", - "instant": true, - "range": false, - "refId": "A" - }], + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(vm_app_version{job=~\"$job\", instance=~\"$instance\"}) by(job, short_version)", + "format": "table", + "instant": true, + "range": false, + "refId": "A" + } + ], "type": "table" }, { @@ -734,7 +780,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -742,17 +788,19 @@ "spaceLength": 10, "stack": false, "steppedLine": true, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sort(sum(up{job=~\"$job\", instance=~\"$instance\"}) by (job, instance))", - "format": "time_series", - "instant": false, - "legendFormat": "{{instance}}({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sort(sum(up{job=~\"$job\", instance=~\"$instance\"}) by (job, instance))", + "format": "time_series", + "instant": false, + "legendFormat": "{{instance}}({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Uptime", @@ -767,7 +815,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:1177", "decimals": 0, "format": "none", @@ -859,17 +908,19 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_rows_inserted_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type,accountID) > 0 ", - "interval": "", - "legendFormat": "{{type}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_rows_inserted_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type,accountID) > 0 ", + "interval": "", + "legendFormat": "{{type}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Datapoints ingestion rate ($instance)", @@ -884,7 +935,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -916,7 +968,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -954,17 +1006,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_http_requests_total{job=~\"$job\", instance=~\"$instance.*\", path!~\"/favicon.ico\"}[$__rate_interval])) by (path) > 0", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{path}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_http_requests_total{job=~\"$job\", instance=~\"$instance.*\", path!~\"/favicon.ico\"}[$__rate_interval])) by (path) > 0", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{path}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Requests rate ($instance)", @@ -979,13 +1033,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:3307", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:3308", "format": "short", "logBase": 1, "min": "0", @@ -1049,17 +1106,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_http_request_errors_total{job=~\"$job\", instance=~\"$instance.*\"}[$__rate_interval])) by (path) > 0", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{path}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_http_request_errors_total{job=~\"$job\", instance=~\"$instance.*\"}[$__rate_interval])) by (path) > 0", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{path}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Requests error rate ($instance)", @@ -1074,7 +1133,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -1144,17 +1204,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "max(vm_request_duration_seconds{job=~\"$job\", instance=~\"$instance\", quantile=~\"(0.5|0.99)\"}) by (path, quantile) > 0", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{quantile}} ({{path}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "max(vm_request_duration_seconds{job=~\"$job\", instance=~\"$instance\", quantile=~\"(0.5|0.99)\"}) by (path, quantile) > 0", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{quantile}} ({{path}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Query duration ($instance)", @@ -1169,13 +1231,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:423", "format": "s", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:424", "format": "short", "logBase": 1, "min": "0", @@ -1231,7 +1296,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 1, "points": false, "renderer": "flot", @@ -1239,7 +1304,8 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -1287,7 +1353,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "rps", "logBase": 1, "min": "0", @@ -1306,10 +1373,11 @@ }, { "aliasColors": {}, - "bars": false, + "bars": true, "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Shows the rate of logging the messages by their level. Unexpected spike in rate is a good reason to check logs.", @@ -1341,7 +1409,7 @@ "total": false, "values": true }, - "lines": true, + "lines": false, "linewidth": 1, "links": [], "nullPointMode": "null", @@ -1349,7 +1417,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1357,20 +1425,24 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_log_messages_total{job=~\"$job\",instance=~\"$instance.*\", level!=\"info\"}[$__rate_interval])) by (job, level) ", - "format": "time_series", - "hide": false, - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{job}} - {{level}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": true, + "expr": "sum(rate(vm_log_messages_total{job=~\"$job\",instance=~\"$instance.*\", level!=\"info\"}[$__rate_interval])) by (job, level) > 0", + "format": "time_series", + "hide": false, + "interval": "5m", + "intervalFactor": 1, + "legendFormat": "{{job}} - {{level}}", + "range": true, + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Logging rate", @@ -1385,13 +1457,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:78", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:79", "format": "short", "logBase": 1, "show": true @@ -1446,7 +1521,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1454,28 +1529,32 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance) /\n(\n sum(vm_free_disk_space_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance) +\n sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance)\n)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], - "thresholds": [{ - "$$hashKey": "object:89", - "colorMode": "critical", - "fill": true, - "line": true, - "op": "gt", - "value": 0.8, - "yaxis": "left" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance) /\n(\n sum(vm_free_disk_space_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance) +\n sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance)\n)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [ + { + "$$hashKey": "object:89", + "colorMode": "critical", + "fill": true, + "line": true, + "op": "gt", + "value": 0.8, + "yaxis": "left" + } + ], "timeRegions": [], "title": "Disk space used ($instance)", "tooltip": { @@ -1489,7 +1568,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "percentunit", "logBase": 1, "min": "0", @@ -1544,17 +1624,19 @@ }, "lines": true, "linewidth": 1, - "links": [{ - "targetBlank": true, - "title": "troubleshooting", - "url": "https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/README.md#troubleshooting" - }], + "links": [ + { + "targetBlank": true, + "title": "troubleshooting", + "url": "https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/README.md#troubleshooting" + } + ], "nullPointMode": "null", "options": { "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1562,17 +1644,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_cache_entries{job=~\"$job\", instance=~\"$instance.*\", type=\"storage/hour_metric_ids\"})", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "Active time series", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(vm_cache_entries{job=~\"$job\", instance=~\"$instance.*\", type=\"storage/hour_metric_ids\"})", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "Active time series", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Active time series ($instance)", @@ -1587,13 +1671,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:253", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:254", "format": "short", "logBase": 1, "min": "0", @@ -1616,7 +1703,8 @@ "y": 34 }, "id": 46, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, @@ -1661,7 +1749,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1669,17 +1757,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(process_resident_memory_bytes{job=~\"$job\", instance=~\"$instance\"}) by (job, instance)", - "interval": "", - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(process_resident_memory_bytes{job=~\"$job\", instance=~\"$instance\"}) by (job, instance)", + "interval": "", + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "RSS memory usage ($instance)", @@ -1694,7 +1784,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "bytes", "logBase": 1, "min": "0", @@ -1756,7 +1847,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1764,17 +1855,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job\", instance=~\"$instance\"}) by (job, instance)", - "interval": "", - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job\", instance=~\"$instance\"}) by (job, instance)", + "interval": "", + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "RSS anonymous memory usage ($instance)", @@ -1789,7 +1882,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:271", "format": "bytes", "logBase": 1, @@ -1851,7 +1945,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1859,19 +1953,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "CPU ($instance)", @@ -1886,7 +1982,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -1948,7 +2045,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1956,28 +2053,32 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": false, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance) / sum(process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], - "thresholds": [{ - "$$hashKey": "object:195", - "colorMode": "critical", - "fill": true, - "line": true, - "op": "gt", - "value": 0.9, - "yaxis": "left" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": false, + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance) / sum(process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], + "thresholds": [ + { + "$$hashKey": "object:195", + "colorMode": "critical", + "fill": true, + "line": true, + "op": "gt", + "value": 0.9, + "yaxis": "left" + } + ], "timeRegions": [], "title": "CPU percentage ($instance)", "tooltip": { @@ -1991,7 +2092,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:75", "format": "percentunit", "logBase": 1, @@ -2054,18 +2156,21 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "/max.*/", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "alias": "/max.*/", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -2106,7 +2211,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "decimals": 0, "format": "short", "logBase": 2, @@ -2168,7 +2274,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2176,19 +2282,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(go_gc_duration_seconds_sum{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)\n/\nsum(rate(go_gc_duration_seconds_count{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 2, - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(go_gc_duration_seconds_sum{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)\n/\nsum(rate(go_gc_duration_seconds_count{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "GC duration ($instance)", @@ -2203,7 +2311,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "s", "logBase": 1, "min": "0", @@ -2263,7 +2372,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2271,19 +2380,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(go_goroutines{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 2, - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(go_goroutines{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Goroutines ($instance)", @@ -2298,7 +2409,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "decimals": 0, "format": "short", "logBase": 1, @@ -2330,7 +2442,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -2360,18 +2472,22 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "/read .*/", - "transform": "negative-Y" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:331", + "alias": "/read .*/", + "transform": "negative-Y" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -2413,12 +2529,15 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:338", "format": "bytes", "logBase": 1, "show": true }, { + "$$hashKey": "object:339", "format": "short", "logBase": 1, "min": "0", @@ -2473,7 +2592,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2481,19 +2600,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(process_num_threads{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 2, - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(process_num_threads{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Threads ($instance)", @@ -2508,7 +2629,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "decimals": 0, "format": "short", "logBase": 1, @@ -2562,7 +2684,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2570,17 +2692,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_tcplistener_conns{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", - "interval": "", - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_tcplistener_conns{job=~\"$job\", instance=~\"$instance\"}) by(job, instance)", + "interval": "", + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "TCP connections ($instance)", @@ -2595,7 +2719,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "show": true @@ -2647,7 +2772,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2655,17 +2780,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_tcplistener_accepts_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", - "interval": "", - "legendFormat": "{{instance}} ({{job}})", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_tcplistener_accepts_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(job, instance)", + "interval": "", + "legendFormat": "{{instance}} ({{job}})", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "TCP connections rate ($instance)", @@ -2680,7 +2807,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "show": true @@ -2711,7 +2839,8 @@ "y": 35 }, "id": 106, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, @@ -2759,14 +2888,17 @@ "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "new series over 24h", - "yaxis": 2 - }], + "seriesOverrides": [ + { + "alias": "new series over 24h", + "yaxis": 2 + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -2804,7 +2936,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -2873,17 +3006,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_indexdb_items_added_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval]))", - "interval": "", - "legendFormat": "items", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_indexdb_items_added_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval]))", + "interval": "", + "legendFormat": "items", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "IndexDB items rate ($instance)", @@ -2898,7 +3033,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:92", "format": "short", "logBase": 1, @@ -2968,25 +3104,29 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_slow_row_inserts_total{job=~\"$job_storage\"}[$__rate_interval])) / sum(rate(vm_rows_inserted_total{job=~\"$job_insert\"}[$__rate_interval]))", - "interval": "", - "legendFormat": "slow inserts", - "refId": "A" - }], - "thresholds": [{ - "$$hashKey": "object:72", - "colorMode": "critical", - "fill": true, - "line": true, - "op": "gt", - "value": 0.1, - "yaxis": "left" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_slow_row_inserts_total{job=~\"$job_storage\"}[$__rate_interval])) / sum(rate(vm_rows_inserted_total{job=~\"$job_insert\"}[$__rate_interval]))", + "interval": "", + "legendFormat": "slow inserts", + "refId": "A" + } + ], + "thresholds": [ + { + "$$hashKey": "object:72", + "colorMode": "critical", + "fill": true, + "line": true, + "op": "gt", + "value": 0.1, + "yaxis": "left" + } + ], "timeRegions": [], "title": "Slow inserts", "tooltip": { @@ -3000,7 +3140,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:738", "format": "percentunit", "logBase": 1, @@ -3070,16 +3211,18 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_slow_queries_total{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval]))", - "interval": "", - "legendFormat": "slow queries rate", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_slow_queries_total{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval]))", + "interval": "", + "legendFormat": "slow queries rate", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Slow queries rate ($instance)", @@ -3094,7 +3237,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:892", "format": "short", "logBase": 1, @@ -3152,11 +3296,13 @@ }, "lines": true, "linewidth": 1, - "links": [{ - "targetBlank": true, - "title": "Readonly mode", - "url": "https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#readonly-mode" - }], + "links": [ + { + "targetBlank": true, + "title": "Readonly mode", + "url": "https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#readonly-mode" + } + ], "nullPointMode": "null", "options": { "alertThreshold": true @@ -3170,19 +3316,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_rpc_vmstorage_is_read_only{job=~\"$job_insert\", instance=~\"$instance\"}) by(instance, addr)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}} => {{addr}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_rpc_vmstorage_is_read_only{job=~\"$job_insert\", instance=~\"$instance\"}) by(instance, addr)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}} => {{addr}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Storage in readonly status for vminsert ($instance)", @@ -3197,7 +3345,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:536", "format": "short", "logBase": 1, @@ -3269,20 +3418,22 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(increase(vm_metrics_with_dropped_labels_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval]))", - "format": "time_series", - "hide": false, - "interval": "", - "intervalFactor": 1, - "legendFormat": "metrics with dropped labels", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(increase(vm_metrics_with_dropped_labels_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval]))", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 1, + "legendFormat": "metrics with dropped labels", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Labels limit exceeded ($instance)", @@ -3297,7 +3448,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:1046", "decimals": 2, "format": "short", @@ -3319,7 +3471,7 @@ }, { "aliasColors": {}, - "bars": false, + "bars": true, "dashLength": 10, "dashes": false, "datasource": { @@ -3355,7 +3507,7 @@ "total": false, "values": true }, - "lines": true, + "lines": false, "linewidth": 1, "links": [], "nullPointMode": "null", @@ -3371,27 +3523,29 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(increase(vm_assisted_merges_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(type, instance) > 0", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "__auto", - "range": true, - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": true, + "expr": "sum(increase(vm_assisted_merges_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(type, instance) > 0", + "format": "time_series", + "interval": "5m", + "intervalFactor": 1, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Assisted merges ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -3400,7 +3554,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:536", "format": "short", "logBase": 1, @@ -3466,17 +3621,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "vm_cache_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"} / vm_cache_size_max_bytes{job=~\"$job\", instance=~\"$instance\"}", - "interval": "", - "legendFormat": "{{ instance }} / {{ type }}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": true, + "expr": "vm_cache_size_bytes{job=~\"$job_storage\", instance=~\"$instance\"} / vm_cache_size_max_bytes{job=~\"$job\", instance=~\"$instance\"}", + "interval": "", + "legendFormat": "{{ instance }} / {{ type }}", + "range": true, + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Cache usage % by vmstorage ($instance)", @@ -3491,7 +3650,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:107", "format": "percentunit", "logBase": 1, @@ -3524,7 +3684,8 @@ "y": 36 }, "id": 48, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, @@ -3577,7 +3738,8 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -3614,7 +3776,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:1108", "format": "short", "logBase": 1, @@ -3685,7 +3848,8 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -3733,7 +3897,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -3755,6 +3920,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "The number of rows rerouted to the vmstorage node from other nodes when they were unhealthy.", @@ -3802,19 +3968,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_rpc_rows_rerouted_to_here_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(addr)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{addr}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_rpc_rows_rerouted_to_here_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(addr) > 0", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{addr}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Rows ($instance) rerouted to ", @@ -3829,13 +3997,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:701", "format": "rps", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:702", "format": "short", "logBase": 1, "show": true @@ -3851,6 +4022,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "The number of rows rerouted from the vmstorage node to healthy nodes when the given node was unhealthy.", @@ -3898,19 +4070,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_rpc_rows_rerouted_from_here_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(addr)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{addr}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_rpc_rows_rerouted_from_here_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(addr) > 0", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{addr}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Rows ($instance) rerouted from", @@ -3925,13 +4099,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:761", "format": "rps", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:762", "format": "short", "logBase": 1, "min": "0", @@ -3957,7 +4134,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -3986,18 +4163,21 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "bytes", - "yaxis": 2 - }], + "seriesOverrides": [ + { + "alias": "bytes", + "yaxis": 2 + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -4030,13 +4210,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:821", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:822", "format": "bytes", "logBase": 1, "min": "0", @@ -4091,7 +4274,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4099,15 +4282,17 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_tcpdialer_written_bytes_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) * 8", - "legendFormat": "network usage", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_tcpdialer_written_bytes_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) * 8", + "legendFormat": "network usage", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "RPC network usage ($instance)", @@ -4122,7 +4307,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "bps", "logBase": 1, "min": "0", @@ -4154,12 +4340,14 @@ "y": 37 }, "id": 60, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "VictoriaMetrics stores various caches in RAM. Memory size for these caches may be limited with -`memory.allowedPercent` flag. Line `max allowed` shows max allowed memory size for cache.", @@ -4169,7 +4357,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -4204,16 +4392,19 @@ "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "max allowed", - "color": "#C4162A", - "fill": 0, - "stack": false - }], + "seriesOverrides": [ + { + "alias": "max allowed", + "color": "#C4162A", + "fill": 0, + "stack": false + } + ], "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -4252,13 +4443,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:889", "format": "bytes", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:890", "format": "short", "logBase": 1, "min": "0", @@ -4317,19 +4511,21 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "editorMode": "code", - "exemplar": true, - "expr": "sum(vm_cache_size_bytes{job=~\"$job\", instance=~\"$instance\"}) by(type) / \nsum(vm_cache_size_max_bytes{job=~\"$job\", instance=~\"$instance\"}) by(type)", - "interval": "", - "legendFormat": "{{ type }}", - "range": true, - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": true, + "expr": "sum(vm_cache_size_bytes{job=~\"$job\", instance=~\"$instance\"}) by(type) / \nsum(vm_cache_size_max_bytes{job=~\"$job\", instance=~\"$instance\"}) by(type)", + "interval": "", + "legendFormat": "{{ type }}", + "range": true, + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Cache usage % ($instance)", @@ -4344,7 +4540,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:107", "format": "percentunit", "logBase": 1, @@ -4415,20 +4612,22 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "1 - (\n sum(rate(vm_cache_misses_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type) /\n sum(rate(vm_cache_requests_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type)\n)", - "format": "time_series", - "hide": false, - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{type}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "1 - (\n sum(rate(vm_cache_misses_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type) /\n sum(rate(vm_cache_requests_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (type)\n)", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{type}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Cache hit ratio ($instance)", @@ -4443,7 +4642,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "percentunit", "logBase": 1, "max": "1", @@ -4477,7 +4677,8 @@ "y": 38 }, "id": 24, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, @@ -4522,7 +4723,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4530,17 +4731,19 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_vminsert_metrics_read_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_vminsert_metrics_read_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Ingestion rate ($instance)", @@ -4555,7 +4758,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "short", "logBase": 1, "min": "0", @@ -4617,7 +4821,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4625,18 +4829,20 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "vm_free_disk_space_bytes{job=~\"$job_storage\", instance=~\"$instance\"} / ignoring(path) ((rate(vm_rows_added_to_storage_total{job=~\"$job_storage\", instance=~\"$instance\"}[1d]) - ignoring(type) rate(vm_deduplicated_samples_total{job=~\"$job_storage\", instance=~\"$instance\", type=\"merge\"}[1d])) * scalar(sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) / sum(vm_rows{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"})))", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "vm_free_disk_space_bytes{job=~\"$job_storage\", instance=~\"$instance\"} / ignoring(path) ((rate(vm_rows_added_to_storage_total{job=~\"$job_storage\", instance=~\"$instance\"}[1d]) - ignoring(type) rate(vm_deduplicated_samples_total{job=~\"$job_storage\", instance=~\"$instance\", type=\"merge\"}[1d])) * scalar(sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) / sum(vm_rows{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"})))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Storage full ETA ($instance)", @@ -4651,7 +4857,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:87", "format": "s", "logBase": 1, @@ -4714,7 +4921,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4722,18 +4929,20 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_rows{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) by(instance)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(vm_rows{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) by(instance)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Datapoints ($instance)", @@ -4748,13 +4957,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1330", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1331", "format": "short", "logBase": 1, "min": "0", @@ -4780,7 +4992,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -4811,18 +5023,21 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "pending index entries", - "yaxis": 2 - }], + "seriesOverrides": [ + { + "alias": "pending index entries", + "yaxis": 2 + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -4861,13 +5076,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1262", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1263", "decimals": 3, "format": "none", "logBase": 1, @@ -4924,7 +5142,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4932,17 +5150,19 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) by(instance)", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type!=\"indexdb\"}) by(instance)", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Disk space usage (datapoints) ($instance)", @@ -4957,7 +5177,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "bytes", "logBase": 1, "min": "0", @@ -5019,7 +5240,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5027,17 +5248,19 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type=\"indexdb\"}) by(instance)", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(vm_data_size_bytes{job=~\"$job_storage\", instance=~\"$instance\", type=\"indexdb\"}) by(instance)", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Disk space usage (index) ($instance)", @@ -5052,7 +5275,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "bytes", "logBase": 1, "min": "0", @@ -5084,7 +5308,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -5113,7 +5337,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5121,21 +5345,23 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_active_merges{job=~\"$job_storage\", instance=~\"$instance\"}) by(type)", - "legendFormat": "{{type}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(vm_active_merges{job=~\"$job_storage\", instance=~\"$instance\"}) by(type)", + "legendFormat": "{{type}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Active merges ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -5144,7 +5370,9 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1399", "decimals": 0, "format": "short", "logBase": 1, @@ -5152,6 +5380,7 @@ "show": true }, { + "$$hashKey": "object:1400", "format": "short", "logBase": 1, "min": "0", @@ -5177,7 +5406,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -5206,7 +5435,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5214,21 +5443,23 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_rows_merged_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(type)", - "legendFormat": "{{type}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_rows_merged_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(type)", + "legendFormat": "{{type}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Merge speed", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -5237,7 +5468,9 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1516", "decimals": 0, "format": "short", "logBase": 1, @@ -5245,6 +5478,7 @@ "show": true }, { + "$$hashKey": "object:1517", "format": "short", "logBase": 1, "min": "0", @@ -5264,7 +5498,7 @@ "uid": "$ds" }, "description": "Shows how many rows were ignored on insertion due to corrupted or out of retention timestamps.", - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -5293,7 +5527,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5301,17 +5535,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(vm_rows_ignored_total{job=~\"$job_storage\", instance=~\"$instance\"}) by (reason)", - "interval": "", - "legendFormat": "{{reason}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(vm_rows_ignored_total{job=~\"$job_storage\", instance=~\"$instance\"}) by (reason)", + "interval": "", + "legendFormat": "{{reason}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Rows ignored ($instance)", @@ -5326,12 +5562,15 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1584", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:1585", "format": "short", "logBase": 1, "show": true @@ -5347,16 +5586,17 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, - "description": "Data parts of LSM tree.\nHigh number of parts could be an evidence of slow merge performance - check the resource utilization.\n* `indexdb` - inverted index\n* `storage/small` - recently added parts of data ingested into storage(hot data)\n* `storage/big` - small parts gradually merged into big parts (cold data)", + "description": "The max number of data parts of LSM tree across all storage nodes.\nHigh number of parts (the hard limit is 512) is an evidence of slow merge performance - check the resource utilization.\n* `indexdb` - inverted index\n* `storage/small` - recently added parts of data ingested into storage (hot data)\n* `storage/big` - small parts gradually merged into big parts (cold data)", "fieldConfig": { "defaults": { "links": [] }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -5386,7 +5626,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5394,18 +5634,20 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(vm_parts{job=~\"$job_storage\", instance=~\"$instance\"}) by (type)", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{type}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "max(sum(vm_parts{job=~\"$job_storage\", instance=~\"$instance\"}) by (type, instance)) by(type)", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{type}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "LSM parts ($instance)", @@ -5420,13 +5662,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1644", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1645", "format": "short", "logBase": 1, "min": "0", @@ -5468,17 +5713,19 @@ }, "lines": true, "linewidth": 1, - "links": [{ - "targetBlank": true, - "title": "Readonly mode", - "url": "https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#readonly-mode" - }], + "links": [ + { + "targetBlank": true, + "title": "Readonly mode", + "url": "https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#readonly-mode" + } + ], "nullPointMode": "null", "options": { "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5486,17 +5733,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "vm_storage_is_read_only{job=~\"$job_storage\", instance=~\"$instance\"}", - "interval": "", - "legendFormat": "{{ instance }}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "vm_storage_is_read_only{job=~\"$job_storage\", instance=~\"$instance\"}", + "interval": "", + "legendFormat": "{{ instance }}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Readonly mode", @@ -5511,7 +5760,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:131", "format": "short", "logBase": 1, @@ -5537,7 +5787,7 @@ "type": "prometheus", "uid": "$ds" }, - "description": "Shows how many ongoing insertions (not API /write calls) on disk are taking place, where:\n* `max` - equal to number of CPUs;\n* `current` - current number of goroutines busy with inserting rows into underlying storage.\n\nEvery successful API /write call results into flush on disk. The `max` is an internal limit and can't be changed. It is always equal to the number of CPUs. \n\nWhen `current` hits `max` constantly, it means storage is overloaded and requires more CPU.", + "description": "Shows how many ongoing insertions (not API /write calls) on disk are taking place, where:\n* `max` - equal to number of CPUs;\n* `current` - current number of goroutines busy with inserting rows into underlying storage.\n\nEvery successful API /write call results into flush on disk. The `max` is an internal limit and can't be changed. It is always equal to the number of CPUs. \n\nWhen `current` hits `max` constantly, it means storage is overloaded and requires more CPU or faster disk.", "fill": 0, "fillGradient": 0, "gridPos": { @@ -5562,28 +5812,33 @@ }, "lines": true, "linewidth": 1, - "links": [{ - "targetBlank": true, - "title": "Related discussion", - "url": "https://github.com/VictoriaMetrics/VictoriaMetrics/issues/632" - }], + "links": [ + { + "targetBlank": true, + "title": "Related discussion", + "url": "https://github.com/VictoriaMetrics/VictoriaMetrics/issues/632" + } + ], "nullPointMode": "null", "options": { "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "alias": "max", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -5621,12 +5876,15 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1768", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:1769", "format": "short", "logBase": 1, "show": true @@ -5645,6 +5903,7 @@ "type": "prometheus", "uid": "$ds" }, + "description": "", "fieldConfig": { "defaults": { "links": [] @@ -5681,30 +5940,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval]))", + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_storage\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "cores used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -5714,9 +5976,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(process_cpu_cores_available{job=~\"$job_storage\", instance=~\"$instance\"})", + "expr": "min(process_cpu_cores_available{job=~\"$job_storage\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -5726,7 +5988,7 @@ "title": "CPU ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -5735,13 +5997,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:1827", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1828", "format": "short", "logBase": 1, "show": true @@ -5796,30 +6061,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) ", + "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_storage\", instance=~\"$instance\"}) by(instance) ", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "mem used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -5829,9 +6097,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(vm_available_memory_bytes{job=~\"$job_storage\", instance=~\"$instance\"})", + "expr": "min(vm_available_memory_bytes{job=~\"$job_storage\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -5850,7 +6118,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:164", "format": "bytes", "logBase": 1, @@ -5884,12 +6153,14 @@ "y": 39 }, "id": 42, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Request rate accepted by vmselect nodes", @@ -5899,7 +6170,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -5929,7 +6200,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -5937,17 +6208,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_http_requests_total{job=~\"$job_select\", instance=~\"$instance.*\", path!~\"/favicon.ico\"}[$__rate_interval])) by (path) > 0", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{path}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_http_requests_total{job=~\"$job_select\", instance=~\"$instance.*\", path!~\"/favicon.ico|/metrics\"}[$__rate_interval])) by (path) > 0", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{path}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Requests rate ($instance)", @@ -5962,13 +6235,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2088", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:2089", "format": "short", "logBase": 1, "min": "0", @@ -5985,16 +6261,17 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, - "description": "Shows how many ongoing insertions are taking place.\n* `max` - equal to number of CPU * 2 by default. May be configured with `search.maxConcurrentRequests` flag\n* `current` - current number of goroutines busy with processing requests\n\nWhen `current` hits `max` constantly, it means vmselect node is overloaded and require more CPU or higher limits.", + "description": "Shows the max number of concurrent selects across instances.\n* `max` - equal to number of CPU * 2 by default. May be configured with `search.maxConcurrentRequests` flag\n* `current` - current number of goroutines busy with processing requests\n\nWhen `current` hits `max` constantly, it means one or more vmselect nodes are overloaded and require more CPU or better load balancing. If CPU panel shows there are free resources - try increasing `search.maxConcurrentRequests`.", "fieldConfig": { "defaults": { "links": [] }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -6026,24 +6303,28 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "max", - "color": "#C4162A", - "fill": 0 - }], + "seriesOverrides": [ + { + "$$hashKey": "object:2150", + "alias": "max", + "color": "#C4162A", + "fill": 0 + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, - "expr": "sum(max_over_time(vm_concurrent_select_current{job=~\"$job_select\", instance=~\"$instance\"}[1m]))", + "expr": "max(max_over_time(vm_concurrent_select_current{job=~\"$job_select\", instance=~\"$instance\"}[1m])) ", "format": "time_series", "interval": "", "intervalFactor": 1, @@ -6055,7 +6336,7 @@ "type": "prometheus", "uid": "$ds" }, - "expr": "sum(vm_concurrent_select_capacity{job=~\"$job_select\", instance=~\"$instance\"})", + "expr": "min(vm_concurrent_select_capacity{job=~\"$job_select\", instance=~\"$instance\"})", "format": "time_series", "intervalFactor": 1, "legendFormat": "max", @@ -6076,7 +6357,9 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2159", "decimals": 0, "format": "short", "logBase": 1, @@ -6084,6 +6367,7 @@ "show": true }, { + "$$hashKey": "object:2160", "decimals": 0, "format": "short", "logBase": 1, @@ -6095,6 +6379,418 @@ "align": false } }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of series read per query.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 16 + }, + "hiddenSeries": false, + "id": 178, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_series_read_per_query_bucket{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Series read per query ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples read per query.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 16 + }, + "hiddenSeries": false, + "id": 179, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_read_per_query_bucket{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows read per query ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples read per queried series.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 24 + }, + "hiddenSeries": false, + "id": 180, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_read_per_series_bucket{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows read per series ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples scanner per query.\n\nThis number can exceed number of RowsReadPerQuery if `step` query arg passed to [/api/v1/query_range](https://prometheus.io/docs/prometheus/latest/querying/api/#range-queries) is smaller than the lookbehind window set in square brackets of [rollup function](https://docs.victoriametrics.com/MetricsQL.html#rollup-functions). For example, if `increase(some_metric[1h])` is executed with the `step=5m`, then the same raw samples on a hour time range are scanned `1h/5m=12` times. See [this article](https://valyala.medium.com/how-to-optimize-promql-and-metricsql-queries-85a1b75bf986) for details.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 24 + }, + "hiddenSeries": false, + "id": 181, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_scanned_per_query_bucket{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows scanned per series ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, { "aliasColors": {}, "bars": false, @@ -6110,13 +6806,13 @@ }, "overrides": [] }, - "fill": 6, + "fill": 0, "fillGradient": 0, "gridPos": { - "h": 7, + "h": 8, "w": 24, "x": 0, - "y": 16 + "y": 32 }, "hiddenSeries": false, "id": 93, @@ -6140,18 +6836,22 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "/read.*/", - "transform": "negative-Y" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:2229", + "alias": "/read.*/", + "transform": "negative-Y" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" @@ -6189,12 +6889,15 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2236", "format": "bps", "logBase": 1, "show": true }, { + "$$hashKey": "object:2237", "format": "short", "logBase": 1, "show": true @@ -6225,7 +6928,7 @@ "h": 8, "w": 12, "x": 0, - "y": 23 + "y": 40 }, "hiddenSeries": false, "id": 163, @@ -6249,30 +6952,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval]))", + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_select\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "cores used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -6282,9 +6988,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(process_cpu_cores_available{job=~\"$job_select\", instance=~\"$instance\"})", + "expr": "min(process_cpu_cores_available{job=~\"$job_select\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -6303,13 +7009,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2303", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:2304", "format": "short", "logBase": 1, "show": true @@ -6340,7 +7049,7 @@ "h": 8, "w": 12, "x": 12, - "y": 23 + "y": 40 }, "hiddenSeries": false, "id": 165, @@ -6364,30 +7073,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_select\", instance=~\"$instance\"}) ", + "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_select\", instance=~\"$instance\"}) by(instance) ", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "mem used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -6397,9 +7109,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(vm_available_memory_bytes{job=~\"$job_select\", instance=~\"$instance\"})", + "expr": "min(vm_available_memory_bytes{job=~\"$job_select\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -6418,7 +7130,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:164", "format": "bytes", "logBase": 1, @@ -6452,7 +7165,8 @@ "y": 40 }, "id": 40, - "panels": [{ + "panels": [ + { "aliasColors": {}, "bars": false, "dashLength": 10, @@ -6467,13 +7181,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 160 + "y": 41 }, "hiddenSeries": false, "id": 97, @@ -6497,7 +7211,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -6505,17 +7219,19 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "sum(rate(vm_http_requests_total{job=~\"$job_insert\", instance=~\"$instance.*\", path!~\"/favicon.ico\"}[$__rate_interval])) by (path) > 0", - "format": "time_series", - "intervalFactor": 1, - "legendFormat": "{{path}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "sum(rate(vm_http_requests_total{job=~\"$job_insert\", instance=~\"$instance.*\", path!~\"/favicon.ico\"}[$__rate_interval])) by (path) > 0", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "{{path}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Requests rate ($instance)", @@ -6530,13 +7246,16 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2715", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:2716", "format": "short", "logBase": 1, "min": "0", @@ -6553,6 +7272,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Shows how many ongoing insertions are taking place.\n* `max` - equal to number of CPU * 4 by default. May be configured with `maxConcurrentInserts` flag;\n* `current` - current number of goroutines busy with processing requests.\n\n`-maxConcurrentInserts` limits the number of insert requests which may be actively processed at any given point in time. All the other insert requests are queued for up to `-insert.maxQueueDuration` in the hope they will get a chance to be processed. This queue is used mostly for absorbing spikes for incoming insert request rate.\n\nWhen `current` hits `max` constantly, it means vminsert node is overloaded and requires more CPU or higher limits.", @@ -6568,7 +7288,7 @@ "h": 8, "w": 12, "x": 12, - "y": 160 + "y": 41 }, "hiddenSeries": false, "id": 99, @@ -6594,28 +7314,34 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "alias": "max", - "color": "#C4162A", - "fill": 0 - }], + "seriesOverrides": [ + { + "$$hashKey": "object:2777", + "alias": "max", + "color": "#C4162A", + "fill": 0 + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, - "expr": "sum(vm_concurrent_insert_current{job=~\"$job_insert\", instance=~\"$instance\"})", + "editorMode": "code", + "expr": "max(vm_concurrent_insert_current{job=~\"$job_insert\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "current", + "legendFormat": "{{instance}}", + "range": true, "refId": "A" }, { @@ -6624,7 +7350,7 @@ "uid": "$ds" }, "exemplar": true, - "expr": "sum(vm_concurrent_insert_capacity{job=~\"$job_insert\", instance=~\"$instance\"})", + "expr": "min(vm_concurrent_insert_capacity{job=~\"$job_insert\", instance=~\"$instance\"})", "format": "time_series", "interval": "", "intervalFactor": 1, @@ -6646,7 +7372,9 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2786", "decimals": 0, "format": "short", "logBase": 1, @@ -6654,6 +7382,7 @@ "show": true }, { + "$$hashKey": "object:2787", "decimals": 0, "format": "short", "logBase": 1, @@ -6687,7 +7416,7 @@ "h": 8, "w": 12, "x": 0, - "y": 168 + "y": 49 }, "hiddenSeries": false, "id": 90, @@ -6711,7 +7440,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -6719,19 +7448,21 @@ "spaceLength": 10, "stack": true, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "sum(rate(vm_tcplistener_read_bytes_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval])) by (instance) * 8 > 0", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "sum(rate(vm_tcplistener_read_bytes_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval])) by (instance) * 8 > 0", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Network usage ($instance)", @@ -6746,7 +7477,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "format": "bps", "logBase": 1, "show": true @@ -6777,13 +7509,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 168 + "y": 49 }, "hiddenSeries": false, "id": 88, @@ -6807,7 +7539,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -6815,18 +7547,20 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "histogram_quantile(0.99, sum(increase(vm_rows_per_insert_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(increase(vm_rows_per_insert_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (instance, vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Rows per insert ($instance)", @@ -6841,7 +7575,9 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { + "$$hashKey": "object:2848", "decimals": 2, "format": "short", "logBase": 1, @@ -6849,6 +7585,7 @@ "show": true }, { + "$$hashKey": "object:2849", "format": "short", "logBase": 1, "min": "0", @@ -6874,13 +7611,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 176 + "y": 57 }, "hiddenSeries": false, "id": 139, @@ -6904,7 +7641,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -6912,28 +7649,32 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": true, - "expr": "rate(vm_rpc_send_duration_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}} => {{addr}}", - "refId": "A" - }], - "thresholds": [{ - "$$hashKey": "object:234", - "colorMode": "critical", - "fill": true, - "line": true, - "op": "gt", - "value": 0.9, - "yaxis": "left" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "exemplar": true, + "expr": "rate(vm_rpc_send_duration_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}} => {{addr}}", + "refId": "A" + } + ], + "thresholds": [ + { + "$$hashKey": "object:234", + "colorMode": "critical", + "fill": true, + "line": true, + "op": "gt", + "value": 0.9, + "yaxis": "left" + } + ], "timeRegions": [], "title": "Storage connection saturation ($instance)", "tooltip": { @@ -6947,7 +7688,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:156", "decimals": 0, "format": "s", @@ -6988,7 +7730,7 @@ "h": 8, "w": 12, "x": 12, - "y": 176 + "y": 57 }, "hiddenSeries": false, "id": 114, @@ -7012,7 +7754,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -7020,18 +7762,20 @@ "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "expr": "vm_rpc_vmstorage_is_reachable{job=~\"$job\", instance=~\"$instance\"}", - "format": "time_series", - "interval": "", - "intervalFactor": 1, - "legendFormat": "{{instance}} => {{addr}}", - "refId": "A" - }], + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "vm_rpc_vmstorage_is_reachable{job=~\"$job\", instance=~\"$instance\"}", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}} => {{addr}}", + "refId": "A" + } + ], "thresholds": [], "timeRegions": [], "title": "Storage reachability ($instance)", @@ -7046,7 +7790,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:1534", "decimals": 0, "format": "short", @@ -7087,7 +7832,7 @@ "h": 7, "w": 12, "x": 0, - "y": 184 + "y": 65 }, "hiddenSeries": false, "id": 164, @@ -7111,30 +7856,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval]))", + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job_insert\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "cores used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -7144,9 +7892,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(process_cpu_cores_available{job=~\"$job_insert\", instance=~\"$instance\"})", + "expr": "min(process_cpu_cores_available{job=~\"$job_insert\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -7165,7 +7913,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:887", "format": "short", "logBase": 1, @@ -7204,7 +7953,7 @@ "h": 7, "w": 12, "x": 12, - "y": 184 + "y": 65 }, "hiddenSeries": false, "id": 169, @@ -7228,30 +7977,33 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [{ - "$$hashKey": "object:186", - "alias": "max", - "color": "#C4162A" - }], + "seriesOverrides": [ + { + "$$hashKey": "object:186", + "alias": "limit", + "color": "#C4162A" + } + ], "spaceLength": 10, "stack": false, "steppedLine": false, - "targets": [{ + "targets": [ + { "datasource": { "type": "prometheus", "uid": "$ds" }, "editorMode": "code", "exemplar": true, - "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_insert\", instance=~\"$instance\"}) ", + "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job_insert\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, - "legendFormat": "mem used", + "legendFormat": "{{instance}}", "range": true, "refId": "A" }, @@ -7261,9 +8013,9 @@ "uid": "$ds" }, "editorMode": "code", - "expr": "sum(vm_available_memory_bytes{job=~\"$job_insert\", instance=~\"$instance\"})", + "expr": "min(vm_available_memory_bytes{job=~\"$job_insert\", instance=~\"$instance\"})", "hide": false, - "legendFormat": "max", + "legendFormat": "limit", "range": true, "refId": "B" } @@ -7282,7 +8034,8 @@ "show": true, "values": [] }, - "yaxes": [{ + "yaxes": [ + { "$$hashKey": "object:164", "format": "bytes", "logBase": 1, @@ -7305,11 +8058,13 @@ "type": "row" } ], + "refresh": false, "schemaVersion": 36, "style": "dark", "tags": [], "templating": { - "list": [{ + "list": [ + { "current": { "selected": false, "text": "VictoriaMetrics", diff --git a/dashboards/victoriametrics.json b/dashboards/victoriametrics.json index 5efea695e..b64221206 100644 --- a/dashboards/victoriametrics.json +++ b/dashboards/victoriametrics.json @@ -55,13 +55,13 @@ } ] }, - "description": "Overview for single node VictoriaMetrics v1.77.0 or higher", + "description": "Overview for single node VictoriaMetrics v1.79.0 or higher", "editable": true, "fiscalYearStartMonth": 0, "gnetId": 10229, "graphTooltip": 0, "id": null, - "iteration": 1659966607833, + "iteration": 1663338736864, "links": [ { "icon": "doc", @@ -281,6 +281,7 @@ }, { "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Average disk usage per datapoint.", @@ -334,7 +335,7 @@ "uid": "$ds" }, "exemplar": true, - "expr": "sum(vm_data_size_bytes{job=~\"$job\", type!=\"indexdb\"}) / sum(vm_rows{job=~\"$job\", type!=\"indexdb\"})", + "expr": "sum(vm_data_size_bytes{job=~\"$job\"}) / sum(vm_rows{job=~\"$job\"})", "format": "time_series", "instant": true, "interval": "", @@ -790,7 +791,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -857,12 +858,14 @@ }, "yaxes": [ { + "$$hashKey": "object:758", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:759", "format": "short", "logBase": 1, "min": "0", @@ -1187,7 +1190,7 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -1219,12 +1222,13 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", "seriesOverrides": [ { + "$$hashKey": "object:840", "alias": "max", "color": "#C4162A" } @@ -1291,6 +1295,420 @@ "align": false } }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples read per query.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 22 + }, + "hiddenSeries": false, + "id": 101, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_read_per_query_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows read per query ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of series read per query.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 30 + }, + "hiddenSeries": false, + "id": 99, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_series_read_per_query_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Series read per query ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples scanner per query.\n\nThis number can exceed number of RowsReadPerQuery if `step` query arg passed to [/api/v1/query_range](https://prometheus.io/docs/prometheus/latest/querying/api/#range-queries) is smaller than the lookbehind window set in square brackets of [rollup function](https://docs.victoriametrics.com/MetricsQL.html#rollup-functions). For example, if `increase(some_metric[1h])` is executed with the `step=5m`, then the same raw samples on a hour time range are scanned `1h/5m=12` times. See [this article](https://valyala.medium.com/how-to-optimize-promql-and-metricsql-queries-85a1b75bf986) for details.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 30 + }, + "hiddenSeries": false, + "id": 105, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_scanned_per_query_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{instance}}", + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows scanned per series ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "99th percentile of number of raw samples read per queried series.", + "fieldConfig": { + "defaults": { + "links": [] + }, + "overrides": [] + }, + "fill": 0, + "fillGradient": 0, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 38 + }, + "hiddenSeries": false, + "id": 103, + "legend": { + "alignAsTable": true, + "avg": true, + "current": true, + "max": true, + "min": false, + "show": true, + "sort": "current", + "sortDesc": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null as zero", + "options": { + "alertThreshold": true + }, + "percentage": false, + "pluginVersion": "9.0.3", + "pointradius": 2, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.99, sum(rate(vm_rows_read_per_series_bucket{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (vmrange))", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{label_name}}", + "range": true, + "refId": "A" + } + ], + "thresholds": [], + "timeRegions": [], + "title": "Rows read per series ($instance)", + "tooltip": { + "shared": true, + "sort": 2, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "mode": "time", + "show": true, + "values": [] + }, + "yaxes": [ + { + "$$hashKey": "object:2848", + "decimals": 2, + "format": "short", + "logBase": 1, + "min": "0", + "show": true + }, + { + "$$hashKey": "object:2849", + "format": "short", + "logBase": 1, + "min": "0", + "show": true + } + ], + "yaxis": { + "align": false + } + }, { "collapsed": true, "datasource": { @@ -1301,7 +1719,7 @@ "h": 1, "w": 24, "x": 0, - "y": 30 + "y": 46 }, "id": 92, "panels": [ @@ -1321,7 +1739,7 @@ "h": 8, "w": 24, "x": 0, - "y": 7 + "y": 31 }, "hiddenSeries": false, "id": 94, @@ -1434,7 +1852,7 @@ "h": 7, "w": 24, "x": 0, - "y": 15 + "y": 39 }, "hiddenSeries": false, "id": 97, @@ -1523,7 +1941,7 @@ "h": 8, "w": 24, "x": 0, - "y": 22 + "y": 46 }, "hiddenSeries": false, "id": 95, @@ -1630,7 +2048,7 @@ "h": 1, "w": 24, "x": 0, - "y": 31 + "y": 47 }, "id": 14, "panels": [ @@ -1655,7 +2073,7 @@ "h": 8, "w": 12, "x": 0, - "y": 3 + "y": 4 }, "hiddenSeries": false, "id": 10, @@ -1680,7 +2098,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1748,13 +2166,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 3 + "y": 4 }, "hiddenSeries": false, "id": 73, @@ -1779,7 +2197,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1817,12 +2235,14 @@ }, "yaxes": [ { + "$$hashKey": "object:919", "format": "s", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:920", "format": "short", "logBase": 1, "min": "0", @@ -1848,13 +2268,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 11 + "y": 12 }, "hiddenSeries": false, "id": 30, @@ -1878,12 +2298,13 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", "seriesOverrides": [ { + "$$hashKey": "object:997", "alias": "bytes-per-datapoint", "yaxis": 2 } @@ -1931,12 +2352,14 @@ }, "yaxes": [ { + "$$hashKey": "object:1004", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1005", "decimals": 2, "format": "bytes", "logBase": 1, @@ -1969,7 +2392,7 @@ "h": 8, "w": 12, "x": 12, - "y": 11 + "y": 12 }, "hiddenSeries": false, "id": 34, @@ -1993,7 +2416,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2078,13 +2501,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 19 + "y": 20 }, "hiddenSeries": false, "id": 53, @@ -2109,7 +2532,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2157,12 +2580,14 @@ }, "yaxes": [ { + "$$hashKey": "object:1136", "format": "bytes", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1137", "format": "short", "logBase": 1, "min": "0", @@ -2188,13 +2613,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 19 + "y": 20 }, "hiddenSeries": false, "id": 36, @@ -2218,7 +2643,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2254,12 +2679,14 @@ }, "yaxes": [ { + "$$hashKey": "object:1066", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1067", "format": "short", "logBase": 1, "min": "0", @@ -2291,7 +2718,7 @@ "h": 8, "w": 12, "x": 0, - "y": 27 + "y": 28 }, "hiddenSeries": false, "id": 55, @@ -2315,7 +2742,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2384,13 +2811,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 27 + "y": 28 }, "hiddenSeries": false, "id": 62, @@ -2413,7 +2840,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2447,6 +2874,7 @@ }, "yaxes": [ { + "$$hashKey": "object:1262", "decimals": 0, "format": "short", "logBase": 1, @@ -2454,6 +2882,7 @@ "show": true }, { + "$$hashKey": "object:1263", "format": "short", "logBase": 1, "min": "0", @@ -2485,7 +2914,7 @@ "h": 8, "w": 12, "x": 0, - "y": 35 + "y": 36 }, "hiddenSeries": false, "id": 58, @@ -2509,7 +2938,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2579,13 +3008,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 35 + "y": 36 }, "hiddenSeries": false, "id": 64, @@ -2608,7 +3037,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2682,7 +3111,7 @@ "h": 8, "w": 12, "x": 12, - "y": 43 + "y": 44 }, "hiddenSeries": false, "id": 67, @@ -2706,7 +3135,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.2", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2780,7 +3209,7 @@ "h": 1, "w": 24, "x": 0, - "y": 32 + "y": 48 }, "id": 71, "panels": [ @@ -2805,7 +3234,7 @@ "h": 8, "w": 12, "x": 0, - "y": 32 + "y": 5 }, "hiddenSeries": false, "id": 66, @@ -2915,7 +3344,7 @@ "h": 8, "w": 12, "x": 12, - "y": 32 + "y": 5 }, "hiddenSeries": false, "id": 96, @@ -3014,7 +3443,7 @@ "h": 8, "w": 12, "x": 0, - "y": 40 + "y": 13 }, "hiddenSeries": false, "id": 68, @@ -3124,7 +3553,7 @@ "h": 8, "w": 12, "x": 12, - "y": 40 + "y": 13 }, "hiddenSeries": false, "id": 60, @@ -3217,7 +3646,7 @@ "h": 8, "w": 12, "x": 0, - "y": 48 + "y": 21 }, "hiddenSeries": false, "id": 90, @@ -3311,7 +3740,7 @@ "h": 8, "w": 12, "x": 12, - "y": 48 + "y": 21 }, "hiddenSeries": false, "id": 74, @@ -3410,7 +3839,7 @@ "h": 1, "w": 24, "x": 0, - "y": 33 + "y": 49 }, "id": 46, "panels": [ @@ -3429,13 +3858,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 5 + "y": 50 }, "hiddenSeries": false, "id": 44, @@ -3459,7 +3888,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3543,12 +3972,14 @@ }, "yaxes": [ { + "$$hashKey": "object:1432", "format": "bytes", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1433", "format": "short", "logBase": 1, "min": "0", @@ -3575,13 +4006,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 12, - "y": 5 + "y": 50 }, "hiddenSeries": false, "id": 57, @@ -3605,7 +4036,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3663,12 +4094,14 @@ }, "yaxes": [ { + "$$hashKey": "object:1370", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:1371", "format": "short", "logBase": 1, "min": "0", @@ -3694,13 +4127,13 @@ }, "overrides": [] }, - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, "w": 12, "x": 0, - "y": 13 + "y": 58 }, "hiddenSeries": false, "id": 75, @@ -3724,12 +4157,13 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", "seriesOverrides": [ { + "$$hashKey": "object:1514", "alias": "max", "color": "#C4162A" } @@ -3777,6 +4211,7 @@ }, "yaxes": [ { + "$$hashKey": "object:1521", "decimals": 0, "format": "short", "logBase": 2, @@ -3784,6 +4219,7 @@ "show": true }, { + "$$hashKey": "object:1522", "format": "short", "logBase": 1, "min": "0", @@ -3815,7 +4251,7 @@ "h": 8, "w": 12, "x": 12, - "y": 13 + "y": 58 }, "hiddenSeries": false, "id": 76, @@ -3839,7 +4275,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3929,7 +4365,7 @@ "h": 8, "w": 12, "x": 0, - "y": 21 + "y": 66 }, "hiddenSeries": false, "id": 47, @@ -3953,7 +4389,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4027,7 +4463,7 @@ "h": 8, "w": 12, "x": 12, - "y": 21 + "y": 66 }, "hiddenSeries": false, "id": 42, @@ -4051,7 +4487,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4123,7 +4559,7 @@ "h": 8, "w": 12, "x": 0, - "y": 29 + "y": 74 }, "hiddenSeries": false, "id": 48, @@ -4147,7 +4583,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4221,7 +4657,7 @@ "h": 8, "w": 12, "x": 12, - "y": 29 + "y": 74 }, "hiddenSeries": false, "id": 37, @@ -4245,7 +4681,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4319,7 +4755,7 @@ "h": 8, "w": 12, "x": 12, - "y": 37 + "y": 82 }, "hiddenSeries": false, "id": 49, @@ -4343,7 +4779,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", diff --git a/dashboards/vmagent.json b/dashboards/vmagent.json index c292ede53..b5dbc5211 100644 --- a/dashboards/vmagent.json +++ b/dashboards/vmagent.json @@ -1,12 +1,12 @@ { "__inputs": [], - "__elements": [], + "__elements": {}, "__requires": [ { "type": "grafana", "id": "grafana", "name": "Grafana", - "version": "8.4.4" + "version": "9.0.3" }, { "type": "panel", @@ -34,8 +34,8 @@ }, { "type": "panel", - "id": "table-old", - "name": "Table (old)", + "id": "table", + "name": "Table", "version": "" } ], @@ -66,7 +66,7 @@ "fiscalYearStartMonth": 0, "graphTooltip": 1, "id": null, - "iteration": 1657810604530, + "iteration": 1663339589357, "links": [ { "icon": "doc", @@ -109,9 +109,149 @@ }, "id": 24, "panels": [], + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "refId": "A" + } + ], "title": "Overview", "type": "row" }, + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "Shows the rate of samples scraped from configured targets.", + "fieldConfig": { + "defaults": { + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 0, + "y": 1 + }, + "id": 103, + "links": [ + { + "title": "Troubleshooting", + "url": "https://docs.victoriametrics.com/vmagent.html#troubleshooting" + } + ], + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "editorMode": "code", + "expr": "sum(rate(vm_promscrape_scraped_samples_sum{job=~\"$job\", instance=~\"$instance\", path!~\"/favicon.ico\"}[$__rate_interval]))", + "interval": "", + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Samples scraped/s", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "description": "Shows the rate of ingested samples", + "fieldConfig": { + "defaults": { + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 4, + "y": 1 + }, + "id": 102, + "links": [ + { + "title": "Troubleshooting", + "url": "https://docs.victoriametrics.com/vmagent.html#troubleshooting" + } + ], + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.0.3", + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "editorMode": "code", + "expr": "sum(rate(vm_ingestserver_requests_total{job=~\"$job\", instance=~\"$instance\", path!~\"/favicon.ico\"}[$__rate_interval]))", + "interval": "", + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Samples ingested/s", + "type": "stat" + }, { "datasource": { "uid": "$ds" @@ -135,7 +275,7 @@ "gridPos": { "h": 3, "w": 4, - "x": 0, + "x": 8, "y": 1 }, "id": 9, @@ -154,9 +294,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(vm_promscrape_targets{job=~\"$job\", instance=~\"$instance\", status=\"up\"})", "interval": "", "legendFormat": "up", @@ -193,7 +336,7 @@ "gridPos": { "h": 3, "w": 4, - "x": 4, + "x": 12, "y": 1 }, "id": 72, @@ -218,9 +361,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(vm_promscrape_targets{job=~\"$job\", instance=~\"$instance\", status=\"down\"})", "interval": "", "legendFormat": "up", @@ -259,7 +405,7 @@ "gridPos": { "h": 3, "w": 4, - "x": 8, + "x": 16, "y": 1 }, "id": 16, @@ -285,9 +431,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(increase(vm_log_messages_total{job=~\"$job\", instance=~\"$instance\", level!=\"info\"}[30m]))", "interval": "", "legendFormat": "", @@ -324,8 +473,8 @@ }, "gridPos": { "h": 3, - "w": 5, - "x": 12, + "w": 4, + "x": 20, "y": 1 }, "id": 56, @@ -344,9 +493,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(vm_persistentqueue_bytes_pending{job=~\"$job\", instance=~\"$instance\"})", "interval": "", "legendFormat": "", @@ -357,92 +509,97 @@ "type": "stat" }, { - "columns": [], "datasource": { + "type": "prometheus", "uid": "$ds" }, - "fontSize": "100%", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false, + "minWidth": 50 + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Time" + }, + "properties": [ + { + "id": "custom.hidden", + "value": true + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Count" + } + ] + } + ] + }, "gridPos": { - "h": 7, - "w": 7, - "x": 17, - "y": 1 + "h": 5, + "w": 8, + "x": 0, + "y": 4 }, - "id": 11, - "scroll": true, - "showHeader": true, - "sort": { - "col": 4, - "desc": false + "id": 101, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true }, - "styles": [ - { - "alias": "uptime", - "align": "auto", - "colorMode": "cell", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "Value", - "thresholds": [ - "1800", - "3600" - ], - "type": "number", - "unit": "s" - }, - { - "alias": "", - "align": "auto", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "instance", - "thresholds": [], - "type": "string", - "unit": "short" - }, - { - "alias": "", - "align": "auto", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "/.*/", - "thresholds": [], - "type": "hidden", - "unit": "short" - } - ], + "pluginVersion": "9.0.3", "targets": [ { - "expr": "sort((time() - vm_app_start_timestamp{job=~\"$job\", instance=~\"$instance\"}) or (up{job=~\"$job\", instance=~\"$instance\"}))", + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(vm_app_version{job=~\"$job\", instance=~\"$instance\"}) by(job, short_version)", "format": "table", - "hide": false, "instant": true, - "interval": "", - "legendFormat": "{{instance}}", + "range": false, "refId": "A" } ], - "title": "Uptime", - "transform": "table", - "type": "table-old" + "type": "table" }, { "aliasColors": {}, @@ -461,9 +618,9 @@ "fill": 0, "fillGradient": 0, "gridPos": { - "h": 4, - "w": 17, - "x": 0, + "h": 5, + "w": 16, + "x": 8, "y": 4 }, "hiddenSeries": false, @@ -490,7 +647,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -500,6 +657,9 @@ "steppedLine": true, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sort(sum(up{job=~\"$job\", instance=~\"$instance\"}) by (job, instance))", "format": "time_series", "instant": false, @@ -566,7 +726,7 @@ "h": 8, "w": 12, "x": 0, - "y": 8 + "y": 9 }, "hiddenSeries": false, "id": 5, @@ -589,7 +749,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -642,11 +802,13 @@ }, "yaxes": [ { + "$$hashKey": "object:537", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:538", "format": "bytes", "logBase": 1, "show": true @@ -678,7 +840,7 @@ "h": 8, "w": 12, "x": 12, - "y": 8 + "y": 9 }, "hiddenSeries": false, "id": 15, @@ -702,7 +864,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -775,7 +937,7 @@ "h": 8, "w": 12, "x": 0, - "y": 16 + "y": 17 }, "hiddenSeries": false, "id": 69, @@ -805,7 +967,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -917,7 +1079,7 @@ "h": 8, "w": 12, "x": 12, - "y": 16 + "y": 17 }, "hiddenSeries": false, "id": 17, @@ -946,7 +1108,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -956,6 +1118,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(vmagent_remotewrite_pending_data_bytes{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}) by (url)", "interval": "", @@ -1015,7 +1180,7 @@ "h": 8, "w": 12, "x": 0, - "y": 24 + "y": 25 }, "hiddenSeries": false, "id": 79, @@ -1039,7 +1204,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1049,6 +1214,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(rate(vmagent_remotewrite_packets_dropped_total{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}[$__interval])) by(url)", "interval": "", @@ -1108,7 +1276,7 @@ "h": 8, "w": 12, "x": 12, - "y": 24 + "y": 25 }, "hiddenSeries": false, "id": 49, @@ -1138,7 +1306,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1148,6 +1316,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(increase(vm_persistentqueue_bytes_dropped_total{job=~\"$job\", instance=~\"$instance\"}[$__interval])) by (path)", "interval": "", "legendFormat": "{{ path }}", @@ -1207,7 +1378,7 @@ "h": 8, "w": 12, "x": 0, - "y": 32 + "y": 33 }, "hiddenSeries": false, "id": 18, @@ -1237,7 +1408,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1320,7 +1491,7 @@ "h": 8, "w": 12, "x": 12, - "y": 32 + "y": 33 }, "hiddenSeries": false, "id": 86, @@ -1344,7 +1515,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1354,6 +1525,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(rate(vm_log_messages_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (level) ", "format": "time_series", @@ -1405,7 +1579,7 @@ "h": 1, "w": 24, "x": 0, - "y": 40 + "y": 41 }, "id": 28, "panels": [ @@ -1429,7 +1603,7 @@ "h": 7, "w": 12, "x": 0, - "y": 2 + "y": 42 }, "hiddenSeries": false, "id": 48, @@ -1452,7 +1626,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1462,6 +1636,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(vm_promscrape_targets{job=~\"$job\", instance=~\"$instance\", status=\"up\"}) by(type) > 0", "format": "time_series", @@ -1521,7 +1698,7 @@ "h": 7, "w": 12, "x": 12, - "y": 2 + "y": 42 }, "hiddenSeries": false, "id": 76, @@ -1544,7 +1721,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1554,6 +1731,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(vm_promscrape_targets{job=~\"$job\", instance=~\"$instance\", status=\"down\"}) by(type) > 0", "format": "time_series", @@ -1614,7 +1794,7 @@ "h": 8, "w": 12, "x": 0, - "y": 9 + "y": 49 }, "hiddenSeries": false, "id": 20, @@ -1637,7 +1817,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1727,7 +1907,7 @@ "h": 8, "w": 12, "x": 12, - "y": 9 + "y": 49 }, "hiddenSeries": false, "id": 31, @@ -1750,7 +1930,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1851,7 +2031,7 @@ "h": 8, "w": 12, "x": 0, - "y": 17 + "y": 57 }, "hiddenSeries": false, "id": 46, @@ -1874,7 +2054,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1884,6 +2064,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "histogram_quantile(0.95, sum(rate(vm_promscrape_scrape_response_size_bytes_bucket{job=~\"$job\", instance=~\"$instance\"}[$__interval])) by(vmrange)) ", "format": "time_series", "interval": "", @@ -1891,6 +2074,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "expr": "histogram_quantile(0.5, sum(rate(vm_promscrape_scrape_response_size_bytes_bucket{job=~\"$job\", instance=~\"$instance\"}[$__interval])) by(vmrange)) ", "interval": "", "legendFormat": "p0.5", @@ -1947,7 +2133,7 @@ "h": 8, "w": 12, "x": 12, - "y": 17 + "y": 57 }, "heatmap": {}, "hideZeroBuckets": false, @@ -1989,6 +2175,14 @@ "yBucketBound": "auto" } ], + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "refId": "A" + } + ], "title": "Scraping", "type": "row" }, @@ -2001,7 +2195,7 @@ "h": 1, "w": 24, "x": 0, - "y": 41 + "y": 42 }, "id": 71, "panels": [ @@ -2027,7 +2221,7 @@ "h": 8, "w": 12, "x": 0, - "y": 3 + "y": 66 }, "hiddenSeries": false, "id": 73, @@ -2051,7 +2245,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2099,12 +2293,14 @@ }, "yaxes": [ { + "$$hashKey": "object:410", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:411", "format": "none", "logBase": 1, "show": true @@ -2136,7 +2332,7 @@ "h": 8, "w": 12, "x": 12, - "y": 3 + "y": 66 }, "hiddenSeries": false, "id": 77, @@ -2160,7 +2356,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2208,12 +2404,14 @@ }, "yaxes": [ { + "$$hashKey": "object:272", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:273", "format": "none", "logBase": 1, "show": true @@ -2245,7 +2443,7 @@ "h": 8, "w": 12, "x": 0, - "y": 11 + "y": 74 }, "hiddenSeries": false, "id": 78, @@ -2269,7 +2467,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2306,12 +2504,14 @@ }, "yaxes": [ { + "$$hashKey": "object:349", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:350", "format": "none", "logBase": 1, "show": true @@ -2343,7 +2543,7 @@ "h": 8, "w": 12, "x": 12, - "y": 11 + "y": 74 }, "hiddenSeries": false, "id": 50, @@ -2366,7 +2566,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2419,6 +2619,14 @@ } } ], + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "refId": "A" + } + ], "title": "Ingestion", "type": "row" }, @@ -2431,7 +2639,7 @@ "h": 1, "w": 24, "x": 0, - "y": 42 + "y": 43 }, "id": 58, "panels": [ @@ -2457,7 +2665,7 @@ "h": 8, "w": 12, "x": 0, - "y": 43 + "y": 83 }, "hiddenSeries": false, "id": 60, @@ -2480,7 +2688,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2506,7 +2714,7 @@ "title": "Requests rate ($instance) to ($url)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -2517,6 +2725,7 @@ }, "yaxes": [ { + "$$hashKey": "object:720", "decimals": 2, "format": "short", "logBase": 1, @@ -2524,6 +2733,7 @@ "show": true }, { + "$$hashKey": "object:721", "format": "short", "logBase": 1, "show": true @@ -2555,7 +2765,7 @@ "h": 8, "w": 12, "x": 12, - "y": 43 + "y": 83 }, "hiddenSeries": false, "id": 66, @@ -2578,7 +2788,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2652,7 +2862,7 @@ "h": 8, "w": 12, "x": 0, - "y": 51 + "y": 91 }, "hiddenSeries": false, "id": 61, @@ -2675,7 +2885,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2690,7 +2900,7 @@ "uid": "$ds" }, "exemplar": true, - "expr": "sum(rate(vmagent_remotewrite_retries_count_total{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}[$__rate_interval])) by(url)", + "expr": "sum(rate(vmagent_remotewrite_retries_count_total{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}[$__rate_interval])) by(url) > 0", "interval": "", "legendFormat": "", "refId": "A" @@ -2712,12 +2922,14 @@ }, "yaxes": [ { + "$$hashKey": "object:789", "format": "short", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:790", "format": "short", "logBase": 1, "show": true @@ -2748,7 +2960,7 @@ "h": 8, "w": 12, "x": 12, - "y": 51 + "y": 91 }, "hiddenSeries": false, "id": 65, @@ -2771,7 +2983,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2781,6 +2993,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(vmagent_remotewrite_conns{job=~\"$job\", instance=~\"$instance\"}) by (instance)", "interval": "", @@ -2837,7 +3052,7 @@ "h": 8, "w": 12, "x": 0, - "y": 59 + "y": 99 }, "heatmap": {}, "hideZeroBuckets": false, @@ -2849,6 +3064,9 @@ "reverseYBuckets": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "buckets_limit(12, prometheus_buckets(sum(rate(vmagent_remotewrite_duration_seconds_bucket{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}[$__interval])) by(vmrange)))", "format": "heatmap", @@ -2897,7 +3115,7 @@ "h": 8, "w": 12, "x": 12, - "y": 59 + "y": 99 }, "hiddenSeries": false, "id": 84, @@ -2920,7 +3138,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2930,6 +3148,10 @@ "steppedLine": false, "targets": [ { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, "exemplar": true, "expr": "sum(rate(vmagent_remotewrite_send_duration_seconds_total{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}[$__rate_interval])) by (instance, url)\n/\nmax(vmagent_remotewrite_queues{job=~\"$job\", instance=~\"$instance\", url=~\"$url\"}) by(instance, url)", "interval": "", @@ -2998,7 +3220,7 @@ "h": 8, "w": 12, "x": 0, - "y": 67 + "y": 107 }, "heatmap": {}, "hideZeroBuckets": false, @@ -3010,6 +3232,9 @@ "reverseYBuckets": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "buckets_limit(12, prometheus_buckets(sum(rate(vmagent_remotewrite_block_size_rows_bucket{job=~\"$job\", instance=~\"$instance\"}[$__interval])) by(vmrange)))", "format": "heatmap", @@ -3054,7 +3279,7 @@ "h": 8, "w": 12, "x": 12, - "y": 67 + "y": 107 }, "heatmap": {}, "hideZeroBuckets": false, @@ -3066,6 +3291,9 @@ "reverseYBuckets": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "buckets_limit(12, prometheus_buckets(sum(rate(vmagent_remotewrite_block_size_bytes_bucket{job=~\"$job\", instance=~\"$instance\"}[$__interval])) by(vmrange)))", "format": "heatmap", "interval": "", @@ -3105,7 +3333,7 @@ "h": 8, "w": 12, "x": 0, - "y": 75 + "y": 115 }, "hiddenSeries": false, "id": 88, @@ -3125,7 +3353,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3135,6 +3363,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "(vmagent_hourly_series_limit_current_series{job=~\"$job\", instance=~\"$instance\"} / vmagent_hourly_series_limit_max_series{job=~\"$job\", instance=~\"$instance\"}) * 100", "interval": "", @@ -3142,6 +3373,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "vmagent_daily_series_limit_max_series{job=~\"$job\", instance=~\"$instance\"}", "hide": true, @@ -3208,7 +3442,7 @@ "h": 8, "w": 12, "x": 12, - "y": 75 + "y": 115 }, "hiddenSeries": false, "id": 90, @@ -3228,7 +3462,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.4.4", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3238,6 +3472,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "(vmagent_daily_series_limit_current_series{job=~\"$job\", instance=~\"$instance\"} / vmagent_daily_series_limit_max_series{job=~\"$job\", instance=~\"$instance\"}) * 100", "interval": "", @@ -3245,6 +3482,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "vmagent_daily_series_limit_max_series{job=~\"$job\", instance=~\"$instance\"}", "hide": true, @@ -3297,6 +3537,14 @@ } } ], + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "refId": "A" + } + ], "title": "Remote write", "type": "row" }, @@ -3310,7 +3558,7 @@ "h": 1, "w": 24, "x": 0, - "y": 43 + "y": 44 }, "id": 94, "panels": [ @@ -3330,7 +3578,7 @@ "h": 8, "w": 12, "x": 0, - "y": 5 + "y": 124 }, "hiddenSeries": false, "id": 92, @@ -3354,7 +3602,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3420,7 +3668,7 @@ "h": 8, "w": 12, "x": 12, - "y": 5 + "y": 124 }, "hiddenSeries": false, "id": 95, @@ -3444,7 +3692,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3516,7 +3764,7 @@ "h": 8, "w": 12, "x": 0, - "y": 13 + "y": 132 }, "hiddenSeries": false, "id": 98, @@ -3539,7 +3787,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3625,7 +3873,7 @@ "h": 8, "w": 12, "x": 12, - "y": 13 + "y": 132 }, "hiddenSeries": false, "id": 99, @@ -3648,7 +3896,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3713,6 +3961,15 @@ } } ], + "targets": [ + { + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "refId": "A" + } + ], "title": "Troubleshooting", "type": "row" }, @@ -3725,7 +3982,7 @@ "h": 1, "w": 24, "x": 0, - "y": 44 + "y": 45 }, "id": 45, "panels": [ @@ -3738,7 +3995,7 @@ "type": "prometheus", "uid": "$ds" }, - "description": "Shows the CPU usage per vmagent instance. \nIf you think that usage is abnormal or unexpected pls file an issue and attach CPU profile if possible.", + "description": "Shows the CPU usage percentage per vmagent instance. \nIf you think that usage is abnormal or unexpected, pls file an issue and attach CPU profile if possible.", "fieldConfig": { "defaults": { "links": [] @@ -3751,7 +4008,7 @@ "h": 8, "w": 12, "x": 0, - "y": 62 + "y": 141 }, "hiddenSeries": false, "id": 35, @@ -3781,17 +4038,11 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [ - { - "$$hashKey": "object:77", - "alias": "/Limit.*/", - "color": "#F2495C" - } - ], + "seriesOverrides": [], "spaceLength": 10, "stack": false, "steppedLine": false, @@ -3802,26 +4053,12 @@ "uid": "$ds" }, "exemplar": false, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance) / max(process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, "legendFormat": "{{instance}}", "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": false, - "expr": "process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}", - "format": "time_series", - "hide": false, - "interval": "", - "intervalFactor": 1, - "legendFormat": "Limit ({{instance}})", - "refId": "B" } ], "thresholds": [], @@ -3829,7 +4066,7 @@ "title": "CPU ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -3840,12 +4077,14 @@ }, "yaxes": [ { - "format": "short", + "$$hashKey": "object:912", + "format": "percentunit", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:913", "format": "short", "logBase": 1, "show": true @@ -3876,7 +4115,7 @@ "h": 8, "w": 12, "x": 12, - "y": 62 + "y": 141 }, "hiddenSeries": false, "id": 37, @@ -3906,7 +4145,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -3916,6 +4155,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(process_resident_memory_bytes{job=~\"$job\", instance=~\"$instance\"}) by (instance)", "interval": "", @@ -3923,6 +4165,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "exemplar": true, "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job\", instance=~\"$instance\"}) by (instance)", "hide": false, @@ -3983,7 +4228,7 @@ "h": 8, "w": 12, "x": 0, - "y": 70 + "y": 149 }, "hiddenSeries": false, "id": 81, @@ -4007,12 +4252,13 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", "seriesOverrides": [ { + "$$hashKey": "object:1206", "alias": "read", "transform": "negative-Y" } @@ -4022,6 +4268,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(rate(process_io_storage_read_bytes_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval]))", "format": "time_series", "hide": false, @@ -4031,6 +4280,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "expr": "sum(rate(process_io_storage_written_bytes_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval]))", "format": "time_series", "hide": false, @@ -4056,11 +4308,13 @@ }, "yaxes": [ { + "$$hashKey": "object:1213", "format": "bytes", "logBase": 1, "show": true }, { + "$$hashKey": "object:1214", "format": "short", "logBase": 1, "min": "0", @@ -4092,7 +4346,7 @@ "h": 8, "w": 12, "x": 12, - "y": 70 + "y": 149 }, "hiddenSeries": false, "id": 7, @@ -4116,7 +4370,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4131,12 +4385,18 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(rate(vm_tcplistener_read_bytes_total{job=~\"$job\", instance=~\"$instance\"}[$__interval])) * 8\n+ sum(rate(vm_promscrape_conn_bytes_read_total{job=~\"$job\", instance=~\"$instance\"}[$__interval])) * 8", "interval": "", "legendFormat": "in", "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "expr": "sum(rate(vmagent_remotewrite_conn_bytes_written_total{job=~\"$job\", instance=~\"$instance\"}[$__interval])) * 8", "interval": "", "legendFormat": "out", @@ -4179,9 +4439,10 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, - "description": "Panel shows the number of open file descriptors in the OS.\nReaching the limit of open files can cause various issues and must be prevented.\n\nSee how to change limits here https://medium.com/@muhammadtriwibowo/set-permanently-ulimit-n-open-files-in-ubuntu-4d61064429a", + "description": "Panel shows the percentage of open file descriptors in the OS per instance.\nReaching the limit of open files (100%) can cause various issues and must be prevented.\n\nSee how to change limits here https://medium.com/@muhammadtriwibowo/set-permanently-ulimit-n-open-files-in-ubuntu-4d61064429a", "fieldConfig": { "defaults": { "links": [] @@ -4194,7 +4455,7 @@ "h": 8, "w": 12, "x": 0, - "y": 78 + "y": 157 }, "hiddenSeries": false, "id": 83, @@ -4218,36 +4479,27 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [ - { - "$$hashKey": "object:913", - "alias": "max", - "color": "#C4162A" - } - ], + "seriesOverrides": [], "spaceLength": 10, "stack": false, "steppedLine": false, "targets": [ { - "expr": "sum(process_open_fds{job=~\"$job\", instance=~\"$instance\"})", + "datasource": { + "uid": "$ds" + }, + "editorMode": "code", + "expr": "max(process_open_fds{job=~\"$job\", instance=~\"$instance\"}) by(instance) \n/\nmin(process_max_fds{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 2, - "legendFormat": "open", + "legendFormat": "__auto", + "range": true, "refId": "A" - }, - { - "expr": "min(process_max_fds{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "interval": "", - "intervalFactor": 2, - "legendFormat": "max", - "refId": "B" } ], "thresholds": [], @@ -4255,7 +4507,7 @@ "title": "Open FDs ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -4266,13 +4518,15 @@ }, "yaxes": [ { - "decimals": 0, - "format": "short", - "logBase": 2, + "$$hashKey": "object:987", + "decimals": 5, + "format": "percentunit", + "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:988", "format": "short", "logBase": 1, "min": "0", @@ -4303,7 +4557,7 @@ "h": 8, "w": 12, "x": 12, - "y": 78 + "y": 157 }, "hiddenSeries": false, "id": 39, @@ -4327,7 +4581,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4337,6 +4591,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(go_goroutines{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", @@ -4397,7 +4654,7 @@ "h": 8, "w": 12, "x": 0, - "y": 86 + "y": 165 }, "hiddenSeries": false, "id": 43, @@ -4421,7 +4678,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4431,6 +4688,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "max(go_gc_duration_seconds{job=~\"$job\", instance=~\"$instance\", quantile=\"1\"}) by(instance)", "format": "time_series", "intervalFactor": 2, @@ -4489,7 +4749,7 @@ "h": 8, "w": 12, "x": 12, - "y": 86 + "y": 165 }, "hiddenSeries": false, "id": 41, @@ -4513,7 +4773,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.5.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -4523,6 +4783,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(process_num_threads{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "intervalFactor": 2, @@ -4563,12 +4826,20 @@ } } ], + "targets": [ + { + "datasource": { + "uid": "$ds" + }, + "refId": "A" + } + ], "title": "Resource usage", "type": "row" } ], "refresh": "", - "schemaVersion": 35, + "schemaVersion": 36, "style": "dark", "tags": [ "vmagent", @@ -4579,8 +4850,8 @@ { "current": { "selected": true, - "text": "VM", - "value": "VM" + "text": "VictoriaMetrics", + "value": "VictoriaMetrics" }, "hide": 0, "includeAll": false, @@ -4595,14 +4866,14 @@ "type": "datasource" }, { - "allValue": "", + "allValue": ".*", "current": {}, "datasource": { "uid": "$ds" }, "definition": "label_values(vm_app_version{version=~\"^vmagent.*\"}, job)", "hide": 0, - "includeAll": false, + "includeAll": true, "multi": true, "name": "job", "options": [], @@ -4666,6 +4937,17 @@ "skipUrlSync": false, "sort": 0, "type": "query" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${ds}" + }, + "filters": [], + "hide": 0, + "name": "adhoc", + "skipUrlSync": false, + "type": "adhoc" } ] }, @@ -4687,7 +4969,7 @@ ] }, "timezone": "", - "title": "vmagent", + "title": "VictoriaMetrics - vmagent", "uid": "G7Z9GzMGz", "version": 1, "weekStart": "" diff --git a/dashboards/vmalert.json b/dashboards/vmalert.json index d4cf3ec11..057163bc5 100644 --- a/dashboards/vmalert.json +++ b/dashboards/vmalert.json @@ -5,7 +5,7 @@ "type": "grafana", "id": "grafana", "name": "Grafana", - "version": "8.3.5" + "version": "9.0.3" }, { "type": "panel", @@ -27,8 +27,8 @@ }, { "type": "panel", - "id": "table-old", - "name": "Table (old)", + "id": "table", + "name": "Table", "version": "" }, { @@ -42,7 +42,10 @@ "list": [ { "builtIn": 1, - "datasource": "-- Grafana --", + "datasource": { + "type": "datasource", + "uid": "grafana" + }, "enable": true, "hide": true, "iconColor": "rgba(0, 211, 255, 1)", @@ -62,7 +65,7 @@ "fiscalYearStartMonth": 0, "graphTooltip": 1, "id": null, - "iteration": 1644909221704, + "iteration": 1663341746917, "links": [ { "asDropdown": false, @@ -105,6 +108,10 @@ "panels": [ { "collapsed": false, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, "gridPos": { "h": 1, "w": 24, @@ -113,6 +120,15 @@ }, "id": 11, "panels": [], + "targets": [ + { + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "refId": "A" + } + ], "title": "General ($instance)", "type": "row" }, @@ -181,9 +197,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "count(vmalert_config_last_reload_successful{job=~\"$job\", instance=~\"$instance\"} < 1 ) or 0", "interval": "", @@ -220,7 +239,7 @@ }, "gridPos": { "h": 3, - "w": 5, + "w": 4, "x": 3, "y": 1 }, @@ -240,9 +259,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "(sum(vmalert_alerting_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) or vector(0)) + \n(sum(vmalert_recording_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) or vector(0))", "interval": "", @@ -275,8 +297,8 @@ }, "gridPos": { "h": 3, - "w": 5, - "x": 8, + "w": 4, + "x": 7, "y": 1 }, "id": 9, @@ -295,9 +317,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "count(vmalert_alerting_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"})", "interval": "", @@ -330,8 +355,8 @@ }, "gridPos": { "h": 3, - "w": 5, - "x": 13, + "w": 4, + "x": 11, "y": 1 }, "id": 7, @@ -350,9 +375,12 @@ "text": {}, "textMode": "auto" }, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "count(vmalert_recording_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"})", "interval": "", @@ -364,93 +392,97 @@ "type": "stat" }, { - "columns": [], "datasource": { + "type": "prometheus", "uid": "$ds" }, - "fontSize": "100%", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false, + "minWidth": 50 + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Time" + }, + "properties": [ + { + "id": "custom.hidden", + "value": true + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Count" + } + ] + } + ] + }, "gridPos": { - "h": 7, - "w": 6, - "x": 18, - "y": 1 + "h": 4, + "w": 9, + "x": 0, + "y": 4 }, - "id": 2, - "scroll": true, - "showHeader": true, - "sort": { - "col": 3, - "desc": false + "id": 45, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true }, - "styles": [ - { - "alias": "uptime", - "align": "auto", - "colorMode": "cell", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "Value", - "thresholds": [ - "1800", - "3600" - ], - "type": "number", - "unit": "s" - }, - { - "alias": "", - "align": "auto", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "instance", - "thresholds": [], - "type": "string", - "unit": "short" - }, - { - "alias": "", - "align": "auto", - "colors": [ - "rgba(245, 54, 54, 0.9)", - "rgba(237, 129, 40, 0.89)", - "rgba(50, 172, 45, 0.97)" - ], - "dateFormat": "YYYY-MM-DD HH:mm:ss", - "decimals": 2, - "mappingType": 1, - "pattern": "/.*/", - "thresholds": [], - "type": "hidden", - "unit": "short" - } - ], + "pluginVersion": "9.0.3", "targets": [ { + "datasource": { + "type": "prometheus", + "uid": "$ds" + }, + "editorMode": "code", "exemplar": false, - "expr": "sort((time() - vm_app_start_timestamp{job=~\"$job\", instance=~\"$instance\"}) or (up{job=~\"$job\", instance=~\"$instance\"}))", + "expr": "sum(vm_app_version{job=~\"$job\", instance=~\"$instance\"}) by(job, short_version)", "format": "table", - "hide": false, "instant": true, - "interval": "", - "legendFormat": "{{instance}}", + "range": false, "refId": "A" } ], - "title": "Uptime", - "transform": "table", - "type": "table-old" + "type": "table" }, { "aliasColors": {}, @@ -470,8 +502,8 @@ "fillGradient": 0, "gridPos": { "h": 4, - "w": 18, - "x": 0, + "w": 15, + "x": 9, "y": 4 }, "hiddenSeries": false, @@ -498,7 +530,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -508,6 +540,9 @@ "steppedLine": true, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sort(sum(up{job=~\"$job\", instance=~\"$instance\"}) by (job, instance))", "format": "time_series", @@ -562,7 +597,7 @@ "uid": "$ds" }, "description": "Shows the number of fired alerts by instance.", - "fill": 1, + "fill": 0, "fillGradient": 0, "gridPos": { "h": 8, @@ -591,7 +626,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -601,6 +636,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(increase(vmalert_alerts_fired_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", "interval": "", @@ -613,7 +651,7 @@ "title": "Alerts fired total", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -624,11 +662,13 @@ }, "yaxes": [ { + "$$hashKey": "object:62", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:63", "format": "short", "logBase": 1, "show": true @@ -682,7 +722,7 @@ "alertThreshold": false }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -692,6 +732,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(rate(vmalert_iteration_duration_seconds_sum{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}[$__rate_interval])) by(group) / \nsum(rate(vmalert_iteration_duration_seconds_count{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}[$__rate_interval])) by(group)", "interval": "", @@ -767,7 +810,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -777,6 +820,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(rate(vmalert_execution_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by (instance)", "interval": "", @@ -800,11 +846,13 @@ }, "yaxes": [ { + "$$hashKey": "object:182", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:183", "format": "short", "logBase": 1, "show": true @@ -820,6 +868,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Shows the error rate while executing configured rules. Non-zero value means there are some issues with existing rules. Check the logs to get more details.", @@ -852,7 +901,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -862,8 +911,11 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, - "expr": "sum(increase(vmalert_execution_errors_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", + "expr": "sum(increase(vmalert_execution_errors_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance) > 0", "interval": "", "legendFormat": "{{instance}}", "refId": "A" @@ -885,11 +937,13 @@ }, "yaxes": [ { + "$$hashKey": "object:244", "format": "short", "logBase": 1, "show": true }, { + "$$hashKey": "object:245", "format": "short", "logBase": 1, "show": true @@ -901,6 +955,10 @@ }, { "collapsed": true, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, "gridPos": { "h": 1, "w": 24, @@ -947,7 +1005,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -957,6 +1015,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(vmalert_alerts_firing{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, alertname) > 0", "interval": "", @@ -1032,7 +1093,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1042,6 +1103,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(vmalert_alerting_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, alertname) > 0", "interval": "", @@ -1117,7 +1181,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1127,6 +1191,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(vmalert_alerts_pending{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, alertname) > 0", "interval": "", @@ -1170,6 +1237,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Shows how many alerts are sent to Alertmanager per second. Only active alerts are sent.", @@ -1202,7 +1270,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1212,8 +1280,11 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, - "expr": "sum(rate(vmalert_alerts_sent_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance, addr)", + "expr": "sum(rate(vmalert_alerts_sent_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance, addr) > 0", "interval": "", "legendFormat": "{{instance}} => {{addr}}", "refId": "A" @@ -1224,7 +1295,7 @@ "title": "Requests rate to Alertmanager ($group)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -1259,6 +1330,7 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, "description": "Shows the error rate for the attempts to send alerts to Alertmanager. If not zero it means there issues on attempt to send notification to Alertmanager and some alerts may be not delivered properly. Check the logs for more details.", @@ -1291,7 +1363,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1301,8 +1373,11 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, - "expr": "sum(rate(vmalert_alerts_send_errors_total{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}[$__rate_interval])) by(instance, addr)", + "expr": "sum(rate(vmalert_alerts_send_errors_total{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}[$__rate_interval])) by(instance, addr) > 0", "interval": "", "legendFormat": "{{instance}} => {{addr}}", "refId": "A" @@ -1343,11 +1418,24 @@ } } ], + "targets": [ + { + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "refId": "A" + } + ], "title": "Alerting rules ($instance)", "type": "row" }, { "collapsed": true, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, "gridPos": { "h": 1, "w": 24, @@ -1371,7 +1459,7 @@ "h": 8, "w": 12, "x": 0, - "y": 26 + "y": 50 }, "hiddenSeries": false, "id": 31, @@ -1394,7 +1482,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1404,6 +1492,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "topk(10, sum(vmalert_recording_rules_last_evaluation_samples{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, recording) > 0)", "interval": "", @@ -1500,7 +1591,7 @@ "h": 8, "w": 12, "x": 12, - "y": 26 + "y": 50 }, "id": 33, "options": { @@ -1514,12 +1605,16 @@ "placement": "bottom" }, "tooltip": { - "mode": "single" + "mode": "single", + "sort": "none" } }, "pluginVersion": "8.0.3", "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(vmalert_recording_rules_last_evaluation_samples{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, recording) < 1", "interval": "", @@ -1544,7 +1639,7 @@ "h": 8, "w": 12, "x": 0, - "y": 34 + "y": 58 }, "hiddenSeries": false, "id": 30, @@ -1567,7 +1662,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.0.3", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1577,6 +1672,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(vmalert_recording_rules_error{job=~\"$job\", instance=~\"$instance\", group=~\"$group\"}) by(group, recording) > 0", "interval": "", @@ -1615,11 +1713,24 @@ } } ], + "targets": [ + { + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "refId": "A" + } + ], "title": "Recording rules ($instance)", "type": "row" }, { "collapsed": true, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, "gridPos": { "h": 1, "w": 24, @@ -1637,7 +1748,7 @@ "type": "prometheus", "uid": "$ds" }, - "description": "Shows the CPU usage per vmalert instance. \nIf you think that usage is abnormal or unexpected pls file an issue and attach CPU profile if possible.", + "description": "Shows the CPU usage percentage per vmalert instance. \nIf you think that usage is abnormal or unexpected pls file an issue and attach CPU profile if possible.", "fieldConfig": { "defaults": { "links": [] @@ -1650,7 +1761,7 @@ "h": 8, "w": 12, "x": 0, - "y": 27 + "y": 67 }, "hiddenSeries": false, "id": 35, @@ -1680,17 +1791,11 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [ - { - "$$hashKey": "object:61", - "alias": "/Limit .*/", - "color": "#F2495C" - } - ], + "seriesOverrides": [], "spaceLength": 10, "stack": false, "steppedLine": false, @@ -1701,26 +1806,12 @@ "uid": "$ds" }, "exemplar": false, - "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance)", + "expr": "sum(rate(process_cpu_seconds_total{job=~\"$job\", instance=~\"$instance\"}[$__rate_interval])) by(instance) / min(process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 1, "legendFormat": "{{instance}}", "refId": "A" - }, - { - "datasource": { - "type": "prometheus", - "uid": "$ds" - }, - "exemplar": false, - "expr": "process_cpu_cores_available{job=~\"$job\", instance=~\"$instance\"}", - "format": "time_series", - "hide": false, - "interval": "", - "intervalFactor": 1, - "legendFormat": "Limit ({{instance}})", - "refId": "B" } ], "thresholds": [], @@ -1728,7 +1819,7 @@ "title": "CPU ($instance)", "tooltip": { "shared": true, - "sort": 0, + "sort": 2, "value_type": "individual" }, "type": "graph", @@ -1739,12 +1830,14 @@ }, "yaxes": [ { - "format": "short", + "$$hashKey": "object:473", + "format": "percentunit", "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:474", "format": "short", "logBase": 1, "show": true @@ -1775,7 +1868,7 @@ "h": 8, "w": 12, "x": 12, - "y": 27 + "y": 67 }, "hiddenSeries": false, "id": 37, @@ -1805,7 +1898,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -1815,6 +1908,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(process_resident_memory_bytes{job=~\"$job\", instance=~\"$instance\"}) by (instance)", "interval": "", @@ -1822,6 +1918,9 @@ "refId": "A" }, { + "datasource": { + "uid": "$ds" + }, "exemplar": false, "expr": "sum(process_resident_memory_anon_bytes{job=~\"$job\", instance=~\"$instance\"}) by (instance)", "hide": false, @@ -1867,9 +1966,10 @@ "dashLength": 10, "dashes": false, "datasource": { + "type": "prometheus", "uid": "$ds" }, - "description": "Panel shows the number of open file descriptors in the OS.\nReaching the limit of open files can cause various issues and must be prevented.\n\nSee how to change limits here https://medium.com/@muhammadtriwibowo/set-permanently-ulimit-n-open-files-in-ubuntu-4d61064429a", + "description": "Panel shows the percentage of open file descriptors in the OS.\nReaching the limit of open files can cause various issues and must be prevented.\n\nSee how to change limits here https://medium.com/@muhammadtriwibowo/set-permanently-ulimit-n-open-files-in-ubuntu-4d61064429a", "fieldConfig": { "defaults": { "links": [] @@ -1882,7 +1982,7 @@ "h": 8, "w": 12, "x": 0, - "y": 35 + "y": 75 }, "hiddenSeries": false, "id": 39, @@ -1906,37 +2006,28 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", - "seriesOverrides": [ - { - "$$hashKey": "object:1161", - "alias": "max", - "color": "#C4162A" - } - ], + "seriesOverrides": [], "spaceLength": 10, "stack": false, "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, + "editorMode": "code", "exemplar": false, - "expr": "sum(process_open_fds{job=~\"$job\", instance=~\"$instance\"}) by (instance)", + "expr": "sum(process_open_fds{job=~\"$job\", instance=~\"$instance\"}) by (instance) \n/\nmin(process_max_fds{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", "intervalFactor": 2, "legendFormat": "open {{instance}}", + "range": true, "refId": "A" - }, - { - "expr": "min(process_max_fds{job=~\"$job\", instance=~\"$instance\"})", - "format": "time_series", - "interval": "", - "intervalFactor": 2, - "legendFormat": "max", - "refId": "B" } ], "thresholds": [], @@ -1955,13 +2046,15 @@ }, "yaxes": [ { - "decimals": 0, - "format": "short", - "logBase": 2, + "$$hashKey": "object:540", + "decimals": 3, + "format": "percentunit", + "logBase": 1, "min": "0", "show": true }, { + "$$hashKey": "object:541", "format": "short", "logBase": 1, "min": "0", @@ -1992,7 +2085,7 @@ "h": 8, "w": 12, "x": 12, - "y": 35 + "y": 75 }, "hiddenSeries": false, "id": 41, @@ -2016,7 +2109,7 @@ "alertThreshold": true }, "percentage": false, - "pluginVersion": "8.3.5", + "pluginVersion": "9.0.3", "pointradius": 2, "points": false, "renderer": "flot", @@ -2026,6 +2119,9 @@ "steppedLine": false, "targets": [ { + "datasource": { + "uid": "$ds" + }, "expr": "sum(go_goroutines{job=~\"$job\", instance=~\"$instance\"}) by(instance)", "format": "time_series", "interval": "", @@ -2067,12 +2163,21 @@ } } ], + "targets": [ + { + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "refId": "A" + } + ], "title": "Resource usage", "type": "row" } ], "refresh": false, - "schemaVersion": 34, + "schemaVersion": 36, "style": "dark", "tags": [ "victoriametrics", @@ -2162,6 +2267,17 @@ "skipUrlSync": false, "sort": 0, "type": "query" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${ds}" + }, + "filters": [], + "hide": 0, + "name": "adhoc", + "skipUrlSync": false, + "type": "adhoc" } ] }, @@ -2171,7 +2287,7 @@ }, "timepicker": {}, "timezone": "", - "title": "vmalert", + "title": "VictoriaMetrics - vmalert", "uid": "LzldHAVnz", "version": 1, "weekStart": "" From 2b55d167d787667017f6acf83bc3607aba7916db Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 14:26:18 +0300 Subject: [PATCH 13/18] app/vmagent/remotewrite: add benchmarks for comparing the performance of standard Snappy encoder with github.com/klauspost/compress/s2 encoder The standard Snappy encoder from github.com/golang/snappy shows quite good performance number for compressing the Prometheus remote_write proto messages according to the added benchmarks, so there is no need in switching to github.com/klauspost/compress/s2 yet. --- app/vmagent/remotewrite/pendingseries_test.go | 62 + .../remotewrite/pendingseries_timing_test.go | 36 + go.mod | 2 +- go.sum | 4 +- .../klauspost/compress/s2/.gitignore | 15 + .../github.com/klauspost/compress/s2/LICENSE | 28 + .../klauspost/compress/s2/README.md | 965 + .../klauspost/compress/s2/decode.go | 1046 + .../klauspost/compress/s2/decode_amd64.s | 568 + .../klauspost/compress/s2/decode_arm64.s | 574 + .../klauspost/compress/s2/decode_asm.go | 17 + .../klauspost/compress/s2/decode_other.go | 267 + .../klauspost/compress/s2/encode.go | 1341 ++ .../klauspost/compress/s2/encode_all.go | 456 + .../klauspost/compress/s2/encode_amd64.go | 142 + .../klauspost/compress/s2/encode_best.go | 630 + .../klauspost/compress/s2/encode_better.go | 431 + .../klauspost/compress/s2/encode_go.go | 307 + .../compress/s2/encodeblock_amd64.go | 191 + .../klauspost/compress/s2/encodeblock_amd64.s | 17779 ++++++++++++++++ .../github.com/klauspost/compress/s2/index.go | 598 + vendor/github.com/klauspost/compress/s2/s2.go | 143 + vendor/modules.txt | 3 +- 23 files changed, 25601 insertions(+), 4 deletions(-) create mode 100644 app/vmagent/remotewrite/pendingseries_test.go create mode 100644 app/vmagent/remotewrite/pendingseries_timing_test.go create mode 100644 vendor/github.com/klauspost/compress/s2/.gitignore create mode 100644 vendor/github.com/klauspost/compress/s2/LICENSE create mode 100644 vendor/github.com/klauspost/compress/s2/README.md create mode 100644 vendor/github.com/klauspost/compress/s2/decode.go create mode 100644 vendor/github.com/klauspost/compress/s2/decode_amd64.s create mode 100644 vendor/github.com/klauspost/compress/s2/decode_arm64.s create mode 100644 vendor/github.com/klauspost/compress/s2/decode_asm.go create mode 100644 vendor/github.com/klauspost/compress/s2/decode_other.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode_all.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode_amd64.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode_best.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode_better.go create mode 100644 vendor/github.com/klauspost/compress/s2/encode_go.go create mode 100644 vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go create mode 100644 vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s create mode 100644 vendor/github.com/klauspost/compress/s2/index.go create mode 100644 vendor/github.com/klauspost/compress/s2/s2.go diff --git a/app/vmagent/remotewrite/pendingseries_test.go b/app/vmagent/remotewrite/pendingseries_test.go new file mode 100644 index 000000000..74900a13b --- /dev/null +++ b/app/vmagent/remotewrite/pendingseries_test.go @@ -0,0 +1,62 @@ +package remotewrite + +import ( + "fmt" + "testing" + + "github.com/golang/snappy" + "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" +) + +func TestPushWriteRequest(t *testing.T) { + for _, rowsCount := range []int{1, 10, 100, 1e3, 1e4} { + t.Run(fmt.Sprintf("%d", rowsCount), func(t *testing.T) { + testPushWriteRequest(t, rowsCount) + }) + } +} + +func testPushWriteRequest(t *testing.T, rowsCount int) { + wr := newTestWriteRequest(rowsCount, 10) + pushBlockLen := 0 + pushBlock := func(block []byte) { + if pushBlockLen > 0 { + panic(fmt.Errorf("BUG: pushBlock called multiple times; pushBlockLen=%d at first call, len(block)=%d at second call", pushBlockLen, len(block))) + } + pushBlockLen = len(block) + } + pushWriteRequest(wr, pushBlock) + b := prompbmarshal.MarshalWriteRequest(nil, wr) + zb := snappy.Encode(nil, b) + maxPushBlockLen := len(zb) + minPushBlockLen := maxPushBlockLen / 2 + if pushBlockLen < minPushBlockLen { + t.Fatalf("unexpected block len after pushWriteRequest; got %d bytes; must be at least %d bytes", pushBlockLen, minPushBlockLen) + } + if pushBlockLen > maxPushBlockLen { + t.Fatalf("unexpected block len after pushWriteRequest; got %d bytes; must be smaller or equal to %d bytes", pushBlockLen, maxPushBlockLen) + } +} + +func newTestWriteRequest(seriesCount, labelsCount int) *prompbmarshal.WriteRequest { + var wr prompbmarshal.WriteRequest + for i := 0; i < seriesCount; i++ { + var labels []prompbmarshal.Label + for j := 0; j < labelsCount; j++ { + labels = append(labels, prompbmarshal.Label{ + Name: fmt.Sprintf("label_%d_%d", i, j), + Value: fmt.Sprintf("value_%d_%d", i, j), + }) + } + wr.Timeseries = append(wr.Timeseries, prompbmarshal.TimeSeries{ + Labels: labels, + Samples: []prompbmarshal.Sample{ + { + Value: float64(i), + Timestamp: 1000*int64(i), + }, + }, + }) + } + return &wr +} diff --git a/app/vmagent/remotewrite/pendingseries_timing_test.go b/app/vmagent/remotewrite/pendingseries_timing_test.go new file mode 100644 index 000000000..72bfcb1ff --- /dev/null +++ b/app/vmagent/remotewrite/pendingseries_timing_test.go @@ -0,0 +1,36 @@ +package remotewrite + +import ( + "fmt" + "testing" + + "github.com/golang/snappy" + "github.com/klauspost/compress/s2" + "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" +) + +func BenchmarkCompressWriteRequestSnappy(b *testing.B) { + b.Run("snappy", func(b *testing.B) { + benchmarkCompressWriteRequest(b, snappy.Encode) + }) + b.Run("s2", func(b *testing.B) { + benchmarkCompressWriteRequest(b, s2.EncodeSnappy) + }) +} + +func benchmarkCompressWriteRequest(b *testing.B, compressFunc func(dst, src []byte) []byte) { + for _, rowsCount := range []int{1, 10, 100, 1e3, 1e4} { + b.Run(fmt.Sprintf("rows_%d", rowsCount), func(b *testing.B) { + wr := newTestWriteRequest(rowsCount, 10) + data := prompbmarshal.MarshalWriteRequest(nil, wr) + b.ReportAllocs() + b.SetBytes(int64(rowsCount)) + b.RunParallel(func(pb *testing.PB) { + var zb []byte + for pb.Next() { + zb = compressFunc(zb[:cap(zb)], data) + } + }) + }) + } +} diff --git a/go.mod b/go.mod index 6b758edd9..b379cfb95 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.19 require ( cloud.google.com/go/storage v1.26.0 - github.com/VictoriaMetrics/fastcache v1.10.0 + github.com/VictoriaMetrics/fastcache v1.12.0 // Do not use the original github.com/valyala/fasthttp because of issues // like https://github.com/valyala/fasthttp/commit/996610f021ff45fdc98c2ce7884d5fa4e7f9199b diff --git a/go.sum b/go.sum index 656f8e70a..211548b25 100644 --- a/go.sum +++ b/go.sum @@ -104,8 +104,8 @@ github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdko github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= github.com/Shopify/sarama v1.19.0/go.mod h1:FVkBWblsNy7DGZRfXLU0O9RCGt5g3g3yEuWXgklEdEo= github.com/Shopify/toxiproxy v2.1.4+incompatible/go.mod h1:OXgGpZ6Cli1/URJOF1DMxUHB2q5Ap20/P/eIdh4G0pI= -github.com/VictoriaMetrics/fastcache v1.10.0 h1:5hDJnLsKLpnUEToub7ETuRu8RCkb40woBZAUiKonXzY= -github.com/VictoriaMetrics/fastcache v1.10.0/go.mod h1:tjiYeEfYXCqacuvYw/7UoDIeJaNxq6132xHICNP77w8= +github.com/VictoriaMetrics/fastcache v1.12.0 h1:vnVi/y9yKDcD9akmc4NqAoqgQhJrOwUF+j9LTgn4QDE= +github.com/VictoriaMetrics/fastcache v1.12.0/go.mod h1:tjiYeEfYXCqacuvYw/7UoDIeJaNxq6132xHICNP77w8= github.com/VictoriaMetrics/fasthttp v1.1.0 h1:3crd4YWHsMwu60GUXRH6OstowiFvqrwS4a/ueoLdLL0= github.com/VictoriaMetrics/fasthttp v1.1.0/go.mod h1:/7DMcogqd+aaD3G3Hg5kFgoFwlR2uydjiWvoLp5ZTqQ= github.com/VictoriaMetrics/metrics v1.18.1/go.mod h1:ArjwVz7WpgpegX/JpB0zpNF2h2232kErkEnzH1sxMmA= diff --git a/vendor/github.com/klauspost/compress/s2/.gitignore b/vendor/github.com/klauspost/compress/s2/.gitignore new file mode 100644 index 000000000..3a89c6e3e --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/.gitignore @@ -0,0 +1,15 @@ +testdata/bench + +# These explicitly listed benchmark data files are for an obsolete version of +# snappy_test.go. +testdata/alice29.txt +testdata/asyoulik.txt +testdata/fireworks.jpeg +testdata/geo.protodata +testdata/html +testdata/html_x_4 +testdata/kppkn.gtb +testdata/lcet10.txt +testdata/paper-100k.pdf +testdata/plrabn12.txt +testdata/urls.10K diff --git a/vendor/github.com/klauspost/compress/s2/LICENSE b/vendor/github.com/klauspost/compress/s2/LICENSE new file mode 100644 index 000000000..1d2d645bd --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/LICENSE @@ -0,0 +1,28 @@ +Copyright (c) 2011 The Snappy-Go Authors. All rights reserved. +Copyright (c) 2019 Klaus Post. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/vendor/github.com/klauspost/compress/s2/README.md b/vendor/github.com/klauspost/compress/s2/README.md new file mode 100644 index 000000000..73c0c462d --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/README.md @@ -0,0 +1,965 @@ +# S2 Compression + +S2 is an extension of [Snappy](https://github.com/google/snappy). + +S2 is aimed for high throughput, which is why it features concurrent compression for bigger payloads. + +Decoding is compatible with Snappy compressed content, but content compressed with S2 cannot be decompressed by Snappy. +This means that S2 can seamlessly replace Snappy without converting compressed content. + +S2 can produce Snappy compatible output, faster and better than Snappy. +If you want full benefit of the changes you should use s2 without Snappy compatibility. + +S2 is designed to have high throughput on content that cannot be compressed. +This is important, so you don't have to worry about spending CPU cycles on already compressed data. + +## Benefits over Snappy + +* Better compression +* Adjustable compression (3 levels) +* Concurrent stream compression +* Faster decompression, even for Snappy compatible content +* Concurrent Snappy/S2 stream decompression +* Ability to quickly skip forward in compressed stream +* Random seeking with indexes +* Compatible with reading Snappy compressed content +* Smaller block size overhead on incompressible blocks +* Block concatenation +* Uncompressed stream mode +* Automatic stream size padding +* Snappy compatible block compression + +## Drawbacks over Snappy + +* Not optimized for 32 bit systems +* Streams use slightly more memory due to larger blocks and concurrency (configurable) + +# Usage + +Installation: `go get -u github.com/klauspost/compress/s2` + +Full package documentation: + +[![godoc][1]][2] + +[1]: https://godoc.org/github.com/klauspost/compress?status.svg +[2]: https://godoc.org/github.com/klauspost/compress/s2 + +## Compression + +```Go +func EncodeStream(src io.Reader, dst io.Writer) error { + enc := s2.NewWriter(dst) + _, err := io.Copy(enc, src) + if err != nil { + enc.Close() + return err + } + // Blocks until compression is done. + return enc.Close() +} +``` + +You should always call `enc.Close()`, otherwise you will leak resources and your encode will be incomplete. + +For the best throughput, you should attempt to reuse the `Writer` using the `Reset()` method. + +The Writer in S2 is always buffered, therefore `NewBufferedWriter` in Snappy can be replaced with `NewWriter` in S2. +It is possible to flush any buffered data using the `Flush()` method. +This will block until all data sent to the encoder has been written to the output. + +S2 also supports the `io.ReaderFrom` interface, which will consume all input from a reader. + +As a final method to compress data, if you have a single block of data you would like to have encoded as a stream, +a slightly more efficient method is to use the `EncodeBuffer` method. +This will take ownership of the buffer until the stream is closed. + +```Go +func EncodeStream(src []byte, dst io.Writer) error { + enc := s2.NewWriter(dst) + // The encoder owns the buffer until Flush or Close is called. + err := enc.EncodeBuffer(buf) + if err != nil { + enc.Close() + return err + } + // Blocks until compression is done. + return enc.Close() +} +``` + +Each call to `EncodeBuffer` will result in discrete blocks being created without buffering, +so it should only be used a single time per stream. +If you need to write several blocks, you should use the regular io.Writer interface. + + +## Decompression + +```Go +func DecodeStream(src io.Reader, dst io.Writer) error { + dec := s2.NewReader(src) + _, err := io.Copy(dst, dec) + return err +} +``` + +Similar to the Writer, a Reader can be reused using the `Reset` method. + +For the best possible throughput, there is a `EncodeBuffer(buf []byte)` function available. +However, it requires that the provided buffer isn't used after it is handed over to S2 and until the stream is flushed or closed. + +For smaller data blocks, there is also a non-streaming interface: `Encode()`, `EncodeBetter()` and `Decode()`. +Do however note that these functions (similar to Snappy) does not provide validation of data, +so data corruption may be undetected. Stream encoding provides CRC checks of data. + +It is possible to efficiently skip forward in a compressed stream using the `Skip()` method. +For big skips the decompressor is able to skip blocks without decompressing them. + +## Single Blocks + +Similar to Snappy S2 offers single block compression. +Blocks do not offer the same flexibility and safety as streams, +but may be preferable for very small payloads, less than 100K. + +Using a simple `dst := s2.Encode(nil, src)` will compress `src` and return the compressed result. +It is possible to provide a destination buffer. +If the buffer has a capacity of `s2.MaxEncodedLen(len(src))` it will be used. +If not a new will be allocated. + +Alternatively `EncodeBetter`/`EncodeBest` can also be used for better, but slightly slower compression. + +Similarly to decompress a block you can use `dst, err := s2.Decode(nil, src)`. +Again an optional destination buffer can be supplied. +The `s2.DecodedLen(src)` can be used to get the minimum capacity needed. +If that is not satisfied a new buffer will be allocated. + +Block function always operate on a single goroutine since it should only be used for small payloads. + +# Commandline tools + +Some very simply commandline tools are provided; `s2c` for compression and `s2d` for decompression. + +Binaries can be downloaded on the [Releases Page](https://github.com/klauspost/compress/releases). + +Installing then requires Go to be installed. To install them, use: + +`go install github.com/klauspost/compress/s2/cmd/s2c@latest && go install github.com/klauspost/compress/s2/cmd/s2d@latest` + +To build binaries to the current folder use: + +`go build github.com/klauspost/compress/s2/cmd/s2c && go build github.com/klauspost/compress/s2/cmd/s2d` + + +## s2c + +``` +Usage: s2c [options] file1 file2 + +Compresses all files supplied as input separately. +Output files are written as 'filename.ext.s2' or 'filename.ext.snappy'. +By default output files will be overwritten. +Use - as the only file name to read from stdin and write to stdout. + +Wildcards are accepted: testdir/*.txt will compress all files in testdir ending with .txt +Directories can be wildcards as well. testdir/*/*.txt will match testdir/subdir/b.txt + +File names beginning with 'http://' and 'https://' will be downloaded and compressed. +Only http response code 200 is accepted. + +Options: + -bench int + Run benchmark n times. No output will be written + -blocksize string + Max block size. Examples: 64K, 256K, 1M, 4M. Must be power of two and <= 4MB (default "4M") + -c Write all output to stdout. Multiple input files will be concatenated + -cpu int + Compress using this amount of threads (default 32) + -faster + Compress faster, but with a minor compression loss + -help + Display help + -index + Add seek index (default true) + -o string + Write output to another file. Single input file only + -pad string + Pad size to a multiple of this value, Examples: 500, 64K, 256K, 1M, 4M, etc (default "1") + -q Don't write any output to terminal, except errors + -rm + Delete source file(s) after successful compression + -safe + Do not overwrite output files + -slower + Compress more, but a lot slower + -snappy + Generate Snappy compatible output stream + -verify + Verify written files + +``` + +## s2d + +``` +Usage: s2d [options] file1 file2 + +Decompresses all files supplied as input. Input files must end with '.s2' or '.snappy'. +Output file names have the extension removed. By default output files will be overwritten. +Use - as the only file name to read from stdin and write to stdout. + +Wildcards are accepted: testdir/*.txt will compress all files in testdir ending with .txt +Directories can be wildcards as well. testdir/*/*.txt will match testdir/subdir/b.txt + +File names beginning with 'http://' and 'https://' will be downloaded and decompressed. +Extensions on downloaded files are ignored. Only http response code 200 is accepted. + +Options: + -bench int + Run benchmark n times. No output will be written + -c Write all output to stdout. Multiple input files will be concatenated + -help + Display help + -o string + Write output to another file. Single input file only + -offset string + Start at offset. Examples: 92, 64K, 256K, 1M, 4M. Requires Index + -q Don't write any output to terminal, except errors + -rm + Delete source file(s) after successful decompression + -safe + Do not overwrite output files + -tail string + Return last of compressed file. Examples: 92, 64K, 256K, 1M, 4M. Requires Index + -verify + Verify files, but do not write output +``` + +## s2sx: self-extracting archives + +s2sx allows creating self-extracting archives with no dependencies. + +By default, executables are created for the same platforms as the host os, +but this can be overridden with `-os` and `-arch` parameters. + +Extracted files have 0666 permissions, except when untar option used. + +``` +Usage: s2sx [options] file1 file2 + +Compresses all files supplied as input separately. +If files have '.s2' extension they are assumed to be compressed already. +Output files are written as 'filename.s2sx' and with '.exe' for windows targets. +If output is big, an additional file with ".more" is written. This must be included as well. +By default output files will be overwritten. + +Wildcards are accepted: testdir/*.txt will compress all files in testdir ending with .txt +Directories can be wildcards as well. testdir/*/*.txt will match testdir/subdir/b.txt + +Options: + -arch string + Destination architecture (default "amd64") + -c Write all output to stdout. Multiple input files will be concatenated + -cpu int + Compress using this amount of threads (default 32) + -help + Display help + -max string + Maximum executable size. Rest will be written to another file. (default "1G") + -os string + Destination operating system (default "windows") + -q Don't write any output to terminal, except errors + -rm + Delete source file(s) after successful compression + -safe + Do not overwrite output files + -untar + Untar on destination +``` + +Available platforms are: + + * darwin-amd64 + * darwin-arm64 + * linux-amd64 + * linux-arm + * linux-arm64 + * linux-mips64 + * linux-ppc64le + * windows-386 + * windows-amd64 + +By default, there is a size limit of 1GB for the output executable. + +When this is exceeded the remaining file content is written to a file called +output+`.more`. This file must be included for a successful extraction and +placed alongside the executable for a successful extraction. + +This file *must* have the same name as the executable, so if the executable is renamed, +so must the `.more` file. + +This functionality is disabled with stdin/stdout. + +### Self-extracting TAR files + +If you wrap a TAR file you can specify `-untar` to make it untar on the destination host. + +Files are extracted to the current folder with the path specified in the tar file. + +Note that tar files are not validated before they are wrapped. + +For security reasons files that move below the root folder are not allowed. + +# Performance + +This section will focus on comparisons to Snappy. +This package is solely aimed at replacing Snappy as a high speed compression package. +If you are mainly looking for better compression [zstandard](https://github.com/klauspost/compress/tree/master/zstd#zstd) +gives better compression, but typically at speeds slightly below "better" mode in this package. + +Compression is increased compared to Snappy, mostly around 5-20% and the throughput is typically 25-40% increased (single threaded) compared to the Snappy Go implementation. + +Streams are concurrently compressed. The stream will be distributed among all available CPU cores for the best possible throughput. + +A "better" compression mode is also available. This allows to trade a bit of speed for a minor compression gain. +The content compressed in this mode is fully compatible with the standard decoder. + +Snappy vs S2 **compression** speed on 16 core (32 thread) computer, using all threads and a single thread (1 CPU): + +| File | S2 speed | S2 Throughput | S2 % smaller | S2 "better" | "better" throughput | "better" % smaller | +|-----------------------------------------------------------------------------------------------------|----------|---------------|--------------|-------------|---------------------|--------------------| +| [rawstudio-mint14.tar](https://files.klauspost.com/compress/rawstudio-mint14.7z) | 12.70x | 10556 MB/s | 7.35% | 4.15x | 3455 MB/s | 12.79% | +| (1 CPU) | 1.14x | 948 MB/s | - | 0.42x | 349 MB/s | - | +| [github-june-2days-2019.json](https://files.klauspost.com/compress/github-june-2days-2019.json.zst) | 17.13x | 14484 MB/s | 31.60% | 10.09x | 8533 MB/s | 37.71% | +| (1 CPU) | 1.33x | 1127 MB/s | - | 0.70x | 589 MB/s | - | +| [github-ranks-backup.bin](https://files.klauspost.com/compress/github-ranks-backup.bin.zst) | 15.14x | 12000 MB/s | -5.79% | 6.59x | 5223 MB/s | 5.80% | +| (1 CPU) | 1.11x | 877 MB/s | - | 0.47x | 370 MB/s | - | +| [consensus.db.10gb](https://files.klauspost.com/compress/consensus.db.10gb.zst) | 14.62x | 12116 MB/s | 15.90% | 5.35x | 4430 MB/s | 16.08% | +| (1 CPU) | 1.38x | 1146 MB/s | - | 0.38x | 312 MB/s | - | +| [adresser.json](https://files.klauspost.com/compress/adresser.json.zst) | 8.83x | 17579 MB/s | 43.86% | 6.54x | 13011 MB/s | 47.23% | +| (1 CPU) | 1.14x | 2259 MB/s | - | 0.74x | 1475 MB/s | - | +| [gob-stream](https://files.klauspost.com/compress/gob-stream.7z) | 16.72x | 14019 MB/s | 24.02% | 10.11x | 8477 MB/s | 30.48% | +| (1 CPU) | 1.24x | 1043 MB/s | - | 0.70x | 586 MB/s | - | +| [10gb.tar](http://mattmahoney.net/dc/10gb.html) | 13.33x | 9254 MB/s | 1.84% | 6.75x | 4686 MB/s | 6.72% | +| (1 CPU) | 0.97x | 672 MB/s | - | 0.53x | 366 MB/s | - | +| sharnd.out.2gb | 2.11x | 12639 MB/s | 0.01% | 1.98x | 11833 MB/s | 0.01% | +| (1 CPU) | 0.93x | 5594 MB/s | - | 1.34x | 8030 MB/s | - | +| [enwik9](http://mattmahoney.net/dc/textdata.html) | 19.34x | 8220 MB/s | 3.98% | 7.87x | 3345 MB/s | 15.82% | +| (1 CPU) | 1.06x | 452 MB/s | - | 0.50x | 213 MB/s | - | +| [silesia.tar](http://sun.aei.polsl.pl/~sdeor/corpus/silesia.zip) | 10.48x | 6124 MB/s | 5.67% | 3.76x | 2197 MB/s | 12.60% | +| (1 CPU) | 0.97x | 568 MB/s | - | 0.46x | 271 MB/s | - | +| [enwik10](https://encode.su/threads/3315-enwik10-benchmark-results) | 21.07x | 9020 MB/s | 6.36% | 6.91x | 2959 MB/s | 16.95% | +| (1 CPU) | 1.07x | 460 MB/s | - | 0.51x | 220 MB/s | - | + +### Legend + +* `S2 speed`: Speed of S2 compared to Snappy, using 16 cores and 1 core. +* `S2 throughput`: Throughput of S2 in MB/s. +* `S2 % smaller`: How many percent of the Snappy output size is S2 better. +* `S2 "better"`: Speed when enabling "better" compression mode in S2 compared to Snappy. +* `"better" throughput`: Speed when enabling "better" compression mode in S2 compared to Snappy. +* `"better" % smaller`: How many percent of the Snappy output size is S2 better when using "better" compression. + +There is a good speedup across the board when using a single thread and a significant speedup when using multiple threads. + +Machine generated data gets by far the biggest compression boost, with size being being reduced by up to 45% of Snappy size. + +The "better" compression mode sees a good improvement in all cases, but usually at a performance cost. + +Incompressible content (`sharnd.out.2gb`, 2GB random data) sees the smallest speedup. +This is likely dominated by synchronization overhead, which is confirmed by the fact that single threaded performance is higher (see above). + +## Decompression + +S2 attempts to create content that is also fast to decompress, except in "better" mode where the smallest representation is used. + +S2 vs Snappy **decompression** speed. Both operating on single core: + +| File | S2 Throughput | vs. Snappy | Better Throughput | vs. Snappy | +|-----------------------------------------------------------------------------------------------------|---------------|------------|-------------------|------------| +| [rawstudio-mint14.tar](https://files.klauspost.com/compress/rawstudio-mint14.7z) | 2117 MB/s | 1.14x | 1738 MB/s | 0.94x | +| [github-june-2days-2019.json](https://files.klauspost.com/compress/github-june-2days-2019.json.zst) | 2401 MB/s | 1.25x | 2307 MB/s | 1.20x | +| [github-ranks-backup.bin](https://files.klauspost.com/compress/github-ranks-backup.bin.zst) | 2075 MB/s | 0.98x | 1764 MB/s | 0.83x | +| [consensus.db.10gb](https://files.klauspost.com/compress/consensus.db.10gb.zst) | 2967 MB/s | 1.05x | 2885 MB/s | 1.02x | +| [adresser.json](https://files.klauspost.com/compress/adresser.json.zst) | 4141 MB/s | 1.07x | 4184 MB/s | 1.08x | +| [gob-stream](https://files.klauspost.com/compress/gob-stream.7z) | 2264 MB/s | 1.12x | 2185 MB/s | 1.08x | +| [10gb.tar](http://mattmahoney.net/dc/10gb.html) | 1525 MB/s | 1.03x | 1347 MB/s | 0.91x | +| sharnd.out.2gb | 3813 MB/s | 0.79x | 3900 MB/s | 0.81x | +| [enwik9](http://mattmahoney.net/dc/textdata.html) | 1246 MB/s | 1.29x | 967 MB/s | 1.00x | +| [silesia.tar](http://sun.aei.polsl.pl/~sdeor/corpus/silesia.zip) | 1433 MB/s | 1.12x | 1203 MB/s | 0.94x | +| [enwik10](https://encode.su/threads/3315-enwik10-benchmark-results) | 1284 MB/s | 1.32x | 1010 MB/s | 1.04x | + +### Legend + +* `S2 Throughput`: Decompression speed of S2 encoded content. +* `Better Throughput`: Decompression speed of S2 "better" encoded content. +* `vs Snappy`: Decompression speed of S2 "better" mode compared to Snappy and absolute speed. + + +While the decompression code hasn't changed, there is a significant speedup in decompression speed. +S2 prefers longer matches and will typically only find matches that are 6 bytes or longer. +While this reduces compression a bit, it improves decompression speed. + +The "better" compression mode will actively look for shorter matches, which is why it has a decompression speed quite similar to Snappy. + +Without assembly decompression is also very fast; single goroutine decompression speed. No assembly: + +| File | S2 Throughput | S2 throughput | +|--------------------------------|--------------|---------------| +| consensus.db.10gb.s2 | 1.84x | 2289.8 MB/s | +| 10gb.tar.s2 | 1.30x | 867.07 MB/s | +| rawstudio-mint14.tar.s2 | 1.66x | 1329.65 MB/s | +| github-june-2days-2019.json.s2 | 2.36x | 1831.59 MB/s | +| github-ranks-backup.bin.s2 | 1.73x | 1390.7 MB/s | +| enwik9.s2 | 1.67x | 681.53 MB/s | +| adresser.json.s2 | 3.41x | 4230.53 MB/s | +| silesia.tar.s2 | 1.52x | 811.58 | + +Even though S2 typically compresses better than Snappy, decompression speed is always better. + +### Concurrent Stream Decompression + +For full stream decompression S2 offers a [DecodeConcurrent](https://pkg.go.dev/github.com/klauspost/compress/s2#Reader.DecodeConcurrent) +that will decode a full stream using multiple goroutines. + +Example scaling, AMD Ryzen 3950X, 16 cores, decompression using `s2d -bench=3 `, best of 3: + +| Input | `-cpu=1` | `-cpu=2` | `-cpu=4` | `-cpu=8` | `-cpu=16` | +|-------------------------------------------|------------|------------|------------|------------|-------------| +| enwik10.snappy | 1098.6MB/s | 1819.8MB/s | 3625.6MB/s | 6910.6MB/s | 10818.2MB/s | +| enwik10.s2 | 1303.5MB/s | 2606.1MB/s | 4847.9MB/s | 8878.4MB/s | 9592.1MB/s | +| sofia-air-quality-dataset.tar.snappy | 1302.0MB/s | 2165.0MB/s | 4244.5MB/s | 8241.0MB/s | 12920.5MB/s | +| sofia-air-quality-dataset.tar.s2 | 1399.2MB/s | 2463.2MB/s | 5196.5MB/s | 9639.8MB/s | 11439.5MB/s | +| sofia-air-quality-dataset.tar.s2 (no asm) | 837.5MB/s | 1652.6MB/s | 3183.6MB/s | 5945.0MB/s | 9620.7MB/s | + +Scaling can be expected to be pretty linear until memory bandwidth is saturated. + +For now the DecodeConcurrent can only be used for full streams without seeking or combining with regular reads. + +## Block compression + + +When compressing blocks no concurrent compression is performed just as Snappy. +This is because blocks are for smaller payloads and generally will not benefit from concurrent compression. + +An important change is that incompressible blocks will not be more than at most 10 bytes bigger than the input. +In rare, worst case scenario Snappy blocks could be significantly bigger than the input. + +### Mixed content blocks + +The most reliable is a wide dataset. +For this we use [`webdevdata.org-2015-01-07-subset`](https://files.klauspost.com/compress/webdevdata.org-2015-01-07-4GB-subset.7z), +53927 files, total input size: 4,014,735,833 bytes. Single goroutine used. + +| * | Input | Output | Reduction | MB/s | +|-------------------|------------|------------|-----------|--------| +| S2 | 4014735833 | 1059723369 | 73.60% | **934.34** | +| S2 Better | 4014735833 | 969670507 | 75.85% | 532.70 | +| S2 Best | 4014735833 | 906625668 | **77.85%** | 46.84 | +| Snappy | 4014735833 | 1128706759 | 71.89% | 762.59 | +| S2, Snappy Output | 4014735833 | 1093821420 | 72.75% | 908.60 | +| LZ4 | 4014735833 | 1079259294 | 73.12% | 526.94 | + +S2 delivers both the best single threaded throughput with regular mode and the best compression rate with "best". +"Better" mode provides the same compression speed as LZ4 with better compression ratio. + +When outputting Snappy compatible output it still delivers better throughput (150MB/s more) and better compression. + +As can be seen from the other benchmarks decompression should also be easier on the S2 generated output. + +Though they cannot be compared due to different decompression speeds here are the speed/size comparisons for +other Go compressors: + +| * | Input | Output | Reduction | MB/s | +|-------------------|------------|------------|-----------|--------| +| Zstd Fastest (Go) | 4014735833 | 794608518 | 80.21% | 236.04 | +| Zstd Best (Go) | 4014735833 | 704603356 | 82.45% | 35.63 | +| Deflate (Go) l1 | 4014735833 | 871294239 | 78.30% | 214.04 | +| Deflate (Go) l9 | 4014735833 | 730389060 | 81.81% | 41.17 | + +### Standard block compression + +Benchmarking single block performance is subject to a lot more variation since it only tests a limited number of file patterns. +So individual benchmarks should only be seen as a guideline and the overall picture is more important. + +These micro-benchmarks are with data in cache and trained branch predictors. For a more realistic benchmark see the mixed content above. + +Block compression. Parallel benchmark running on 16 cores, 16 goroutines. + +AMD64 assembly is use for both S2 and Snappy. + +| Absolute Perf | Snappy size | S2 Size | Snappy Speed | S2 Speed | Snappy dec | S2 dec | +|-----------------------|-------------|---------|--------------|-------------|-------------|-------------| +| html | 22843 | 21111 | 16246 MB/s | 17438 MB/s | 40972 MB/s | 49263 MB/s | +| urls.10K | 335492 | 287326 | 7943 MB/s | 9693 MB/s | 22523 MB/s | 26484 MB/s | +| fireworks.jpeg | 123034 | 123100 | 349544 MB/s | 273889 MB/s | 718321 MB/s | 827552 MB/s | +| fireworks.jpeg (200B) | 146 | 155 | 8869 MB/s | 17773 MB/s | 33691 MB/s | 52421 MB/s | +| paper-100k.pdf | 85304 | 84459 | 167546 MB/s | 101263 MB/s | 326905 MB/s | 291944 MB/s | +| html_x_4 | 92234 | 21113 | 15194 MB/s | 50670 MB/s | 30843 MB/s | 32217 MB/s | +| alice29.txt | 88034 | 85975 | 5936 MB/s | 6139 MB/s | 12882 MB/s | 20044 MB/s | +| asyoulik.txt | 77503 | 79650 | 5517 MB/s | 6366 MB/s | 12735 MB/s | 22806 MB/s | +| lcet10.txt | 234661 | 220670 | 6235 MB/s | 6067 MB/s | 14519 MB/s | 18697 MB/s | +| plrabn12.txt | 319267 | 317985 | 5159 MB/s | 5726 MB/s | 11923 MB/s | 19901 MB/s | +| geo.protodata | 23335 | 18690 | 21220 MB/s | 26529 MB/s | 56271 MB/s | 62540 MB/s | +| kppkn.gtb | 69526 | 65312 | 9732 MB/s | 8559 MB/s | 18491 MB/s | 18969 MB/s | +| alice29.txt (128B) | 80 | 82 | 6691 MB/s | 15489 MB/s | 31883 MB/s | 38874 MB/s | +| alice29.txt (1000B) | 774 | 774 | 12204 MB/s | 13000 MB/s | 48056 MB/s | 52341 MB/s | +| alice29.txt (10000B) | 6648 | 6933 | 10044 MB/s | 12806 MB/s | 32378 MB/s | 46322 MB/s | +| alice29.txt (20000B) | 12686 | 13574 | 7733 MB/s | 11210 MB/s | 30566 MB/s | 58969 MB/s | + + +| Relative Perf | Snappy size | S2 size improved | S2 Speed | S2 Dec Speed | +|-----------------------|-------------|------------------|----------|--------------| +| html | 22.31% | 7.58% | 1.07x | 1.20x | +| urls.10K | 47.78% | 14.36% | 1.22x | 1.18x | +| fireworks.jpeg | 99.95% | -0.05% | 0.78x | 1.15x | +| fireworks.jpeg (200B) | 73.00% | -6.16% | 2.00x | 1.56x | +| paper-100k.pdf | 83.30% | 0.99% | 0.60x | 0.89x | +| html_x_4 | 22.52% | 77.11% | 3.33x | 1.04x | +| alice29.txt | 57.88% | 2.34% | 1.03x | 1.56x | +| asyoulik.txt | 61.91% | -2.77% | 1.15x | 1.79x | +| lcet10.txt | 54.99% | 5.96% | 0.97x | 1.29x | +| plrabn12.txt | 66.26% | 0.40% | 1.11x | 1.67x | +| geo.protodata | 19.68% | 19.91% | 1.25x | 1.11x | +| kppkn.gtb | 37.72% | 6.06% | 0.88x | 1.03x | +| alice29.txt (128B) | 62.50% | -2.50% | 2.31x | 1.22x | +| alice29.txt (1000B) | 77.40% | 0.00% | 1.07x | 1.09x | +| alice29.txt (10000B) | 66.48% | -4.29% | 1.27x | 1.43x | +| alice29.txt (20000B) | 63.43% | -7.00% | 1.45x | 1.93x | + +Speed is generally at or above Snappy. Small blocks gets a significant speedup, although at the expense of size. + +Decompression speed is better than Snappy, except in one case. + +Since payloads are very small the variance in terms of size is rather big, so they should only be seen as a general guideline. + +Size is on average around Snappy, but varies on content type. +In cases where compression is worse, it usually is compensated by a speed boost. + + +### Better compression + +Benchmarking single block performance is subject to a lot more variation since it only tests a limited number of file patterns. +So individual benchmarks should only be seen as a guideline and the overall picture is more important. + +| Absolute Perf | Snappy size | Better Size | Snappy Speed | Better Speed | Snappy dec | Better dec | +|-----------------------|-------------|-------------|--------------|--------------|-------------|-------------| +| html | 22843 | 19833 | 16246 MB/s | 7731 MB/s | 40972 MB/s | 40292 MB/s | +| urls.10K | 335492 | 253529 | 7943 MB/s | 3980 MB/s | 22523 MB/s | 20981 MB/s | +| fireworks.jpeg | 123034 | 123100 | 349544 MB/s | 9760 MB/s | 718321 MB/s | 823698 MB/s | +| fireworks.jpeg (200B) | 146 | 142 | 8869 MB/s | 594 MB/s | 33691 MB/s | 30101 MB/s | +| paper-100k.pdf | 85304 | 82915 | 167546 MB/s | 7470 MB/s | 326905 MB/s | 198869 MB/s | +| html_x_4 | 92234 | 19841 | 15194 MB/s | 23403 MB/s | 30843 MB/s | 30937 MB/s | +| alice29.txt | 88034 | 73218 | 5936 MB/s | 2945 MB/s | 12882 MB/s | 16611 MB/s | +| asyoulik.txt | 77503 | 66844 | 5517 MB/s | 2739 MB/s | 12735 MB/s | 14975 MB/s | +| lcet10.txt | 234661 | 190589 | 6235 MB/s | 3099 MB/s | 14519 MB/s | 16634 MB/s | +| plrabn12.txt | 319267 | 270828 | 5159 MB/s | 2600 MB/s | 11923 MB/s | 13382 MB/s | +| geo.protodata | 23335 | 18278 | 21220 MB/s | 11208 MB/s | 56271 MB/s | 57961 MB/s | +| kppkn.gtb | 69526 | 61851 | 9732 MB/s | 4556 MB/s | 18491 MB/s | 16524 MB/s | +| alice29.txt (128B) | 80 | 81 | 6691 MB/s | 529 MB/s | 31883 MB/s | 34225 MB/s | +| alice29.txt (1000B) | 774 | 748 | 12204 MB/s | 1943 MB/s | 48056 MB/s | 42068 MB/s | +| alice29.txt (10000B) | 6648 | 6234 | 10044 MB/s | 2949 MB/s | 32378 MB/s | 28813 MB/s | +| alice29.txt (20000B) | 12686 | 11584 | 7733 MB/s | 2822 MB/s | 30566 MB/s | 27315 MB/s | + + +| Relative Perf | Snappy size | Better size | Better Speed | Better dec | +|-----------------------|-------------|-------------|--------------|------------| +| html | 22.31% | 13.18% | 0.48x | 0.98x | +| urls.10K | 47.78% | 24.43% | 0.50x | 0.93x | +| fireworks.jpeg | 99.95% | -0.05% | 0.03x | 1.15x | +| fireworks.jpeg (200B) | 73.00% | 2.74% | 0.07x | 0.89x | +| paper-100k.pdf | 83.30% | 2.80% | 0.07x | 0.61x | +| html_x_4 | 22.52% | 78.49% | 0.04x | 1.00x | +| alice29.txt | 57.88% | 16.83% | 1.54x | 1.29x | +| asyoulik.txt | 61.91% | 13.75% | 0.50x | 1.18x | +| lcet10.txt | 54.99% | 18.78% | 0.50x | 1.15x | +| plrabn12.txt | 66.26% | 15.17% | 0.50x | 1.12x | +| geo.protodata | 19.68% | 21.67% | 0.50x | 1.03x | +| kppkn.gtb | 37.72% | 11.04% | 0.53x | 0.89x | +| alice29.txt (128B) | 62.50% | -1.25% | 0.47x | 1.07x | +| alice29.txt (1000B) | 77.40% | 3.36% | 0.08x | 0.88x | +| alice29.txt (10000B) | 66.48% | 6.23% | 0.16x | 0.89x | +| alice29.txt (20000B) | 63.43% | 8.69% | 0.29x | 0.89x | + +Except for the mostly incompressible JPEG image compression is better and usually in the +double digits in terms of percentage reduction over Snappy. + +The PDF sample shows a significant slowdown compared to Snappy, as this mode tries harder +to compress the data. Very small blocks are also not favorable for better compression, so throughput is way down. + +This mode aims to provide better compression at the expense of performance and achieves that +without a huge performance penalty, except on very small blocks. + +Decompression speed suffers a little compared to the regular S2 mode, +but still manages to be close to Snappy in spite of increased compression. + +# Best compression mode + +S2 offers a "best" compression mode. + +This will compress as much as possible with little regard to CPU usage. + +Mainly for offline compression, but where decompression speed should still +be high and compatible with other S2 compressed data. + +Some examples compared on 16 core CPU, amd64 assembly used: + +``` +* enwik10 +Default... 10000000000 -> 4761467548 [47.61%]; 1.098s, 8685.6MB/s +Better... 10000000000 -> 4219438251 [42.19%]; 1.925s, 4954.2MB/s +Best... 10000000000 -> 3627364337 [36.27%]; 43.051s, 221.5MB/s + +* github-june-2days-2019.json +Default... 6273951764 -> 1043196283 [16.63%]; 431ms, 13882.3MB/s +Better... 6273951764 -> 949146808 [15.13%]; 547ms, 10938.4MB/s +Best... 6273951764 -> 832855506 [13.27%]; 9.455s, 632.8MB/s + +* nyc-taxi-data-10M.csv +Default... 3325605752 -> 1095998837 [32.96%]; 324ms, 9788.7MB/s +Better... 3325605752 -> 954776589 [28.71%]; 491ms, 6459.4MB/s +Best... 3325605752 -> 779098746 [23.43%]; 8.29s, 382.6MB/s + +* 10gb.tar +Default... 10065157632 -> 5916578242 [58.78%]; 1.028s, 9337.4MB/s +Better... 10065157632 -> 5649207485 [56.13%]; 1.597s, 6010.6MB/s +Best... 10065157632 -> 5208719802 [51.75%]; 32.78s, 292.8MB/ + +* consensus.db.10gb +Default... 10737418240 -> 4562648848 [42.49%]; 882ms, 11610.0MB/s +Better... 10737418240 -> 4542428129 [42.30%]; 1.533s, 6679.7MB/s +Best... 10737418240 -> 4244773384 [39.53%]; 42.96s, 238.4MB/s +``` + +Decompression speed should be around the same as using the 'better' compression mode. + +# Snappy Compatibility + +S2 now offers full compatibility with Snappy. + +This means that the efficient encoders of S2 can be used to generate fully Snappy compatible output. + +There is a [snappy](https://github.com/klauspost/compress/tree/master/snappy) package that can be used by +simply changing imports from `github.com/golang/snappy` to `github.com/klauspost/compress/snappy`. +This uses "better" mode for all operations. +If you would like more control, you can use the s2 package as described below: + +## Blocks + +Snappy compatible blocks can be generated with the S2 encoder. +Compression and speed is typically a bit better `MaxEncodedLen` is also smaller for smaller memory usage. Replace + +| Snappy | S2 replacement | +|----------------------------|-------------------------| +| snappy.Encode(...) | s2.EncodeSnappy(...) | +| snappy.MaxEncodedLen(...) | s2.MaxEncodedLen(...) | + +`s2.EncodeSnappy` can be replaced with `s2.EncodeSnappyBetter` or `s2.EncodeSnappyBest` to get more efficiently compressed snappy compatible output. + +`s2.ConcatBlocks` is compatible with snappy blocks. + +Comparison of [`webdevdata.org-2015-01-07-subset`](https://files.klauspost.com/compress/webdevdata.org-2015-01-07-4GB-subset.7z), +53927 files, total input size: 4,014,735,833 bytes. amd64, single goroutine used: + +| Encoder | Size | MB/s | Reduction | +|-----------------------|------------|------------|------------ +| snappy.Encode | 1128706759 | 725.59 | 71.89% | +| s2.EncodeSnappy | 1093823291 | **899.16** | 72.75% | +| s2.EncodeSnappyBetter | 1001158548 | 578.49 | 75.06% | +| s2.EncodeSnappyBest | 944507998 | 66.00 | **76.47%**| + +## Streams + +For streams, replace `enc = snappy.NewBufferedWriter(w)` with `enc = s2.NewWriter(w, s2.WriterSnappyCompat())`. +All other options are available, but note that block size limit is different for snappy. + +Comparison of different streams, AMD Ryzen 3950x, 16 cores. Size and throughput: + +| File | snappy.NewWriter | S2 Snappy | S2 Snappy, Better | S2 Snappy, Best | +|-----------------------------|--------------------------|---------------------------|--------------------------|-------------------------| +| nyc-taxi-data-10M.csv | 1316042016 - 539.47MB/s | 1307003093 - 10132.73MB/s | 1174534014 - 5002.44MB/s | 1115904679 - 177.97MB/s | +| enwik10 (xml) | 5088294643 - 451.13MB/s | 5175840939 - 9440.69MB/s | 4560784526 - 4487.21MB/s | 4340299103 - 158.92MB/s | +| 10gb.tar (mixed) | 6056946612 - 729.73MB/s | 6208571995 - 9978.05MB/s | 5741646126 - 4919.98MB/s | 5548973895 - 180.44MB/s | +| github-june-2days-2019.json | 1525176492 - 933.00MB/s | 1476519054 - 13150.12MB/s | 1400547532 - 5803.40MB/s | 1321887137 - 204.29MB/s | +| consensus.db.10gb (db) | 5412897703 - 1102.14MB/s | 5354073487 - 13562.91MB/s | 5335069899 - 5294.73MB/s | 5201000954 - 175.72MB/s | + +# Decompression + +All decompression functions map directly to equivalent s2 functions. + +| Snappy | S2 replacement | +|------------------------|--------------------| +| snappy.Decode(...) | s2.Decode(...) | +| snappy.DecodedLen(...) | s2.DecodedLen(...) | +| snappy.NewReader(...) | s2.NewReader(...) | + +Features like [quick forward skipping without decompression](https://pkg.go.dev/github.com/klauspost/compress/s2#Reader.Skip) +are also available for Snappy streams. + +If you know you are only decompressing snappy streams, setting [`ReaderMaxBlockSize(64<<10)`](https://pkg.go.dev/github.com/klauspost/compress/s2#ReaderMaxBlockSize) +on your Reader will reduce memory consumption. + +# Concatenating blocks and streams. + +Concatenating streams will concatenate the output of both without recompressing them. +While this is inefficient in terms of compression it might be usable in certain scenarios. +The 10 byte 'stream identifier' of the second stream can optionally be stripped, but it is not a requirement. + +Blocks can be concatenated using the `ConcatBlocks` function. + +Snappy blocks/streams can safely be concatenated with S2 blocks and streams. +Streams with indexes (see below) will currently not work on concatenated streams. + +# Stream Seek Index + +S2 and Snappy streams can have indexes. These indexes will allow random seeking within the compressed data. + +The index can either be appended to the stream as a skippable block or returned for separate storage. + +When the index is appended to a stream it will be skipped by regular decoders, +so the output remains compatible with other decoders. + +## Creating an Index + +To automatically add an index to a stream, add `WriterAddIndex()` option to your writer. +Then the index will be added to the stream when `Close()` is called. + +``` + // Add Index to stream... + enc := s2.NewWriter(w, s2.WriterAddIndex()) + io.Copy(enc, r) + enc.Close() +``` + +If you want to store the index separately, you can use `CloseIndex()` instead of the regular `Close()`. +This will return the index. Note that `CloseIndex()` should only be called once, and you shouldn't call `Close()`. + +``` + // Get index for separate storage... + enc := s2.NewWriter(w) + io.Copy(enc, r) + index, err := enc.CloseIndex() +``` + +The `index` can then be used needing to read from the stream. +This means the index can be used without needing to seek to the end of the stream +or for manually forwarding streams. See below. + +Finally, an existing S2/Snappy stream can be indexed using the `s2.IndexStream(r io.Reader)` function. + +## Using Indexes + +To use indexes there is a `ReadSeeker(random bool, index []byte) (*ReadSeeker, error)` function available. + +Calling ReadSeeker will return an [io.ReadSeeker](https://pkg.go.dev/io#ReadSeeker) compatible version of the reader. + +If 'random' is specified the returned io.Seeker can be used for random seeking, otherwise only forward seeking is supported. +Enabling random seeking requires the original input to support the [io.Seeker](https://pkg.go.dev/io#Seeker) interface. + +``` + dec := s2.NewReader(r) + rs, err := dec.ReadSeeker(false, nil) + rs.Seek(wantOffset, io.SeekStart) +``` + +Get a seeker to seek forward. Since no index is provided, the index is read from the stream. +This requires that an index was added and that `r` supports the [io.Seeker](https://pkg.go.dev/io#Seeker) interface. + +A custom index can be specified which will be used if supplied. +When using a custom index, it will not be read from the input stream. + +``` + dec := s2.NewReader(r) + rs, err := dec.ReadSeeker(false, index) + rs.Seek(wantOffset, io.SeekStart) +``` + +This will read the index from `index`. Since we specify non-random (forward only) seeking `r` does not have to be an io.Seeker + +``` + dec := s2.NewReader(r) + rs, err := dec.ReadSeeker(true, index) + rs.Seek(wantOffset, io.SeekStart) +``` + +Finally, since we specify that we want to do random seeking `r` must be an io.Seeker. + +The returned [ReadSeeker](https://pkg.go.dev/github.com/klauspost/compress/s2#ReadSeeker) contains a shallow reference to the existing Reader, +meaning changes performed to one is reflected in the other. + +To check if a stream contains an index at the end, the `(*Index).LoadStream(rs io.ReadSeeker) error` can be used. + +## Manually Forwarding Streams + +Indexes can also be read outside the decoder using the [Index](https://pkg.go.dev/github.com/klauspost/compress/s2#Index) type. +This can be used for parsing indexes, either separate or in streams. + +In some cases it may not be possible to serve a seekable stream. +This can for instance be an HTTP stream, where the Range request +is sent at the start of the stream. + +With a little bit of extra code it is still possible to use indexes +to forward to specific offset with a single forward skip. + +It is possible to load the index manually like this: +``` + var index s2.Index + _, err = index.Load(idxBytes) +``` + +This can be used to figure out how much to offset the compressed stream: + +``` + compressedOffset, uncompressedOffset, err := index.Find(wantOffset) +``` + +The `compressedOffset` is the number of bytes that should be skipped +from the beginning of the compressed file. + +The `uncompressedOffset` will then be offset of the uncompressed bytes returned +when decoding from that position. This will always be <= wantOffset. + +When creating a decoder it must be specified that it should *not* expect a stream identifier +at the beginning of the stream. Assuming the io.Reader `r` has been forwarded to `compressedOffset` +we create the decoder like this: + +``` + dec := s2.NewReader(r, s2.ReaderIgnoreStreamIdentifier()) +``` + +We are not completely done. We still need to forward the stream the uncompressed bytes we didn't want. +This is done using the regular "Skip" function: + +``` + err = dec.Skip(wantOffset - uncompressedOffset) +``` + +This will ensure that we are at exactly the offset we want, and reading from `dec` will start at the requested offset. + +## Index Format: + +Each block is structured as a snappy skippable block, with the chunk ID 0x99. + +The block can be read from the front, but contains information so it can be read from the back as well. + +Numbers are stored as fixed size little endian values or [zigzag encoded](https://developers.google.com/protocol-buffers/docs/encoding#signed_integers) [base 128 varints](https://developers.google.com/protocol-buffers/docs/encoding), +with un-encoded value length of 64 bits, unless other limits are specified. + +| Content | Format | +|---------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------| +| ID, `[1]byte` | Always 0x99. | +| Data Length, `[3]byte` | 3 byte little-endian length of the chunk in bytes, following this. | +| Header `[6]byte` | Header, must be `[115, 50, 105, 100, 120, 0]` or in text: "s2idx\x00". | +| UncompressedSize, Varint | Total Uncompressed size. | +| CompressedSize, Varint | Total Compressed size if known. Should be -1 if unknown. | +| EstBlockSize, Varint | Block Size, used for guessing uncompressed offsets. Must be >= 0. | +| Entries, Varint | Number of Entries in index, must be < 65536 and >=0. | +| HasUncompressedOffsets `byte` | 0 if no uncompressed offsets are present, 1 if present. Other values are invalid. | +| UncompressedOffsets, [Entries]VarInt | Uncompressed offsets. See below how to decode. | +| CompressedOffsets, [Entries]VarInt | Compressed offsets. See below how to decode. | +| Block Size, `[4]byte` | Little Endian total encoded size (including header and trailer). Can be used for searching backwards to start of block. | +| Trailer `[6]byte` | Trailer, must be `[0, 120, 100, 105, 50, 115]` or in text: "\x00xdi2s". Can be used for identifying block from end of stream. | + +For regular streams the uncompressed offsets are fully predictable, +so `HasUncompressedOffsets` allows to specify that compressed blocks all have +exactly `EstBlockSize` bytes of uncompressed content. + +Entries *must* be in order, starting with the lowest offset, +and there *must* be no uncompressed offset duplicates. +Entries *may* point to the start of a skippable block, +but it is then not allowed to also have an entry for the next block since +that would give an uncompressed offset duplicate. + +There is no requirement for all blocks to be represented in the index. +In fact there is a maximum of 65536 block entries in an index. + +The writer can use any method to reduce the number of entries. +An implicit block start at 0,0 can be assumed. + +### Decoding entries: + +``` +// Read Uncompressed entries. +// Each assumes EstBlockSize delta from previous. +for each entry { + uOff = 0 + if HasUncompressedOffsets == 1 { + uOff = ReadVarInt // Read value from stream + } + + // Except for the first entry, use previous values. + if entryNum == 0 { + entry[entryNum].UncompressedOffset = uOff + continue + } + + // Uncompressed uses previous offset and adds EstBlockSize + entry[entryNum].UncompressedOffset = entry[entryNum-1].UncompressedOffset + EstBlockSize + uOff +} + + +// Guess that the first block will be 50% of uncompressed size. +// Integer truncating division must be used. +CompressGuess := EstBlockSize / 2 + +// Read Compressed entries. +// Each assumes CompressGuess delta from previous. +// CompressGuess is adjusted for each value. +for each entry { + cOff = ReadVarInt // Read value from stream + + // Except for the first entry, use previous values. + if entryNum == 0 { + entry[entryNum].CompressedOffset = cOff + continue + } + + // Compressed uses previous and our estimate. + entry[entryNum].CompressedOffset = entry[entryNum-1].CompressedOffset + CompressGuess + cOff + + // Adjust compressed offset for next loop, integer truncating division must be used. + CompressGuess += cOff/2 +} +``` + +To decode from any given uncompressed offset `(wantOffset)`: + +* Iterate entries until `entry[n].UncompressedOffset > wantOffset`. +* Start decoding from `entry[n-1].CompressedOffset`. +* Discard `entry[n-1].UncompressedOffset - wantOffset` bytes from the decoded stream. + +See [using indexes](https://github.com/klauspost/compress/tree/master/s2#using-indexes) for functions that perform the operations with a simpler interface. + +# Format Extensions + +* Frame [Stream identifier](https://github.com/google/snappy/blob/master/framing_format.txt#L68) changed from `sNaPpY` to `S2sTwO`. +* [Framed compressed blocks](https://github.com/google/snappy/blob/master/format_description.txt) can be up to 4MB (up from 64KB). +* Compressed blocks can have an offset of `0`, which indicates to repeat the last seen offset. + +Repeat offsets must be encoded as a [2.2.1. Copy with 1-byte offset (01)](https://github.com/google/snappy/blob/master/format_description.txt#L89), where the offset is 0. + +The length is specified by reading the 3-bit length specified in the tag and decode using this table: + +| Length | Actual Length | +|--------|----------------------| +| 0 | 4 | +| 1 | 5 | +| 2 | 6 | +| 3 | 7 | +| 4 | 8 | +| 5 | 8 + read 1 byte | +| 6 | 260 + read 2 bytes | +| 7 | 65540 + read 3 bytes | + +This allows any repeat offset + length to be represented by 2 to 5 bytes. + +Lengths are stored as little endian values. + +The first copy of a block cannot be a repeat offset and the offset is not carried across blocks in streams. + +Default streaming block size is 1MB. + +# LICENSE + +This code is based on the [Snappy-Go](https://github.com/golang/snappy) implementation. + +Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. diff --git a/vendor/github.com/klauspost/compress/s2/decode.go b/vendor/github.com/klauspost/compress/s2/decode.go new file mode 100644 index 000000000..27c0f3c2c --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/decode.go @@ -0,0 +1,1046 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "encoding/binary" + "errors" + "fmt" + "io" + "io/ioutil" + "math" + "runtime" + "sync" +) + +var ( + // ErrCorrupt reports that the input is invalid. + ErrCorrupt = errors.New("s2: corrupt input") + // ErrCRC reports that the input failed CRC validation (streams only) + ErrCRC = errors.New("s2: corrupt input, crc mismatch") + // ErrTooLarge reports that the uncompressed length is too large. + ErrTooLarge = errors.New("s2: decoded block is too large") + // ErrUnsupported reports that the input isn't supported. + ErrUnsupported = errors.New("s2: unsupported input") +) + +// ErrCantSeek is returned if the stream cannot be seeked. +type ErrCantSeek struct { + Reason string +} + +// Error returns the error as string. +func (e ErrCantSeek) Error() string { + return fmt.Sprintf("s2: Can't seek because %s", e.Reason) +} + +// DecodedLen returns the length of the decoded block. +func DecodedLen(src []byte) (int, error) { + v, _, err := decodedLen(src) + return v, err +} + +// decodedLen returns the length of the decoded block and the number of bytes +// that the length header occupied. +func decodedLen(src []byte) (blockLen, headerLen int, err error) { + v, n := binary.Uvarint(src) + if n <= 0 || v > 0xffffffff { + return 0, 0, ErrCorrupt + } + + const wordSize = 32 << (^uint(0) >> 32 & 1) + if wordSize == 32 && v > 0x7fffffff { + return 0, 0, ErrTooLarge + } + return int(v), n, nil +} + +const ( + decodeErrCodeCorrupt = 1 +) + +// Decode returns the decoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire decoded block. +// Otherwise, a newly allocated slice will be returned. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +func Decode(dst, src []byte) ([]byte, error) { + dLen, s, err := decodedLen(src) + if err != nil { + return nil, err + } + if dLen <= cap(dst) { + dst = dst[:dLen] + } else { + dst = make([]byte, dLen) + } + if s2Decode(dst, src[s:]) != 0 { + return nil, ErrCorrupt + } + return dst, nil +} + +// NewReader returns a new Reader that decompresses from r, using the framing +// format described at +// https://github.com/google/snappy/blob/master/framing_format.txt with S2 changes. +func NewReader(r io.Reader, opts ...ReaderOption) *Reader { + nr := Reader{ + r: r, + maxBlock: maxBlockSize, + } + for _, opt := range opts { + if err := opt(&nr); err != nil { + nr.err = err + return &nr + } + } + nr.maxBufSize = MaxEncodedLen(nr.maxBlock) + checksumSize + if nr.lazyBuf > 0 { + nr.buf = make([]byte, MaxEncodedLen(nr.lazyBuf)+checksumSize) + } else { + nr.buf = make([]byte, MaxEncodedLen(defaultBlockSize)+checksumSize) + } + nr.readHeader = nr.ignoreStreamID + nr.paramsOK = true + return &nr +} + +// ReaderOption is an option for creating a decoder. +type ReaderOption func(*Reader) error + +// ReaderMaxBlockSize allows to control allocations if the stream +// has been compressed with a smaller WriterBlockSize, or with the default 1MB. +// Blocks must be this size or smaller to decompress, +// otherwise the decoder will return ErrUnsupported. +// +// For streams compressed with Snappy this can safely be set to 64KB (64 << 10). +// +// Default is the maximum limit of 4MB. +func ReaderMaxBlockSize(blockSize int) ReaderOption { + return func(r *Reader) error { + if blockSize > maxBlockSize || blockSize <= 0 { + return errors.New("s2: block size too large. Must be <= 4MB and > 0") + } + if r.lazyBuf == 0 && blockSize < defaultBlockSize { + r.lazyBuf = blockSize + } + r.maxBlock = blockSize + return nil + } +} + +// ReaderAllocBlock allows to control upfront stream allocations +// and not allocate for frames bigger than this initially. +// If frames bigger than this is seen a bigger buffer will be allocated. +// +// Default is 1MB, which is default output size. +func ReaderAllocBlock(blockSize int) ReaderOption { + return func(r *Reader) error { + if blockSize > maxBlockSize || blockSize < 1024 { + return errors.New("s2: invalid ReaderAllocBlock. Must be <= 4MB and >= 1024") + } + r.lazyBuf = blockSize + return nil + } +} + +// ReaderIgnoreStreamIdentifier will make the reader skip the expected +// stream identifier at the beginning of the stream. +// This can be used when serving a stream that has been forwarded to a specific point. +func ReaderIgnoreStreamIdentifier() ReaderOption { + return func(r *Reader) error { + r.ignoreStreamID = true + return nil + } +} + +// ReaderSkippableCB will register a callback for chuncks with the specified ID. +// ID must be a Reserved skippable chunks ID, 0x80-0xfd (inclusive). +// For each chunk with the ID, the callback is called with the content. +// Any returned non-nil error will abort decompression. +// Only one callback per ID is supported, latest sent will be used. +func ReaderSkippableCB(id uint8, fn func(r io.Reader) error) ReaderOption { + return func(r *Reader) error { + if id < 0x80 || id > 0xfd { + return fmt.Errorf("ReaderSkippableCB: Invalid id provided, must be 0x80-0xfd (inclusive)") + } + r.skippableCB[id] = fn + return nil + } +} + +// ReaderIgnoreCRC will make the reader skip CRC calculation and checks. +func ReaderIgnoreCRC() ReaderOption { + return func(r *Reader) error { + r.ignoreCRC = true + return nil + } +} + +// Reader is an io.Reader that can read Snappy-compressed bytes. +type Reader struct { + r io.Reader + err error + decoded []byte + buf []byte + skippableCB [0x80]func(r io.Reader) error + blockStart int64 // Uncompressed offset at start of current. + index *Index + + // decoded[i:j] contains decoded bytes that have not yet been passed on. + i, j int + // maximum block size allowed. + maxBlock int + // maximum expected buffer size. + maxBufSize int + // alloc a buffer this size if > 0. + lazyBuf int + readHeader bool + paramsOK bool + snappyFrame bool + ignoreStreamID bool + ignoreCRC bool +} + +// ensureBufferSize will ensure that the buffer can take at least n bytes. +// If false is returned the buffer exceeds maximum allowed size. +func (r *Reader) ensureBufferSize(n int) bool { + if n > r.maxBufSize { + r.err = ErrCorrupt + return false + } + if cap(r.buf) >= n { + return true + } + // Realloc buffer. + r.buf = make([]byte, n) + return true +} + +// Reset discards any buffered data, resets all state, and switches the Snappy +// reader to read from r. This permits reusing a Reader rather than allocating +// a new one. +func (r *Reader) Reset(reader io.Reader) { + if !r.paramsOK { + return + } + r.index = nil + r.r = reader + r.err = nil + r.i = 0 + r.j = 0 + r.blockStart = 0 + r.readHeader = r.ignoreStreamID +} + +func (r *Reader) readFull(p []byte, allowEOF bool) (ok bool) { + if _, r.err = io.ReadFull(r.r, p); r.err != nil { + if r.err == io.ErrUnexpectedEOF || (r.err == io.EOF && !allowEOF) { + r.err = ErrCorrupt + } + return false + } + return true +} + +// skippable will skip n bytes. +// If the supplied reader supports seeking that is used. +// tmp is used as a temporary buffer for reading. +// The supplied slice does not need to be the size of the read. +func (r *Reader) skippable(tmp []byte, n int, allowEOF bool, id uint8) (ok bool) { + if id < 0x80 { + r.err = fmt.Errorf("interbal error: skippable id < 0x80") + return false + } + if fn := r.skippableCB[id-0x80]; fn != nil { + rd := io.LimitReader(r.r, int64(n)) + r.err = fn(rd) + if r.err != nil { + return false + } + _, r.err = io.CopyBuffer(ioutil.Discard, rd, tmp) + return r.err == nil + } + if rs, ok := r.r.(io.ReadSeeker); ok { + _, err := rs.Seek(int64(n), io.SeekCurrent) + if err == nil { + return true + } + if err == io.ErrUnexpectedEOF || (r.err == io.EOF && !allowEOF) { + r.err = ErrCorrupt + return false + } + } + for n > 0 { + if n < len(tmp) { + tmp = tmp[:n] + } + if _, r.err = io.ReadFull(r.r, tmp); r.err != nil { + if r.err == io.ErrUnexpectedEOF || (r.err == io.EOF && !allowEOF) { + r.err = ErrCorrupt + } + return false + } + n -= len(tmp) + } + return true +} + +// Read satisfies the io.Reader interface. +func (r *Reader) Read(p []byte) (int, error) { + if r.err != nil { + return 0, r.err + } + for { + if r.i < r.j { + n := copy(p, r.decoded[r.i:r.j]) + r.i += n + return n, nil + } + if !r.readFull(r.buf[:4], true) { + return 0, r.err + } + chunkType := r.buf[0] + if !r.readHeader { + if chunkType != chunkTypeStreamIdentifier { + r.err = ErrCorrupt + return 0, r.err + } + r.readHeader = true + } + chunkLen := int(r.buf[1]) | int(r.buf[2])<<8 | int(r.buf[3])<<16 + + // The chunk types are specified at + // https://github.com/google/snappy/blob/master/framing_format.txt + switch chunkType { + case chunkTypeCompressedData: + r.blockStart += int64(r.j) + // Section 4.2. Compressed data (chunk type 0x00). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + if !r.ensureBufferSize(chunkLen) { + if r.err == nil { + r.err = ErrUnsupported + } + return 0, r.err + } + buf := r.buf[:chunkLen] + if !r.readFull(buf, false) { + return 0, r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + buf = buf[checksumSize:] + + n, err := DecodedLen(buf) + if err != nil { + r.err = err + return 0, r.err + } + if r.snappyFrame && n > maxSnappyBlockSize { + r.err = ErrCorrupt + return 0, r.err + } + + if n > len(r.decoded) { + if n > r.maxBlock { + r.err = ErrCorrupt + return 0, r.err + } + r.decoded = make([]byte, n) + } + if _, err := Decode(r.decoded, buf); err != nil { + r.err = err + return 0, r.err + } + if !r.ignoreCRC && crc(r.decoded[:n]) != checksum { + r.err = ErrCRC + return 0, r.err + } + r.i, r.j = 0, n + continue + + case chunkTypeUncompressedData: + r.blockStart += int64(r.j) + // Section 4.3. Uncompressed data (chunk type 0x01). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + if !r.ensureBufferSize(chunkLen) { + if r.err == nil { + r.err = ErrUnsupported + } + return 0, r.err + } + buf := r.buf[:checksumSize] + if !r.readFull(buf, false) { + return 0, r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + // Read directly into r.decoded instead of via r.buf. + n := chunkLen - checksumSize + if r.snappyFrame && n > maxSnappyBlockSize { + r.err = ErrCorrupt + return 0, r.err + } + if n > len(r.decoded) { + if n > r.maxBlock { + r.err = ErrCorrupt + return 0, r.err + } + r.decoded = make([]byte, n) + } + if !r.readFull(r.decoded[:n], false) { + return 0, r.err + } + if !r.ignoreCRC && crc(r.decoded[:n]) != checksum { + r.err = ErrCRC + return 0, r.err + } + r.i, r.j = 0, n + continue + + case chunkTypeStreamIdentifier: + // Section 4.1. Stream identifier (chunk type 0xff). + if chunkLen != len(magicBody) { + r.err = ErrCorrupt + return 0, r.err + } + if !r.readFull(r.buf[:len(magicBody)], false) { + return 0, r.err + } + if string(r.buf[:len(magicBody)]) != magicBody { + if string(r.buf[:len(magicBody)]) != magicBodySnappy { + r.err = ErrCorrupt + return 0, r.err + } else { + r.snappyFrame = true + } + } else { + r.snappyFrame = false + } + continue + } + + if chunkType <= 0x7f { + // Section 4.5. Reserved unskippable chunks (chunk types 0x02-0x7f). + // fmt.Printf("ERR chunktype: 0x%x\n", chunkType) + r.err = ErrUnsupported + return 0, r.err + } + // Section 4.4 Padding (chunk type 0xfe). + // Section 4.6. Reserved skippable chunks (chunk types 0x80-0xfd). + if chunkLen > maxChunkSize { + // fmt.Printf("ERR chunkLen: 0x%x\n", chunkLen) + r.err = ErrUnsupported + return 0, r.err + } + + // fmt.Printf("skippable: ID: 0x%x, len: 0x%x\n", chunkType, chunkLen) + if !r.skippable(r.buf, chunkLen, false, chunkType) { + return 0, r.err + } + } +} + +// DecodeConcurrent will decode the full stream to w. +// This function should not be combined with reading, seeking or other operations. +// Up to 'concurrent' goroutines will be used. +// If <= 0, runtime.NumCPU will be used. +// On success the number of bytes decompressed nil and is returned. +// This is mainly intended for bigger streams. +func (r *Reader) DecodeConcurrent(w io.Writer, concurrent int) (written int64, err error) { + if r.i > 0 || r.j > 0 || r.blockStart > 0 { + return 0, errors.New("DecodeConcurrent called after ") + } + if concurrent <= 0 { + concurrent = runtime.NumCPU() + } + + // Write to output + var errMu sync.Mutex + var aErr error + setErr := func(e error) (ok bool) { + errMu.Lock() + defer errMu.Unlock() + if e == nil { + return aErr == nil + } + if aErr == nil { + aErr = e + } + return false + } + hasErr := func() (ok bool) { + errMu.Lock() + v := aErr != nil + errMu.Unlock() + return v + } + + var aWritten int64 + toRead := make(chan []byte, concurrent) + writtenBlocks := make(chan []byte, concurrent) + queue := make(chan chan []byte, concurrent) + reUse := make(chan chan []byte, concurrent) + for i := 0; i < concurrent; i++ { + toRead <- make([]byte, 0, r.maxBufSize) + writtenBlocks <- make([]byte, 0, r.maxBufSize) + reUse <- make(chan []byte, 1) + } + // Writer + var wg sync.WaitGroup + wg.Add(1) + go func() { + defer wg.Done() + for toWrite := range queue { + entry := <-toWrite + reUse <- toWrite + if hasErr() { + writtenBlocks <- entry + continue + } + n, err := w.Write(entry) + want := len(entry) + writtenBlocks <- entry + if err != nil { + setErr(err) + continue + } + if n != want { + setErr(io.ErrShortWrite) + continue + } + aWritten += int64(n) + } + }() + + // Reader + defer func() { + close(queue) + if r.err != nil { + err = r.err + setErr(r.err) + } + wg.Wait() + if err == nil { + err = aErr + } + written = aWritten + }() + + for !hasErr() { + if !r.readFull(r.buf[:4], true) { + if r.err == io.EOF { + r.err = nil + } + return 0, r.err + } + chunkType := r.buf[0] + if !r.readHeader { + if chunkType != chunkTypeStreamIdentifier { + r.err = ErrCorrupt + return 0, r.err + } + r.readHeader = true + } + chunkLen := int(r.buf[1]) | int(r.buf[2])<<8 | int(r.buf[3])<<16 + + // The chunk types are specified at + // https://github.com/google/snappy/blob/master/framing_format.txt + switch chunkType { + case chunkTypeCompressedData: + r.blockStart += int64(r.j) + // Section 4.2. Compressed data (chunk type 0x00). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + if chunkLen > r.maxBufSize { + r.err = ErrCorrupt + return 0, r.err + } + orgBuf := <-toRead + buf := orgBuf[:chunkLen] + + if !r.readFull(buf, false) { + return 0, r.err + } + + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + buf = buf[checksumSize:] + + n, err := DecodedLen(buf) + if err != nil { + r.err = err + return 0, r.err + } + if r.snappyFrame && n > maxSnappyBlockSize { + r.err = ErrCorrupt + return 0, r.err + } + + if n > r.maxBlock { + r.err = ErrCorrupt + return 0, r.err + } + wg.Add(1) + + decoded := <-writtenBlocks + entry := <-reUse + queue <- entry + go func() { + defer wg.Done() + decoded = decoded[:n] + _, err := Decode(decoded, buf) + toRead <- orgBuf + if err != nil { + writtenBlocks <- decoded + setErr(err) + return + } + if !r.ignoreCRC && crc(decoded) != checksum { + writtenBlocks <- decoded + setErr(ErrCRC) + return + } + entry <- decoded + }() + continue + + case chunkTypeUncompressedData: + + // Section 4.3. Uncompressed data (chunk type 0x01). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + if chunkLen > r.maxBufSize { + r.err = ErrCorrupt + return 0, r.err + } + // Grab write buffer + orgBuf := <-writtenBlocks + buf := orgBuf[:checksumSize] + if !r.readFull(buf, false) { + return 0, r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + // Read content. + n := chunkLen - checksumSize + + if r.snappyFrame && n > maxSnappyBlockSize { + r.err = ErrCorrupt + return 0, r.err + } + if n > r.maxBlock { + r.err = ErrCorrupt + return 0, r.err + } + // Read uncompressed + buf = orgBuf[:n] + if !r.readFull(buf, false) { + return 0, r.err + } + + if !r.ignoreCRC && crc(buf) != checksum { + r.err = ErrCRC + return 0, r.err + } + entry := <-reUse + queue <- entry + entry <- buf + continue + + case chunkTypeStreamIdentifier: + // Section 4.1. Stream identifier (chunk type 0xff). + if chunkLen != len(magicBody) { + r.err = ErrCorrupt + return 0, r.err + } + if !r.readFull(r.buf[:len(magicBody)], false) { + return 0, r.err + } + if string(r.buf[:len(magicBody)]) != magicBody { + if string(r.buf[:len(magicBody)]) != magicBodySnappy { + r.err = ErrCorrupt + return 0, r.err + } else { + r.snappyFrame = true + } + } else { + r.snappyFrame = false + } + continue + } + + if chunkType <= 0x7f { + // Section 4.5. Reserved unskippable chunks (chunk types 0x02-0x7f). + // fmt.Printf("ERR chunktype: 0x%x\n", chunkType) + r.err = ErrUnsupported + return 0, r.err + } + // Section 4.4 Padding (chunk type 0xfe). + // Section 4.6. Reserved skippable chunks (chunk types 0x80-0xfd). + if chunkLen > maxChunkSize { + // fmt.Printf("ERR chunkLen: 0x%x\n", chunkLen) + r.err = ErrUnsupported + return 0, r.err + } + + // fmt.Printf("skippable: ID: 0x%x, len: 0x%x\n", chunkType, chunkLen) + if !r.skippable(r.buf, chunkLen, false, chunkType) { + return 0, r.err + } + } + return 0, r.err +} + +// Skip will skip n bytes forward in the decompressed output. +// For larger skips this consumes less CPU and is faster than reading output and discarding it. +// CRC is not checked on skipped blocks. +// io.ErrUnexpectedEOF is returned if the stream ends before all bytes have been skipped. +// If a decoding error is encountered subsequent calls to Read will also fail. +func (r *Reader) Skip(n int64) error { + if n < 0 { + return errors.New("attempted negative skip") + } + if r.err != nil { + return r.err + } + + for n > 0 { + if r.i < r.j { + // Skip in buffer. + // decoded[i:j] contains decoded bytes that have not yet been passed on. + left := int64(r.j - r.i) + if left >= n { + tmp := int64(r.i) + n + if tmp > math.MaxInt32 { + return errors.New("s2: internal overflow in skip") + } + r.i = int(tmp) + return nil + } + n -= int64(r.j - r.i) + r.i = r.j + } + + // Buffer empty; read blocks until we have content. + if !r.readFull(r.buf[:4], true) { + if r.err == io.EOF { + r.err = io.ErrUnexpectedEOF + } + return r.err + } + chunkType := r.buf[0] + if !r.readHeader { + if chunkType != chunkTypeStreamIdentifier { + r.err = ErrCorrupt + return r.err + } + r.readHeader = true + } + chunkLen := int(r.buf[1]) | int(r.buf[2])<<8 | int(r.buf[3])<<16 + + // The chunk types are specified at + // https://github.com/google/snappy/blob/master/framing_format.txt + switch chunkType { + case chunkTypeCompressedData: + r.blockStart += int64(r.j) + // Section 4.2. Compressed data (chunk type 0x00). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return r.err + } + if !r.ensureBufferSize(chunkLen) { + if r.err == nil { + r.err = ErrUnsupported + } + return r.err + } + buf := r.buf[:chunkLen] + if !r.readFull(buf, false) { + return r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + buf = buf[checksumSize:] + + dLen, err := DecodedLen(buf) + if err != nil { + r.err = err + return r.err + } + if dLen > r.maxBlock { + r.err = ErrCorrupt + return r.err + } + // Check if destination is within this block + if int64(dLen) > n { + if len(r.decoded) < dLen { + r.decoded = make([]byte, dLen) + } + if _, err := Decode(r.decoded, buf); err != nil { + r.err = err + return r.err + } + if crc(r.decoded[:dLen]) != checksum { + r.err = ErrCorrupt + return r.err + } + } else { + // Skip block completely + n -= int64(dLen) + r.blockStart += int64(dLen) + dLen = 0 + } + r.i, r.j = 0, dLen + continue + case chunkTypeUncompressedData: + r.blockStart += int64(r.j) + // Section 4.3. Uncompressed data (chunk type 0x01). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return r.err + } + if !r.ensureBufferSize(chunkLen) { + if r.err != nil { + r.err = ErrUnsupported + } + return r.err + } + buf := r.buf[:checksumSize] + if !r.readFull(buf, false) { + return r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + // Read directly into r.decoded instead of via r.buf. + n2 := chunkLen - checksumSize + if n2 > len(r.decoded) { + if n2 > r.maxBlock { + r.err = ErrCorrupt + return r.err + } + r.decoded = make([]byte, n2) + } + if !r.readFull(r.decoded[:n2], false) { + return r.err + } + if int64(n2) < n { + if crc(r.decoded[:n2]) != checksum { + r.err = ErrCorrupt + return r.err + } + } + r.i, r.j = 0, n2 + continue + case chunkTypeStreamIdentifier: + // Section 4.1. Stream identifier (chunk type 0xff). + if chunkLen != len(magicBody) { + r.err = ErrCorrupt + return r.err + } + if !r.readFull(r.buf[:len(magicBody)], false) { + return r.err + } + if string(r.buf[:len(magicBody)]) != magicBody { + if string(r.buf[:len(magicBody)]) != magicBodySnappy { + r.err = ErrCorrupt + return r.err + } + } + + continue + } + + if chunkType <= 0x7f { + // Section 4.5. Reserved unskippable chunks (chunk types 0x02-0x7f). + r.err = ErrUnsupported + return r.err + } + if chunkLen > maxChunkSize { + r.err = ErrUnsupported + return r.err + } + // Section 4.4 Padding (chunk type 0xfe). + // Section 4.6. Reserved skippable chunks (chunk types 0x80-0xfd). + if !r.skippable(r.buf, chunkLen, false, chunkType) { + return r.err + } + } + return nil +} + +// ReadSeeker provides random or forward seeking in compressed content. +// See Reader.ReadSeeker +type ReadSeeker struct { + *Reader +} + +// ReadSeeker will return an io.ReadSeeker compatible version of the reader. +// If 'random' is specified the returned io.Seeker can be used for +// random seeking, otherwise only forward seeking is supported. +// Enabling random seeking requires the original input to support +// the io.Seeker interface. +// A custom index can be specified which will be used if supplied. +// When using a custom index, it will not be read from the input stream. +// The returned ReadSeeker contains a shallow reference to the existing Reader, +// meaning changes performed to one is reflected in the other. +func (r *Reader) ReadSeeker(random bool, index []byte) (*ReadSeeker, error) { + // Read index if provided. + if len(index) != 0 { + if r.index == nil { + r.index = &Index{} + } + if _, err := r.index.Load(index); err != nil { + return nil, ErrCantSeek{Reason: "loading index returned: " + err.Error()} + } + } + + // Check if input is seekable + rs, ok := r.r.(io.ReadSeeker) + if !ok { + if !random { + return &ReadSeeker{Reader: r}, nil + } + return nil, ErrCantSeek{Reason: "input stream isn't seekable"} + } + + if r.index != nil { + // Seekable and index, ok... + return &ReadSeeker{Reader: r}, nil + } + + // Load from stream. + r.index = &Index{} + + // Read current position. + pos, err := rs.Seek(0, io.SeekCurrent) + if err != nil { + return nil, ErrCantSeek{Reason: "seeking input returned: " + err.Error()} + } + err = r.index.LoadStream(rs) + if err != nil { + if err == ErrUnsupported { + // If we don't require random seeking, reset input and return. + if !random { + _, err = rs.Seek(pos, io.SeekStart) + if err != nil { + return nil, ErrCantSeek{Reason: "resetting stream returned: " + err.Error()} + } + r.index = nil + return &ReadSeeker{Reader: r}, nil + } + return nil, ErrCantSeek{Reason: "input stream does not contain an index"} + } + return nil, ErrCantSeek{Reason: "reading index returned: " + err.Error()} + } + + // reset position. + _, err = rs.Seek(pos, io.SeekStart) + if err != nil { + return nil, ErrCantSeek{Reason: "seeking input returned: " + err.Error()} + } + return &ReadSeeker{Reader: r}, nil +} + +// Seek allows seeking in compressed data. +func (r *ReadSeeker) Seek(offset int64, whence int) (int64, error) { + if r.err != nil { + return 0, r.err + } + if offset == 0 && whence == io.SeekCurrent { + return r.blockStart + int64(r.i), nil + } + if !r.readHeader { + // Make sure we read the header. + _, r.err = r.Read([]byte{}) + } + rs, ok := r.r.(io.ReadSeeker) + if r.index == nil || !ok { + if whence == io.SeekCurrent && offset >= 0 { + err := r.Skip(offset) + return r.blockStart + int64(r.i), err + } + if whence == io.SeekStart && offset >= r.blockStart+int64(r.i) { + err := r.Skip(offset - r.blockStart - int64(r.i)) + return r.blockStart + int64(r.i), err + } + return 0, ErrUnsupported + + } + + switch whence { + case io.SeekCurrent: + offset += r.blockStart + int64(r.i) + case io.SeekEnd: + if offset > 0 { + return 0, errors.New("seek after end of file") + } + offset = r.index.TotalUncompressed + offset + } + + if offset < 0 { + return 0, errors.New("seek before start of file") + } + + c, u, err := r.index.Find(offset) + if err != nil { + return r.blockStart + int64(r.i), err + } + + // Seek to next block + _, err = rs.Seek(c, io.SeekStart) + if err != nil { + return 0, err + } + + r.i = r.j // Remove rest of current block. + if u < offset { + // Forward inside block + return offset, r.Skip(offset - u) + } + return offset, nil +} + +// ReadByte satisfies the io.ByteReader interface. +func (r *Reader) ReadByte() (byte, error) { + if r.err != nil { + return 0, r.err + } + if r.i < r.j { + c := r.decoded[r.i] + r.i++ + return c, nil + } + var tmp [1]byte + for i := 0; i < 10; i++ { + n, err := r.Read(tmp[:]) + if err != nil { + return 0, err + } + if n == 1 { + return tmp[0], nil + } + } + return 0, io.ErrNoProgress +} + +// SkippableCB will register a callback for chunks with the specified ID. +// ID must be a Reserved skippable chunks ID, 0x80-0xfe (inclusive). +// For each chunk with the ID, the callback is called with the content. +// Any returned non-nil error will abort decompression. +// Only one callback per ID is supported, latest sent will be used. +// Sending a nil function will disable previous callbacks. +func (r *Reader) SkippableCB(id uint8, fn func(r io.Reader) error) error { + if id < 0x80 || id > chunkTypePadding { + return fmt.Errorf("ReaderSkippableCB: Invalid id provided, must be 0x80-0xfe (inclusive)") + } + r.skippableCB[id] = fn + return nil +} diff --git a/vendor/github.com/klauspost/compress/s2/decode_amd64.s b/vendor/github.com/klauspost/compress/s2/decode_amd64.s new file mode 100644 index 000000000..9b105e03c --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/decode_amd64.s @@ -0,0 +1,568 @@ +// Copyright 2016 The Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build !appengine +// +build gc +// +build !noasm + +#include "textflag.h" + +#define R_TMP0 AX +#define R_TMP1 BX +#define R_LEN CX +#define R_OFF DX +#define R_SRC SI +#define R_DST DI +#define R_DBASE R8 +#define R_DLEN R9 +#define R_DEND R10 +#define R_SBASE R11 +#define R_SLEN R12 +#define R_SEND R13 +#define R_TMP2 R14 +#define R_TMP3 R15 + +// The asm code generally follows the pure Go code in decode_other.go, except +// where marked with a "!!!". + +// func decode(dst, src []byte) int +// +// All local variables fit into registers. The non-zero stack size is only to +// spill registers and push args when issuing a CALL. The register allocation: +// - R_TMP0 scratch +// - R_TMP1 scratch +// - R_LEN length or x (shared) +// - R_OFF offset +// - R_SRC &src[s] +// - R_DST &dst[d] +// + R_DBASE dst_base +// + R_DLEN dst_len +// + R_DEND dst_base + dst_len +// + R_SBASE src_base +// + R_SLEN src_len +// + R_SEND src_base + src_len +// - R_TMP2 used by doCopy +// - R_TMP3 used by doCopy +// +// The registers R_DBASE-R_SEND (marked with a "+") are set at the start of the +// function, and after a CALL returns, and are not otherwise modified. +// +// The d variable is implicitly R_DST - R_DBASE, and len(dst)-d is R_DEND - R_DST. +// The s variable is implicitly R_SRC - R_SBASE, and len(src)-s is R_SEND - R_SRC. +TEXT ·s2Decode(SB), NOSPLIT, $48-56 + // Initialize R_SRC, R_DST and R_DBASE-R_SEND. + MOVQ dst_base+0(FP), R_DBASE + MOVQ dst_len+8(FP), R_DLEN + MOVQ R_DBASE, R_DST + MOVQ R_DBASE, R_DEND + ADDQ R_DLEN, R_DEND + MOVQ src_base+24(FP), R_SBASE + MOVQ src_len+32(FP), R_SLEN + MOVQ R_SBASE, R_SRC + MOVQ R_SBASE, R_SEND + ADDQ R_SLEN, R_SEND + XORQ R_OFF, R_OFF + +loop: + // for s < len(src) + CMPQ R_SRC, R_SEND + JEQ end + + // R_LEN = uint32(src[s]) + // + // switch src[s] & 0x03 + MOVBLZX (R_SRC), R_LEN + MOVL R_LEN, R_TMP1 + ANDL $3, R_TMP1 + CMPL R_TMP1, $1 + JAE tagCopy + + // ---------------------------------------- + // The code below handles literal tags. + + // case tagLiteral: + // x := uint32(src[s] >> 2) + // switch + SHRL $2, R_LEN + CMPL R_LEN, $60 + JAE tagLit60Plus + + // case x < 60: + // s++ + INCQ R_SRC + +doLit: + // This is the end of the inner "switch", when we have a literal tag. + // + // We assume that R_LEN == x and x fits in a uint32, where x is the variable + // used in the pure Go decode_other.go code. + + // length = int(x) + 1 + // + // Unlike the pure Go code, we don't need to check if length <= 0 because + // R_LEN can hold 64 bits, so the increment cannot overflow. + INCQ R_LEN + + // Prepare to check if copying length bytes will run past the end of dst or + // src. + // + // R_TMP0 = len(dst) - d + // R_TMP1 = len(src) - s + MOVQ R_DEND, R_TMP0 + SUBQ R_DST, R_TMP0 + MOVQ R_SEND, R_TMP1 + SUBQ R_SRC, R_TMP1 + + // !!! Try a faster technique for short (16 or fewer bytes) copies. + // + // if length > 16 || len(dst)-d < 16 || len(src)-s < 16 { + // goto callMemmove // Fall back on calling runtime·memmove. + // } + // + // The C++ snappy code calls this TryFastAppend. It also checks len(src)-s + // against 21 instead of 16, because it cannot assume that all of its input + // is contiguous in memory and so it needs to leave enough source bytes to + // read the next tag without refilling buffers, but Go's Decode assumes + // contiguousness (the src argument is a []byte). + CMPQ R_LEN, $16 + JGT callMemmove + CMPQ R_TMP0, $16 + JLT callMemmove + CMPQ R_TMP1, $16 + JLT callMemmove + + // !!! Implement the copy from src to dst as a 16-byte load and store. + // (Decode's documentation says that dst and src must not overlap.) + // + // This always copies 16 bytes, instead of only length bytes, but that's + // OK. If the input is a valid Snappy encoding then subsequent iterations + // will fix up the overrun. Otherwise, Decode returns a nil []byte (and a + // non-nil error), so the overrun will be ignored. + // + // Note that on amd64, it is legal and cheap to issue unaligned 8-byte or + // 16-byte loads and stores. This technique probably wouldn't be as + // effective on architectures that are fussier about alignment. + MOVOU 0(R_SRC), X0 + MOVOU X0, 0(R_DST) + + // d += length + // s += length + ADDQ R_LEN, R_DST + ADDQ R_LEN, R_SRC + JMP loop + +callMemmove: + // if length > len(dst)-d || length > len(src)-s { etc } + CMPQ R_LEN, R_TMP0 + JGT errCorrupt + CMPQ R_LEN, R_TMP1 + JGT errCorrupt + + // copy(dst[d:], src[s:s+length]) + // + // This means calling runtime·memmove(&dst[d], &src[s], length), so we push + // R_DST, R_SRC and R_LEN as arguments. Coincidentally, we also need to spill those + // three registers to the stack, to save local variables across the CALL. + MOVQ R_DST, 0(SP) + MOVQ R_SRC, 8(SP) + MOVQ R_LEN, 16(SP) + MOVQ R_DST, 24(SP) + MOVQ R_SRC, 32(SP) + MOVQ R_LEN, 40(SP) + MOVQ R_OFF, 48(SP) + CALL runtime·memmove(SB) + + // Restore local variables: unspill registers from the stack and + // re-calculate R_DBASE-R_SEND. + MOVQ 24(SP), R_DST + MOVQ 32(SP), R_SRC + MOVQ 40(SP), R_LEN + MOVQ 48(SP), R_OFF + MOVQ dst_base+0(FP), R_DBASE + MOVQ dst_len+8(FP), R_DLEN + MOVQ R_DBASE, R_DEND + ADDQ R_DLEN, R_DEND + MOVQ src_base+24(FP), R_SBASE + MOVQ src_len+32(FP), R_SLEN + MOVQ R_SBASE, R_SEND + ADDQ R_SLEN, R_SEND + + // d += length + // s += length + ADDQ R_LEN, R_DST + ADDQ R_LEN, R_SRC + JMP loop + +tagLit60Plus: + // !!! This fragment does the + // + // s += x - 58; if uint(s) > uint(len(src)) { etc } + // + // checks. In the asm version, we code it once instead of once per switch case. + ADDQ R_LEN, R_SRC + SUBQ $58, R_SRC + CMPQ R_SRC, R_SEND + JA errCorrupt + + // case x == 60: + CMPL R_LEN, $61 + JEQ tagLit61 + JA tagLit62Plus + + // x = uint32(src[s-1]) + MOVBLZX -1(R_SRC), R_LEN + JMP doLit + +tagLit61: + // case x == 61: + // x = uint32(src[s-2]) | uint32(src[s-1])<<8 + MOVWLZX -2(R_SRC), R_LEN + JMP doLit + +tagLit62Plus: + CMPL R_LEN, $62 + JA tagLit63 + + // case x == 62: + // x = uint32(src[s-3]) | uint32(src[s-2])<<8 | uint32(src[s-1])<<16 + // We read one byte, safe to read one back, since we are just reading tag. + // x = binary.LittleEndian.Uint32(src[s-1:]) >> 8 + MOVL -4(R_SRC), R_LEN + SHRL $8, R_LEN + JMP doLit + +tagLit63: + // case x == 63: + // x = uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24 + MOVL -4(R_SRC), R_LEN + JMP doLit + +// The code above handles literal tags. +// ---------------------------------------- +// The code below handles copy tags. + +tagCopy4: + // case tagCopy4: + // s += 5 + ADDQ $5, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // length = 1 + int(src[s-5])>>2 + SHRQ $2, R_LEN + INCQ R_LEN + + // offset = int(uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24) + MOVLQZX -4(R_SRC), R_OFF + JMP doCopy + +tagCopy2: + // case tagCopy2: + // s += 3 + ADDQ $3, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // length = 1 + int(src[s-3])>>2 + SHRQ $2, R_LEN + INCQ R_LEN + + // offset = int(uint32(src[s-2]) | uint32(src[s-1])<<8) + MOVWQZX -2(R_SRC), R_OFF + JMP doCopy + +tagCopy: + // We have a copy tag. We assume that: + // - R_TMP1 == src[s] & 0x03 + // - R_LEN == src[s] + CMPQ R_TMP1, $2 + JEQ tagCopy2 + JA tagCopy4 + + // case tagCopy1: + // s += 2 + ADDQ $2, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // offset = int(uint32(src[s-2])&0xe0<<3 | uint32(src[s-1])) + // length = 4 + int(src[s-2])>>2&0x7 + MOVBQZX -1(R_SRC), R_TMP1 + MOVQ R_LEN, R_TMP0 + SHRQ $2, R_LEN + ANDQ $0xe0, R_TMP0 + ANDQ $7, R_LEN + SHLQ $3, R_TMP0 + ADDQ $4, R_LEN + ORQ R_TMP1, R_TMP0 + + // check if repeat code, ZF set by ORQ. + JZ repeatCode + + // This is a regular copy, transfer our temporary value to R_OFF (length) + MOVQ R_TMP0, R_OFF + JMP doCopy + +// This is a repeat code. +repeatCode: + // If length < 9, reuse last offset, with the length already calculated. + CMPQ R_LEN, $9 + JL doCopyRepeat + + // Read additional bytes for length. + JE repeatLen1 + + // Rare, so the extra branch shouldn't hurt too much. + CMPQ R_LEN, $10 + JE repeatLen2 + JMP repeatLen3 + +// Read repeat lengths. +repeatLen1: + // s ++ + ADDQ $1, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // length = src[s-1] + 8 + MOVBQZX -1(R_SRC), R_LEN + ADDL $8, R_LEN + JMP doCopyRepeat + +repeatLen2: + // s +=2 + ADDQ $2, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // length = uint32(src[s-2]) | (uint32(src[s-1])<<8) + (1 << 8) + MOVWQZX -2(R_SRC), R_LEN + ADDL $260, R_LEN + JMP doCopyRepeat + +repeatLen3: + // s +=3 + ADDQ $3, R_SRC + + // if uint(s) > uint(len(src)) { etc } + CMPQ R_SRC, R_SEND + JA errCorrupt + + // length = uint32(src[s-3]) | (uint32(src[s-2])<<8) | (uint32(src[s-1])<<16) + (1 << 16) + // Read one byte further back (just part of the tag, shifted out) + MOVL -4(R_SRC), R_LEN + SHRL $8, R_LEN + ADDL $65540, R_LEN + JMP doCopyRepeat + +doCopy: + // This is the end of the outer "switch", when we have a copy tag. + // + // We assume that: + // - R_LEN == length && R_LEN > 0 + // - R_OFF == offset + + // if d < offset { etc } + MOVQ R_DST, R_TMP1 + SUBQ R_DBASE, R_TMP1 + CMPQ R_TMP1, R_OFF + JLT errCorrupt + + // Repeat values can skip the test above, since any offset > 0 will be in dst. +doCopyRepeat: + // if offset <= 0 { etc } + CMPQ R_OFF, $0 + JLE errCorrupt + + // if length > len(dst)-d { etc } + MOVQ R_DEND, R_TMP1 + SUBQ R_DST, R_TMP1 + CMPQ R_LEN, R_TMP1 + JGT errCorrupt + + // forwardCopy(dst[d:d+length], dst[d-offset:]); d += length + // + // Set: + // - R_TMP2 = len(dst)-d + // - R_TMP3 = &dst[d-offset] + MOVQ R_DEND, R_TMP2 + SUBQ R_DST, R_TMP2 + MOVQ R_DST, R_TMP3 + SUBQ R_OFF, R_TMP3 + + // !!! Try a faster technique for short (16 or fewer bytes) forward copies. + // + // First, try using two 8-byte load/stores, similar to the doLit technique + // above. Even if dst[d:d+length] and dst[d-offset:] can overlap, this is + // still OK if offset >= 8. Note that this has to be two 8-byte load/stores + // and not one 16-byte load/store, and the first store has to be before the + // second load, due to the overlap if offset is in the range [8, 16). + // + // if length > 16 || offset < 8 || len(dst)-d < 16 { + // goto slowForwardCopy + // } + // copy 16 bytes + // d += length + CMPQ R_LEN, $16 + JGT slowForwardCopy + CMPQ R_OFF, $8 + JLT slowForwardCopy + CMPQ R_TMP2, $16 + JLT slowForwardCopy + MOVQ 0(R_TMP3), R_TMP0 + MOVQ R_TMP0, 0(R_DST) + MOVQ 8(R_TMP3), R_TMP1 + MOVQ R_TMP1, 8(R_DST) + ADDQ R_LEN, R_DST + JMP loop + +slowForwardCopy: + // !!! If the forward copy is longer than 16 bytes, or if offset < 8, we + // can still try 8-byte load stores, provided we can overrun up to 10 extra + // bytes. As above, the overrun will be fixed up by subsequent iterations + // of the outermost loop. + // + // The C++ snappy code calls this technique IncrementalCopyFastPath. Its + // commentary says: + // + // ---- + // + // The main part of this loop is a simple copy of eight bytes at a time + // until we've copied (at least) the requested amount of bytes. However, + // if d and d-offset are less than eight bytes apart (indicating a + // repeating pattern of length < 8), we first need to expand the pattern in + // order to get the correct results. For instance, if the buffer looks like + // this, with the eight-byte and patterns marked as + // intervals: + // + // abxxxxxxxxxxxx + // [------] d-offset + // [------] d + // + // a single eight-byte copy from to will repeat the pattern + // once, after which we can move two bytes without moving : + // + // ababxxxxxxxxxx + // [------] d-offset + // [------] d + // + // and repeat the exercise until the two no longer overlap. + // + // This allows us to do very well in the special case of one single byte + // repeated many times, without taking a big hit for more general cases. + // + // The worst case of extra writing past the end of the match occurs when + // offset == 1 and length == 1; the last copy will read from byte positions + // [0..7] and write to [4..11], whereas it was only supposed to write to + // position 1. Thus, ten excess bytes. + // + // ---- + // + // That "10 byte overrun" worst case is confirmed by Go's + // TestSlowForwardCopyOverrun, which also tests the fixUpSlowForwardCopy + // and finishSlowForwardCopy algorithm. + // + // if length > len(dst)-d-10 { + // goto verySlowForwardCopy + // } + SUBQ $10, R_TMP2 + CMPQ R_LEN, R_TMP2 + JGT verySlowForwardCopy + + // We want to keep the offset, so we use R_TMP2 from here. + MOVQ R_OFF, R_TMP2 + +makeOffsetAtLeast8: + // !!! As above, expand the pattern so that offset >= 8 and we can use + // 8-byte load/stores. + // + // for offset < 8 { + // copy 8 bytes from dst[d-offset:] to dst[d:] + // length -= offset + // d += offset + // offset += offset + // // The two previous lines together means that d-offset, and therefore + // // R_TMP3, is unchanged. + // } + CMPQ R_TMP2, $8 + JGE fixUpSlowForwardCopy + MOVQ (R_TMP3), R_TMP1 + MOVQ R_TMP1, (R_DST) + SUBQ R_TMP2, R_LEN + ADDQ R_TMP2, R_DST + ADDQ R_TMP2, R_TMP2 + JMP makeOffsetAtLeast8 + +fixUpSlowForwardCopy: + // !!! Add length (which might be negative now) to d (implied by R_DST being + // &dst[d]) so that d ends up at the right place when we jump back to the + // top of the loop. Before we do that, though, we save R_DST to R_TMP0 so that, if + // length is positive, copying the remaining length bytes will write to the + // right place. + MOVQ R_DST, R_TMP0 + ADDQ R_LEN, R_DST + +finishSlowForwardCopy: + // !!! Repeat 8-byte load/stores until length <= 0. Ending with a negative + // length means that we overrun, but as above, that will be fixed up by + // subsequent iterations of the outermost loop. + CMPQ R_LEN, $0 + JLE loop + MOVQ (R_TMP3), R_TMP1 + MOVQ R_TMP1, (R_TMP0) + ADDQ $8, R_TMP3 + ADDQ $8, R_TMP0 + SUBQ $8, R_LEN + JMP finishSlowForwardCopy + +verySlowForwardCopy: + // verySlowForwardCopy is a simple implementation of forward copy. In C + // parlance, this is a do/while loop instead of a while loop, since we know + // that length > 0. In Go syntax: + // + // for { + // dst[d] = dst[d - offset] + // d++ + // length-- + // if length == 0 { + // break + // } + // } + MOVB (R_TMP3), R_TMP1 + MOVB R_TMP1, (R_DST) + INCQ R_TMP3 + INCQ R_DST + DECQ R_LEN + JNZ verySlowForwardCopy + JMP loop + +// The code above handles copy tags. +// ---------------------------------------- + +end: + // This is the end of the "for s < len(src)". + // + // if d != len(dst) { etc } + CMPQ R_DST, R_DEND + JNE errCorrupt + + // return 0 + MOVQ $0, ret+48(FP) + RET + +errCorrupt: + // return decodeErrCodeCorrupt + MOVQ $1, ret+48(FP) + RET diff --git a/vendor/github.com/klauspost/compress/s2/decode_arm64.s b/vendor/github.com/klauspost/compress/s2/decode_arm64.s new file mode 100644 index 000000000..4b63d5086 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/decode_arm64.s @@ -0,0 +1,574 @@ +// Copyright 2020 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build !appengine +// +build gc +// +build !noasm + +#include "textflag.h" + +#define R_TMP0 R2 +#define R_TMP1 R3 +#define R_LEN R4 +#define R_OFF R5 +#define R_SRC R6 +#define R_DST R7 +#define R_DBASE R8 +#define R_DLEN R9 +#define R_DEND R10 +#define R_SBASE R11 +#define R_SLEN R12 +#define R_SEND R13 +#define R_TMP2 R14 +#define R_TMP3 R15 + +// TEST_SRC will check if R_SRC is <= SRC_END +#define TEST_SRC() \ + CMP R_SEND, R_SRC \ + BGT errCorrupt + +// MOVD R_SRC, R_TMP1 +// SUB R_SBASE, R_TMP1, R_TMP1 +// CMP R_SLEN, R_TMP1 +// BGT errCorrupt + +// The asm code generally follows the pure Go code in decode_other.go, except +// where marked with a "!!!". + +// func decode(dst, src []byte) int +// +// All local variables fit into registers. The non-zero stack size is only to +// spill registers and push args when issuing a CALL. The register allocation: +// - R_TMP0 scratch +// - R_TMP1 scratch +// - R_LEN length or x +// - R_OFF offset +// - R_SRC &src[s] +// - R_DST &dst[d] +// + R_DBASE dst_base +// + R_DLEN dst_len +// + R_DEND dst_base + dst_len +// + R_SBASE src_base +// + R_SLEN src_len +// + R_SEND src_base + src_len +// - R_TMP2 used by doCopy +// - R_TMP3 used by doCopy +// +// The registers R_DBASE-R_SEND (marked with a "+") are set at the start of the +// function, and after a CALL returns, and are not otherwise modified. +// +// The d variable is implicitly R_DST - R_DBASE, and len(dst)-d is R_DEND - R_DST. +// The s variable is implicitly R_SRC - R_SBASE, and len(src)-s is R_SEND - R_SRC. +TEXT ·s2Decode(SB), NOSPLIT, $56-64 + // Initialize R_SRC, R_DST and R_DBASE-R_SEND. + MOVD dst_base+0(FP), R_DBASE + MOVD dst_len+8(FP), R_DLEN + MOVD R_DBASE, R_DST + MOVD R_DBASE, R_DEND + ADD R_DLEN, R_DEND, R_DEND + MOVD src_base+24(FP), R_SBASE + MOVD src_len+32(FP), R_SLEN + MOVD R_SBASE, R_SRC + MOVD R_SBASE, R_SEND + ADD R_SLEN, R_SEND, R_SEND + MOVD $0, R_OFF + +loop: + // for s < len(src) + CMP R_SEND, R_SRC + BEQ end + + // R_LEN = uint32(src[s]) + // + // switch src[s] & 0x03 + MOVBU (R_SRC), R_LEN + MOVW R_LEN, R_TMP1 + ANDW $3, R_TMP1 + MOVW $1, R1 + CMPW R1, R_TMP1 + BGE tagCopy + + // ---------------------------------------- + // The code below handles literal tags. + + // case tagLiteral: + // x := uint32(src[s] >> 2) + // switch + MOVW $60, R1 + LSRW $2, R_LEN, R_LEN + CMPW R_LEN, R1 + BLS tagLit60Plus + + // case x < 60: + // s++ + ADD $1, R_SRC, R_SRC + +doLit: + // This is the end of the inner "switch", when we have a literal tag. + // + // We assume that R_LEN == x and x fits in a uint32, where x is the variable + // used in the pure Go decode_other.go code. + + // length = int(x) + 1 + // + // Unlike the pure Go code, we don't need to check if length <= 0 because + // R_LEN can hold 64 bits, so the increment cannot overflow. + ADD $1, R_LEN, R_LEN + + // Prepare to check if copying length bytes will run past the end of dst or + // src. + // + // R_TMP0 = len(dst) - d + // R_TMP1 = len(src) - s + MOVD R_DEND, R_TMP0 + SUB R_DST, R_TMP0, R_TMP0 + MOVD R_SEND, R_TMP1 + SUB R_SRC, R_TMP1, R_TMP1 + + // !!! Try a faster technique for short (16 or fewer bytes) copies. + // + // if length > 16 || len(dst)-d < 16 || len(src)-s < 16 { + // goto callMemmove // Fall back on calling runtime·memmove. + // } + // + // The C++ snappy code calls this TryFastAppend. It also checks len(src)-s + // against 21 instead of 16, because it cannot assume that all of its input + // is contiguous in memory and so it needs to leave enough source bytes to + // read the next tag without refilling buffers, but Go's Decode assumes + // contiguousness (the src argument is a []byte). + CMP $16, R_LEN + BGT callMemmove + CMP $16, R_TMP0 + BLT callMemmove + CMP $16, R_TMP1 + BLT callMemmove + + // !!! Implement the copy from src to dst as a 16-byte load and store. + // (Decode's documentation says that dst and src must not overlap.) + // + // This always copies 16 bytes, instead of only length bytes, but that's + // OK. If the input is a valid Snappy encoding then subsequent iterations + // will fix up the overrun. Otherwise, Decode returns a nil []byte (and a + // non-nil error), so the overrun will be ignored. + // + // Note that on arm64, it is legal and cheap to issue unaligned 8-byte or + // 16-byte loads and stores. This technique probably wouldn't be as + // effective on architectures that are fussier about alignment. + LDP 0(R_SRC), (R_TMP2, R_TMP3) + STP (R_TMP2, R_TMP3), 0(R_DST) + + // d += length + // s += length + ADD R_LEN, R_DST, R_DST + ADD R_LEN, R_SRC, R_SRC + B loop + +callMemmove: + // if length > len(dst)-d || length > len(src)-s { etc } + CMP R_TMP0, R_LEN + BGT errCorrupt + CMP R_TMP1, R_LEN + BGT errCorrupt + + // copy(dst[d:], src[s:s+length]) + // + // This means calling runtime·memmove(&dst[d], &src[s], length), so we push + // R_DST, R_SRC and R_LEN as arguments. Coincidentally, we also need to spill those + // three registers to the stack, to save local variables across the CALL. + MOVD R_DST, 8(RSP) + MOVD R_SRC, 16(RSP) + MOVD R_LEN, 24(RSP) + MOVD R_DST, 32(RSP) + MOVD R_SRC, 40(RSP) + MOVD R_LEN, 48(RSP) + MOVD R_OFF, 56(RSP) + CALL runtime·memmove(SB) + + // Restore local variables: unspill registers from the stack and + // re-calculate R_DBASE-R_SEND. + MOVD 32(RSP), R_DST + MOVD 40(RSP), R_SRC + MOVD 48(RSP), R_LEN + MOVD 56(RSP), R_OFF + MOVD dst_base+0(FP), R_DBASE + MOVD dst_len+8(FP), R_DLEN + MOVD R_DBASE, R_DEND + ADD R_DLEN, R_DEND, R_DEND + MOVD src_base+24(FP), R_SBASE + MOVD src_len+32(FP), R_SLEN + MOVD R_SBASE, R_SEND + ADD R_SLEN, R_SEND, R_SEND + + // d += length + // s += length + ADD R_LEN, R_DST, R_DST + ADD R_LEN, R_SRC, R_SRC + B loop + +tagLit60Plus: + // !!! This fragment does the + // + // s += x - 58; if uint(s) > uint(len(src)) { etc } + // + // checks. In the asm version, we code it once instead of once per switch case. + ADD R_LEN, R_SRC, R_SRC + SUB $58, R_SRC, R_SRC + TEST_SRC() + + // case x == 60: + MOVW $61, R1 + CMPW R1, R_LEN + BEQ tagLit61 + BGT tagLit62Plus + + // x = uint32(src[s-1]) + MOVBU -1(R_SRC), R_LEN + B doLit + +tagLit61: + // case x == 61: + // x = uint32(src[s-2]) | uint32(src[s-1])<<8 + MOVHU -2(R_SRC), R_LEN + B doLit + +tagLit62Plus: + CMPW $62, R_LEN + BHI tagLit63 + + // case x == 62: + // x = uint32(src[s-3]) | uint32(src[s-2])<<8 | uint32(src[s-1])<<16 + MOVHU -3(R_SRC), R_LEN + MOVBU -1(R_SRC), R_TMP1 + ORR R_TMP1<<16, R_LEN + B doLit + +tagLit63: + // case x == 63: + // x = uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24 + MOVWU -4(R_SRC), R_LEN + B doLit + + // The code above handles literal tags. + // ---------------------------------------- + // The code below handles copy tags. + +tagCopy4: + // case tagCopy4: + // s += 5 + ADD $5, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + MOVD R_SRC, R_TMP1 + SUB R_SBASE, R_TMP1, R_TMP1 + CMP R_SLEN, R_TMP1 + BGT errCorrupt + + // length = 1 + int(src[s-5])>>2 + MOVD $1, R1 + ADD R_LEN>>2, R1, R_LEN + + // offset = int(uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24) + MOVWU -4(R_SRC), R_OFF + B doCopy + +tagCopy2: + // case tagCopy2: + // s += 3 + ADD $3, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + TEST_SRC() + + // length = 1 + int(src[s-3])>>2 + MOVD $1, R1 + ADD R_LEN>>2, R1, R_LEN + + // offset = int(uint32(src[s-2]) | uint32(src[s-1])<<8) + MOVHU -2(R_SRC), R_OFF + B doCopy + +tagCopy: + // We have a copy tag. We assume that: + // - R_TMP1 == src[s] & 0x03 + // - R_LEN == src[s] + CMP $2, R_TMP1 + BEQ tagCopy2 + BGT tagCopy4 + + // case tagCopy1: + // s += 2 + ADD $2, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + TEST_SRC() + + // offset = int(uint32(src[s-2])&0xe0<<3 | uint32(src[s-1])) + // Calculate offset in R_TMP0 in case it is a repeat. + MOVD R_LEN, R_TMP0 + AND $0xe0, R_TMP0 + MOVBU -1(R_SRC), R_TMP1 + ORR R_TMP0<<3, R_TMP1, R_TMP0 + + // length = 4 + int(src[s-2])>>2&0x7 + MOVD $7, R1 + AND R_LEN>>2, R1, R_LEN + ADD $4, R_LEN, R_LEN + + // check if repeat code with offset 0. + CMP $0, R_TMP0 + BEQ repeatCode + + // This is a regular copy, transfer our temporary value to R_OFF (offset) + MOVD R_TMP0, R_OFF + B doCopy + + // This is a repeat code. +repeatCode: + // If length < 9, reuse last offset, with the length already calculated. + CMP $9, R_LEN + BLT doCopyRepeat + BEQ repeatLen1 + CMP $10, R_LEN + BEQ repeatLen2 + +repeatLen3: + // s +=3 + ADD $3, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + TEST_SRC() + + // length = uint32(src[s-3]) | (uint32(src[s-2])<<8) | (uint32(src[s-1])<<16) + 65540 + MOVBU -1(R_SRC), R_TMP0 + MOVHU -3(R_SRC), R_LEN + ORR R_TMP0<<16, R_LEN, R_LEN + ADD $65540, R_LEN, R_LEN + B doCopyRepeat + +repeatLen2: + // s +=2 + ADD $2, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + TEST_SRC() + + // length = uint32(src[s-2]) | (uint32(src[s-1])<<8) + 260 + MOVHU -2(R_SRC), R_LEN + ADD $260, R_LEN, R_LEN + B doCopyRepeat + +repeatLen1: + // s +=1 + ADD $1, R_SRC, R_SRC + + // if uint(s) > uint(len(src)) { etc } + TEST_SRC() + + // length = src[s-1] + 8 + MOVBU -1(R_SRC), R_LEN + ADD $8, R_LEN, R_LEN + B doCopyRepeat + +doCopy: + // This is the end of the outer "switch", when we have a copy tag. + // + // We assume that: + // - R_LEN == length && R_LEN > 0 + // - R_OFF == offset + + // if d < offset { etc } + MOVD R_DST, R_TMP1 + SUB R_DBASE, R_TMP1, R_TMP1 + CMP R_OFF, R_TMP1 + BLT errCorrupt + + // Repeat values can skip the test above, since any offset > 0 will be in dst. +doCopyRepeat: + + // if offset <= 0 { etc } + CMP $0, R_OFF + BLE errCorrupt + + // if length > len(dst)-d { etc } + MOVD R_DEND, R_TMP1 + SUB R_DST, R_TMP1, R_TMP1 + CMP R_TMP1, R_LEN + BGT errCorrupt + + // forwardCopy(dst[d:d+length], dst[d-offset:]); d += length + // + // Set: + // - R_TMP2 = len(dst)-d + // - R_TMP3 = &dst[d-offset] + MOVD R_DEND, R_TMP2 + SUB R_DST, R_TMP2, R_TMP2 + MOVD R_DST, R_TMP3 + SUB R_OFF, R_TMP3, R_TMP3 + + // !!! Try a faster technique for short (16 or fewer bytes) forward copies. + // + // First, try using two 8-byte load/stores, similar to the doLit technique + // above. Even if dst[d:d+length] and dst[d-offset:] can overlap, this is + // still OK if offset >= 8. Note that this has to be two 8-byte load/stores + // and not one 16-byte load/store, and the first store has to be before the + // second load, due to the overlap if offset is in the range [8, 16). + // + // if length > 16 || offset < 8 || len(dst)-d < 16 { + // goto slowForwardCopy + // } + // copy 16 bytes + // d += length + CMP $16, R_LEN + BGT slowForwardCopy + CMP $8, R_OFF + BLT slowForwardCopy + CMP $16, R_TMP2 + BLT slowForwardCopy + MOVD 0(R_TMP3), R_TMP0 + MOVD R_TMP0, 0(R_DST) + MOVD 8(R_TMP3), R_TMP1 + MOVD R_TMP1, 8(R_DST) + ADD R_LEN, R_DST, R_DST + B loop + +slowForwardCopy: + // !!! If the forward copy is longer than 16 bytes, or if offset < 8, we + // can still try 8-byte load stores, provided we can overrun up to 10 extra + // bytes. As above, the overrun will be fixed up by subsequent iterations + // of the outermost loop. + // + // The C++ snappy code calls this technique IncrementalCopyFastPath. Its + // commentary says: + // + // ---- + // + // The main part of this loop is a simple copy of eight bytes at a time + // until we've copied (at least) the requested amount of bytes. However, + // if d and d-offset are less than eight bytes apart (indicating a + // repeating pattern of length < 8), we first need to expand the pattern in + // order to get the correct results. For instance, if the buffer looks like + // this, with the eight-byte and patterns marked as + // intervals: + // + // abxxxxxxxxxxxx + // [------] d-offset + // [------] d + // + // a single eight-byte copy from to will repeat the pattern + // once, after which we can move two bytes without moving : + // + // ababxxxxxxxxxx + // [------] d-offset + // [------] d + // + // and repeat the exercise until the two no longer overlap. + // + // This allows us to do very well in the special case of one single byte + // repeated many times, without taking a big hit for more general cases. + // + // The worst case of extra writing past the end of the match occurs when + // offset == 1 and length == 1; the last copy will read from byte positions + // [0..7] and write to [4..11], whereas it was only supposed to write to + // position 1. Thus, ten excess bytes. + // + // ---- + // + // That "10 byte overrun" worst case is confirmed by Go's + // TestSlowForwardCopyOverrun, which also tests the fixUpSlowForwardCopy + // and finishSlowForwardCopy algorithm. + // + // if length > len(dst)-d-10 { + // goto verySlowForwardCopy + // } + SUB $10, R_TMP2, R_TMP2 + CMP R_TMP2, R_LEN + BGT verySlowForwardCopy + + // We want to keep the offset, so we use R_TMP2 from here. + MOVD R_OFF, R_TMP2 + +makeOffsetAtLeast8: + // !!! As above, expand the pattern so that offset >= 8 and we can use + // 8-byte load/stores. + // + // for offset < 8 { + // copy 8 bytes from dst[d-offset:] to dst[d:] + // length -= offset + // d += offset + // offset += offset + // // The two previous lines together means that d-offset, and therefore + // // R_TMP3, is unchanged. + // } + CMP $8, R_TMP2 + BGE fixUpSlowForwardCopy + MOVD (R_TMP3), R_TMP1 + MOVD R_TMP1, (R_DST) + SUB R_TMP2, R_LEN, R_LEN + ADD R_TMP2, R_DST, R_DST + ADD R_TMP2, R_TMP2, R_TMP2 + B makeOffsetAtLeast8 + +fixUpSlowForwardCopy: + // !!! Add length (which might be negative now) to d (implied by R_DST being + // &dst[d]) so that d ends up at the right place when we jump back to the + // top of the loop. Before we do that, though, we save R_DST to R_TMP0 so that, if + // length is positive, copying the remaining length bytes will write to the + // right place. + MOVD R_DST, R_TMP0 + ADD R_LEN, R_DST, R_DST + +finishSlowForwardCopy: + // !!! Repeat 8-byte load/stores until length <= 0. Ending with a negative + // length means that we overrun, but as above, that will be fixed up by + // subsequent iterations of the outermost loop. + MOVD $0, R1 + CMP R1, R_LEN + BLE loop + MOVD (R_TMP3), R_TMP1 + MOVD R_TMP1, (R_TMP0) + ADD $8, R_TMP3, R_TMP3 + ADD $8, R_TMP0, R_TMP0 + SUB $8, R_LEN, R_LEN + B finishSlowForwardCopy + +verySlowForwardCopy: + // verySlowForwardCopy is a simple implementation of forward copy. In C + // parlance, this is a do/while loop instead of a while loop, since we know + // that length > 0. In Go syntax: + // + // for { + // dst[d] = dst[d - offset] + // d++ + // length-- + // if length == 0 { + // break + // } + // } + MOVB (R_TMP3), R_TMP1 + MOVB R_TMP1, (R_DST) + ADD $1, R_TMP3, R_TMP3 + ADD $1, R_DST, R_DST + SUB $1, R_LEN, R_LEN + CBNZ R_LEN, verySlowForwardCopy + B loop + + // The code above handles copy tags. + // ---------------------------------------- + +end: + // This is the end of the "for s < len(src)". + // + // if d != len(dst) { etc } + CMP R_DEND, R_DST + BNE errCorrupt + + // return 0 + MOVD $0, ret+48(FP) + RET + +errCorrupt: + // return decodeErrCodeCorrupt + MOVD $1, R_TMP0 + MOVD R_TMP0, ret+48(FP) + RET diff --git a/vendor/github.com/klauspost/compress/s2/decode_asm.go b/vendor/github.com/klauspost/compress/s2/decode_asm.go new file mode 100644 index 000000000..cb3576edd --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/decode_asm.go @@ -0,0 +1,17 @@ +// Copyright 2016 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +//go:build (amd64 || arm64) && !appengine && gc && !noasm +// +build amd64 arm64 +// +build !appengine +// +build gc +// +build !noasm + +package s2 + +// decode has the same semantics as in decode_other.go. +// +//go:noescape +func s2Decode(dst, src []byte) int diff --git a/vendor/github.com/klauspost/compress/s2/decode_other.go b/vendor/github.com/klauspost/compress/s2/decode_other.go new file mode 100644 index 000000000..1074ebd21 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/decode_other.go @@ -0,0 +1,267 @@ +// Copyright 2016 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +//go:build (!amd64 && !arm64) || appengine || !gc || noasm +// +build !amd64,!arm64 appengine !gc noasm + +package s2 + +import ( + "fmt" + "strconv" +) + +// decode writes the decoding of src to dst. It assumes that the varint-encoded +// length of the decompressed bytes has already been read, and that len(dst) +// equals that length. +// +// It returns 0 on success or a decodeErrCodeXxx error code on failure. +func s2Decode(dst, src []byte) int { + const debug = false + if debug { + fmt.Println("Starting decode, dst len:", len(dst)) + } + var d, s, length int + offset := 0 + + // As long as we can read at least 5 bytes... + for s < len(src)-5 { + switch src[s] & 0x03 { + case tagLiteral: + x := uint32(src[s] >> 2) + switch { + case x < 60: + s++ + case x == 60: + s += 2 + x = uint32(src[s-1]) + case x == 61: + s += 3 + x = uint32(src[s-2]) | uint32(src[s-1])<<8 + case x == 62: + s += 4 + x = uint32(src[s-3]) | uint32(src[s-2])<<8 | uint32(src[s-1])<<16 + case x == 63: + s += 5 + x = uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24 + } + length = int(x) + 1 + if length > len(dst)-d || length > len(src)-s || (strconv.IntSize == 32 && length <= 0) { + return decodeErrCodeCorrupt + } + if debug { + fmt.Println("literals, length:", length, "d-after:", d+length) + } + + copy(dst[d:], src[s:s+length]) + d += length + s += length + continue + + case tagCopy1: + s += 2 + length = int(src[s-2]) >> 2 & 0x7 + toffset := int(uint32(src[s-2])&0xe0<<3 | uint32(src[s-1])) + if toffset == 0 { + if debug { + fmt.Print("(repeat) ") + } + // keep last offset + switch length { + case 5: + s += 1 + length = int(uint32(src[s-1])) + 4 + case 6: + s += 2 + length = int(uint32(src[s-2])|(uint32(src[s-1])<<8)) + (1 << 8) + case 7: + s += 3 + length = int(uint32(src[s-3])|(uint32(src[s-2])<<8)|(uint32(src[s-1])<<16)) + (1 << 16) + default: // 0-> 4 + } + } else { + offset = toffset + } + length += 4 + case tagCopy2: + s += 3 + length = 1 + int(src[s-3])>>2 + offset = int(uint32(src[s-2]) | uint32(src[s-1])<<8) + + case tagCopy4: + s += 5 + length = 1 + int(src[s-5])>>2 + offset = int(uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24) + } + + if offset <= 0 || d < offset || length > len(dst)-d { + return decodeErrCodeCorrupt + } + + if debug { + fmt.Println("copy, length:", length, "offset:", offset, "d-after:", d+length) + } + + // Copy from an earlier sub-slice of dst to a later sub-slice. + // If no overlap, use the built-in copy: + if offset > length { + copy(dst[d:d+length], dst[d-offset:]) + d += length + continue + } + + // Unlike the built-in copy function, this byte-by-byte copy always runs + // forwards, even if the slices overlap. Conceptually, this is: + // + // d += forwardCopy(dst[d:d+length], dst[d-offset:]) + // + // We align the slices into a and b and show the compiler they are the same size. + // This allows the loop to run without bounds checks. + a := dst[d : d+length] + b := dst[d-offset:] + b = b[:len(a)] + for i := range a { + a[i] = b[i] + } + d += length + } + + // Remaining with extra checks... + for s < len(src) { + switch src[s] & 0x03 { + case tagLiteral: + x := uint32(src[s] >> 2) + switch { + case x < 60: + s++ + case x == 60: + s += 2 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + x = uint32(src[s-1]) + case x == 61: + s += 3 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + x = uint32(src[s-2]) | uint32(src[s-1])<<8 + case x == 62: + s += 4 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + x = uint32(src[s-3]) | uint32(src[s-2])<<8 | uint32(src[s-1])<<16 + case x == 63: + s += 5 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + x = uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24 + } + length = int(x) + 1 + if length > len(dst)-d || length > len(src)-s || (strconv.IntSize == 32 && length <= 0) { + return decodeErrCodeCorrupt + } + if debug { + fmt.Println("literals, length:", length, "d-after:", d+length) + } + + copy(dst[d:], src[s:s+length]) + d += length + s += length + continue + + case tagCopy1: + s += 2 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = int(src[s-2]) >> 2 & 0x7 + toffset := int(uint32(src[s-2])&0xe0<<3 | uint32(src[s-1])) + if toffset == 0 { + if debug { + fmt.Print("(repeat) ") + } + // keep last offset + switch length { + case 5: + s += 1 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = int(uint32(src[s-1])) + 4 + case 6: + s += 2 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = int(uint32(src[s-2])|(uint32(src[s-1])<<8)) + (1 << 8) + case 7: + s += 3 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = int(uint32(src[s-3])|(uint32(src[s-2])<<8)|(uint32(src[s-1])<<16)) + (1 << 16) + default: // 0-> 4 + } + } else { + offset = toffset + } + length += 4 + case tagCopy2: + s += 3 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = 1 + int(src[s-3])>>2 + offset = int(uint32(src[s-2]) | uint32(src[s-1])<<8) + + case tagCopy4: + s += 5 + if uint(s) > uint(len(src)) { // The uint conversions catch overflow from the previous line. + return decodeErrCodeCorrupt + } + length = 1 + int(src[s-5])>>2 + offset = int(uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24) + } + + if offset <= 0 || d < offset || length > len(dst)-d { + return decodeErrCodeCorrupt + } + + if debug { + fmt.Println("copy, length:", length, "offset:", offset, "d-after:", d+length) + } + + // Copy from an earlier sub-slice of dst to a later sub-slice. + // If no overlap, use the built-in copy: + if offset > length { + copy(dst[d:d+length], dst[d-offset:]) + d += length + continue + } + + // Unlike the built-in copy function, this byte-by-byte copy always runs + // forwards, even if the slices overlap. Conceptually, this is: + // + // d += forwardCopy(dst[d:d+length], dst[d-offset:]) + // + // We align the slices into a and b and show the compiler they are the same size. + // This allows the loop to run without bounds checks. + a := dst[d : d+length] + b := dst[d-offset:] + b = b[:len(a)] + for i := range a { + a[i] = b[i] + } + d += length + } + + if d != len(dst) { + return decodeErrCodeCorrupt + } + return 0 +} diff --git a/vendor/github.com/klauspost/compress/s2/encode.go b/vendor/github.com/klauspost/compress/s2/encode.go new file mode 100644 index 000000000..1aefabf31 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode.go @@ -0,0 +1,1341 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "crypto/rand" + "encoding/binary" + "errors" + "fmt" + "io" + "math" + "math/bits" + "runtime" + "sync" +) + +// Encode returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func Encode(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if cap(dst) < n { + dst = make([]byte, n) + } else { + dst = dst[:n] + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + n := encodeBlock(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// EncodeBetter returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// EncodeBetter compresses better than Encode but typically with a +// 10-40% speed decrease on both compression and decompression. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func EncodeBetter(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if len(dst) < n { + dst = make([]byte, n) + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + n := encodeBlockBetter(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// EncodeBest returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// EncodeBest compresses as good as reasonably possible but with a +// big speed decrease. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func EncodeBest(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if len(dst) < n { + dst = make([]byte, n) + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + n := encodeBlockBest(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// EncodeSnappy returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// The output is Snappy compatible and will likely decompress faster. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func EncodeSnappy(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if cap(dst) < n { + dst = make([]byte, n) + } else { + dst = dst[:n] + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + + n := encodeBlockSnappy(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// EncodeSnappyBetter returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// The output is Snappy compatible and will likely decompress faster. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func EncodeSnappyBetter(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if cap(dst) < n { + dst = make([]byte, n) + } else { + dst = dst[:n] + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + + n := encodeBlockBetterSnappy(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// EncodeSnappyBest returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// +// The output is Snappy compatible and will likely decompress faster. +// +// The dst and src must not overlap. It is valid to pass a nil dst. +// +// The blocks will require the same amount of memory to decode as encoding, +// and does not make for concurrent decoding. +// Also note that blocks do not contain CRC information, so corruption may be undetected. +// +// If you need to encode larger amounts of data, consider using +// the streaming interface which gives all of these features. +func EncodeSnappyBest(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if cap(dst) < n { + dst = make([]byte, n) + } else { + dst = dst[:n] + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + + n := encodeBlockBestSnappy(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// ConcatBlocks will concatenate the supplied blocks and append them to the supplied destination. +// If the destination is nil or too small, a new will be allocated. +// The blocks are not validated, so garbage in = garbage out. +// dst may not overlap block data. +// Any data in dst is preserved as is, so it will not be considered a block. +func ConcatBlocks(dst []byte, blocks ...[]byte) ([]byte, error) { + totalSize := uint64(0) + compSize := 0 + for _, b := range blocks { + l, hdr, err := decodedLen(b) + if err != nil { + return nil, err + } + totalSize += uint64(l) + compSize += len(b) - hdr + } + if totalSize == 0 { + dst = append(dst, 0) + return dst, nil + } + if totalSize > math.MaxUint32 { + return nil, ErrTooLarge + } + var tmp [binary.MaxVarintLen32]byte + hdrSize := binary.PutUvarint(tmp[:], totalSize) + wantSize := hdrSize + compSize + + if cap(dst)-len(dst) < wantSize { + dst = append(make([]byte, 0, wantSize+len(dst)), dst...) + } + dst = append(dst, tmp[:hdrSize]...) + for _, b := range blocks { + _, hdr, err := decodedLen(b) + if err != nil { + return nil, err + } + dst = append(dst, b[hdr:]...) + } + return dst, nil +} + +// inputMargin is the minimum number of extra input bytes to keep, inside +// encodeBlock's inner loop. On some architectures, this margin lets us +// implement a fast path for emitLiteral, where the copy of short (<= 16 byte) +// literals can be implemented as a single load to and store from a 16-byte +// register. That literal's actual length can be as short as 1 byte, so this +// can copy up to 15 bytes too much, but that's OK as subsequent iterations of +// the encoding loop will fix up the copy overrun, and this inputMargin ensures +// that we don't overrun the dst and src buffers. +const inputMargin = 8 + +// minNonLiteralBlockSize is the minimum size of the input to encodeBlock that +// will be accepted by the encoder. +const minNonLiteralBlockSize = 32 + +// MaxBlockSize is the maximum value where MaxEncodedLen will return a valid block size. +// Blocks this big are highly discouraged, though. +const MaxBlockSize = math.MaxUint32 - binary.MaxVarintLen32 - 5 + +// MaxEncodedLen returns the maximum length of a snappy block, given its +// uncompressed length. +// +// It will return a negative value if srcLen is too large to encode. +// 32 bit platforms will have lower thresholds for rejecting big content. +func MaxEncodedLen(srcLen int) int { + n := uint64(srcLen) + if n > 0xffffffff { + // Also includes negative. + return -1 + } + // Size of the varint encoded block size. + n = n + uint64((bits.Len64(n)+7)/7) + + // Add maximum size of encoding block as literals. + n += uint64(literalExtraSize(int64(srcLen))) + if n > 0xffffffff { + return -1 + } + return int(n) +} + +var errClosed = errors.New("s2: Writer is closed") + +// NewWriter returns a new Writer that compresses to w, using the +// framing format described at +// https://github.com/google/snappy/blob/master/framing_format.txt +// +// Users must call Close to guarantee all data has been forwarded to +// the underlying io.Writer and that resources are released. +// They may also call Flush zero or more times before calling Close. +func NewWriter(w io.Writer, opts ...WriterOption) *Writer { + w2 := Writer{ + blockSize: defaultBlockSize, + concurrency: runtime.GOMAXPROCS(0), + randSrc: rand.Reader, + level: levelFast, + } + for _, opt := range opts { + if err := opt(&w2); err != nil { + w2.errState = err + return &w2 + } + } + w2.obufLen = obufHeaderLen + MaxEncodedLen(w2.blockSize) + w2.paramsOK = true + w2.ibuf = make([]byte, 0, w2.blockSize) + w2.buffers.New = func() interface{} { + return make([]byte, w2.obufLen) + } + w2.Reset(w) + return &w2 +} + +// Writer is an io.Writer that can write Snappy-compressed bytes. +type Writer struct { + errMu sync.Mutex + errState error + + // ibuf is a buffer for the incoming (uncompressed) bytes. + ibuf []byte + + blockSize int + obufLen int + concurrency int + written int64 + uncompWritten int64 // Bytes sent to compression + output chan chan result + buffers sync.Pool + pad int + + writer io.Writer + randSrc io.Reader + writerWg sync.WaitGroup + index Index + + // wroteStreamHeader is whether we have written the stream header. + wroteStreamHeader bool + paramsOK bool + snappy bool + flushOnWrite bool + appendIndex bool + level uint8 +} + +const ( + levelUncompressed = iota + 1 + levelFast + levelBetter + levelBest +) + +type result struct { + b []byte + // Uncompressed start offset + startOffset int64 +} + +// err returns the previously set error. +// If no error has been set it is set to err if not nil. +func (w *Writer) err(err error) error { + w.errMu.Lock() + errSet := w.errState + if errSet == nil && err != nil { + w.errState = err + errSet = err + } + w.errMu.Unlock() + return errSet +} + +// Reset discards the writer's state and switches the Snappy writer to write to w. +// This permits reusing a Writer rather than allocating a new one. +func (w *Writer) Reset(writer io.Writer) { + if !w.paramsOK { + return + } + // Close previous writer, if any. + if w.output != nil { + close(w.output) + w.writerWg.Wait() + w.output = nil + } + w.errState = nil + w.ibuf = w.ibuf[:0] + w.wroteStreamHeader = false + w.written = 0 + w.writer = writer + w.uncompWritten = 0 + w.index.reset(w.blockSize) + + // If we didn't get a writer, stop here. + if writer == nil { + return + } + // If no concurrency requested, don't spin up writer goroutine. + if w.concurrency == 1 { + return + } + + toWrite := make(chan chan result, w.concurrency) + w.output = toWrite + w.writerWg.Add(1) + + // Start a writer goroutine that will write all output in order. + go func() { + defer w.writerWg.Done() + + // Get a queued write. + for write := range toWrite { + // Wait for the data to be available. + input := <-write + in := input.b + if len(in) > 0 { + if w.err(nil) == nil { + // Don't expose data from previous buffers. + toWrite := in[:len(in):len(in)] + // Write to output. + n, err := writer.Write(toWrite) + if err == nil && n != len(toWrite) { + err = io.ErrShortBuffer + } + _ = w.err(err) + w.err(w.index.add(w.written, input.startOffset)) + w.written += int64(n) + } + } + if cap(in) >= w.obufLen { + w.buffers.Put(in) + } + // close the incoming write request. + // This can be used for synchronizing flushes. + close(write) + } + }() +} + +// Write satisfies the io.Writer interface. +func (w *Writer) Write(p []byte) (nRet int, errRet error) { + if err := w.err(nil); err != nil { + return 0, err + } + if w.flushOnWrite { + return w.write(p) + } + // If we exceed the input buffer size, start writing + for len(p) > (cap(w.ibuf)-len(w.ibuf)) && w.err(nil) == nil { + var n int + if len(w.ibuf) == 0 { + // Large write, empty buffer. + // Write directly from p to avoid copy. + n, _ = w.write(p) + } else { + n = copy(w.ibuf[len(w.ibuf):cap(w.ibuf)], p) + w.ibuf = w.ibuf[:len(w.ibuf)+n] + w.write(w.ibuf) + w.ibuf = w.ibuf[:0] + } + nRet += n + p = p[n:] + } + if err := w.err(nil); err != nil { + return nRet, err + } + // p should always be able to fit into w.ibuf now. + n := copy(w.ibuf[len(w.ibuf):cap(w.ibuf)], p) + w.ibuf = w.ibuf[:len(w.ibuf)+n] + nRet += n + return nRet, nil +} + +// ReadFrom implements the io.ReaderFrom interface. +// Using this is typically more efficient since it avoids a memory copy. +// ReadFrom reads data from r until EOF or error. +// The return value n is the number of bytes read. +// Any error except io.EOF encountered during the read is also returned. +func (w *Writer) ReadFrom(r io.Reader) (n int64, err error) { + if err := w.err(nil); err != nil { + return 0, err + } + if len(w.ibuf) > 0 { + err := w.Flush() + if err != nil { + return 0, err + } + } + if br, ok := r.(byter); ok { + buf := br.Bytes() + if err := w.EncodeBuffer(buf); err != nil { + return 0, err + } + return int64(len(buf)), w.Flush() + } + for { + inbuf := w.buffers.Get().([]byte)[:w.blockSize+obufHeaderLen] + n2, err := io.ReadFull(r, inbuf[obufHeaderLen:]) + if err != nil { + if err == io.ErrUnexpectedEOF { + err = io.EOF + } + if err != io.EOF { + return n, w.err(err) + } + } + if n2 == 0 { + break + } + n += int64(n2) + err2 := w.writeFull(inbuf[:n2+obufHeaderLen]) + if w.err(err2) != nil { + break + } + + if err != nil { + // We got EOF and wrote everything + break + } + } + + return n, w.err(nil) +} + +// AddSkippableBlock will add a skippable block to the stream. +// The ID must be 0x80-0xfe (inclusive). +// Length of the skippable block must be <= 16777215 bytes. +func (w *Writer) AddSkippableBlock(id uint8, data []byte) (err error) { + if err := w.err(nil); err != nil { + return err + } + if len(data) == 0 { + return nil + } + if id < 0x80 || id > chunkTypePadding { + return fmt.Errorf("invalid skippable block id %x", id) + } + if len(data) > maxChunkSize { + return fmt.Errorf("skippable block excessed maximum size") + } + var header [4]byte + chunkLen := 4 + len(data) + header[0] = id + header[1] = uint8(chunkLen >> 0) + header[2] = uint8(chunkLen >> 8) + header[3] = uint8(chunkLen >> 16) + if w.concurrency == 1 { + write := func(b []byte) error { + n, err := w.writer.Write(b) + if err = w.err(err); err != nil { + return err + } + if n != len(data) { + return w.err(io.ErrShortWrite) + } + w.written += int64(n) + return w.err(nil) + } + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + if w.snappy { + if err := write([]byte(magicChunkSnappy)); err != nil { + return err + } + } else { + if err := write([]byte(magicChunk)); err != nil { + return err + } + } + } + if err := write(header[:]); err != nil { + return err + } + if err := write(data); err != nil { + return err + } + } + + // Create output... + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + hWriter := make(chan result) + w.output <- hWriter + if w.snappy { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunkSnappy)} + } else { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunk)} + } + } + + // Copy input. + inbuf := w.buffers.Get().([]byte)[:4] + copy(inbuf, header[:]) + inbuf = append(inbuf, data...) + + output := make(chan result, 1) + // Queue output. + w.output <- output + output <- result{startOffset: w.uncompWritten, b: inbuf} + + return nil +} + +// EncodeBuffer will add a buffer to the stream. +// This is the fastest way to encode a stream, +// but the input buffer cannot be written to by the caller +// until Flush or Close has been called when concurrency != 1. +// +// If you cannot control that, use the regular Write function. +// +// Note that input is not buffered. +// This means that each write will result in discrete blocks being created. +// For buffered writes, use the regular Write function. +func (w *Writer) EncodeBuffer(buf []byte) (err error) { + if err := w.err(nil); err != nil { + return err + } + + if w.flushOnWrite { + _, err := w.write(buf) + return err + } + // Flush queued data first. + if len(w.ibuf) > 0 { + err := w.Flush() + if err != nil { + return err + } + } + if w.concurrency == 1 { + _, err := w.writeSync(buf) + return err + } + + // Spawn goroutine and write block to output channel. + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + hWriter := make(chan result) + w.output <- hWriter + if w.snappy { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunkSnappy)} + } else { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunk)} + } + } + + for len(buf) > 0 { + // Cut input. + uncompressed := buf + if len(uncompressed) > w.blockSize { + uncompressed = uncompressed[:w.blockSize] + } + buf = buf[len(uncompressed):] + // Get an output buffer. + obuf := w.buffers.Get().([]byte)[:len(uncompressed)+obufHeaderLen] + output := make(chan result) + // Queue output now, so we keep order. + w.output <- output + res := result{ + startOffset: w.uncompWritten, + } + w.uncompWritten += int64(len(uncompressed)) + go func() { + checksum := crc(uncompressed) + + // Set to uncompressed. + chunkType := uint8(chunkTypeUncompressedData) + chunkLen := 4 + len(uncompressed) + + // Attempt compressing. + n := binary.PutUvarint(obuf[obufHeaderLen:], uint64(len(uncompressed))) + n2 := w.encodeBlock(obuf[obufHeaderLen+n:], uncompressed) + + // Check if we should use this, or store as uncompressed instead. + if n2 > 0 { + chunkType = uint8(chunkTypeCompressedData) + chunkLen = 4 + n + n2 + obuf = obuf[:obufHeaderLen+n+n2] + } else { + // copy uncompressed + copy(obuf[obufHeaderLen:], uncompressed) + } + + // Fill in the per-chunk header that comes before the body. + obuf[0] = chunkType + obuf[1] = uint8(chunkLen >> 0) + obuf[2] = uint8(chunkLen >> 8) + obuf[3] = uint8(chunkLen >> 16) + obuf[4] = uint8(checksum >> 0) + obuf[5] = uint8(checksum >> 8) + obuf[6] = uint8(checksum >> 16) + obuf[7] = uint8(checksum >> 24) + + // Queue final output. + res.b = obuf + output <- res + }() + } + return nil +} + +func (w *Writer) encodeBlock(obuf, uncompressed []byte) int { + if w.snappy { + switch w.level { + case levelFast: + return encodeBlockSnappy(obuf, uncompressed) + case levelBetter: + return encodeBlockBetterSnappy(obuf, uncompressed) + case levelBest: + return encodeBlockBestSnappy(obuf, uncompressed) + } + return 0 + } + switch w.level { + case levelFast: + return encodeBlock(obuf, uncompressed) + case levelBetter: + return encodeBlockBetter(obuf, uncompressed) + case levelBest: + return encodeBlockBest(obuf, uncompressed) + } + return 0 +} + +func (w *Writer) write(p []byte) (nRet int, errRet error) { + if err := w.err(nil); err != nil { + return 0, err + } + if w.concurrency == 1 { + return w.writeSync(p) + } + + // Spawn goroutine and write block to output channel. + for len(p) > 0 { + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + hWriter := make(chan result) + w.output <- hWriter + if w.snappy { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunkSnappy)} + } else { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunk)} + } + } + + var uncompressed []byte + if len(p) > w.blockSize { + uncompressed, p = p[:w.blockSize], p[w.blockSize:] + } else { + uncompressed, p = p, nil + } + + // Copy input. + // If the block is incompressible, this is used for the result. + inbuf := w.buffers.Get().([]byte)[:len(uncompressed)+obufHeaderLen] + obuf := w.buffers.Get().([]byte)[:w.obufLen] + copy(inbuf[obufHeaderLen:], uncompressed) + uncompressed = inbuf[obufHeaderLen:] + + output := make(chan result) + // Queue output now, so we keep order. + w.output <- output + res := result{ + startOffset: w.uncompWritten, + } + w.uncompWritten += int64(len(uncompressed)) + + go func() { + checksum := crc(uncompressed) + + // Set to uncompressed. + chunkType := uint8(chunkTypeUncompressedData) + chunkLen := 4 + len(uncompressed) + + // Attempt compressing. + n := binary.PutUvarint(obuf[obufHeaderLen:], uint64(len(uncompressed))) + n2 := w.encodeBlock(obuf[obufHeaderLen+n:], uncompressed) + + // Check if we should use this, or store as uncompressed instead. + if n2 > 0 { + chunkType = uint8(chunkTypeCompressedData) + chunkLen = 4 + n + n2 + obuf = obuf[:obufHeaderLen+n+n2] + } else { + // Use input as output. + obuf, inbuf = inbuf, obuf + } + + // Fill in the per-chunk header that comes before the body. + obuf[0] = chunkType + obuf[1] = uint8(chunkLen >> 0) + obuf[2] = uint8(chunkLen >> 8) + obuf[3] = uint8(chunkLen >> 16) + obuf[4] = uint8(checksum >> 0) + obuf[5] = uint8(checksum >> 8) + obuf[6] = uint8(checksum >> 16) + obuf[7] = uint8(checksum >> 24) + + // Queue final output. + res.b = obuf + output <- res + + // Put unused buffer back in pool. + w.buffers.Put(inbuf) + }() + nRet += len(uncompressed) + } + return nRet, nil +} + +// writeFull is a special version of write that will always write the full buffer. +// Data to be compressed should start at offset obufHeaderLen and fill the remainder of the buffer. +// The data will be written as a single block. +// The caller is not allowed to use inbuf after this function has been called. +func (w *Writer) writeFull(inbuf []byte) (errRet error) { + if err := w.err(nil); err != nil { + return err + } + + if w.concurrency == 1 { + _, err := w.writeSync(inbuf[obufHeaderLen:]) + return err + } + + // Spawn goroutine and write block to output channel. + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + hWriter := make(chan result) + w.output <- hWriter + if w.snappy { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunkSnappy)} + } else { + hWriter <- result{startOffset: w.uncompWritten, b: []byte(magicChunk)} + } + } + + // Get an output buffer. + obuf := w.buffers.Get().([]byte)[:w.obufLen] + uncompressed := inbuf[obufHeaderLen:] + + output := make(chan result) + // Queue output now, so we keep order. + w.output <- output + res := result{ + startOffset: w.uncompWritten, + } + w.uncompWritten += int64(len(uncompressed)) + + go func() { + checksum := crc(uncompressed) + + // Set to uncompressed. + chunkType := uint8(chunkTypeUncompressedData) + chunkLen := 4 + len(uncompressed) + + // Attempt compressing. + n := binary.PutUvarint(obuf[obufHeaderLen:], uint64(len(uncompressed))) + n2 := w.encodeBlock(obuf[obufHeaderLen+n:], uncompressed) + + // Check if we should use this, or store as uncompressed instead. + if n2 > 0 { + chunkType = uint8(chunkTypeCompressedData) + chunkLen = 4 + n + n2 + obuf = obuf[:obufHeaderLen+n+n2] + } else { + // Use input as output. + obuf, inbuf = inbuf, obuf + } + + // Fill in the per-chunk header that comes before the body. + obuf[0] = chunkType + obuf[1] = uint8(chunkLen >> 0) + obuf[2] = uint8(chunkLen >> 8) + obuf[3] = uint8(chunkLen >> 16) + obuf[4] = uint8(checksum >> 0) + obuf[5] = uint8(checksum >> 8) + obuf[6] = uint8(checksum >> 16) + obuf[7] = uint8(checksum >> 24) + + // Queue final output. + res.b = obuf + output <- res + + // Put unused buffer back in pool. + w.buffers.Put(inbuf) + }() + return nil +} + +func (w *Writer) writeSync(p []byte) (nRet int, errRet error) { + if err := w.err(nil); err != nil { + return 0, err + } + if !w.wroteStreamHeader { + w.wroteStreamHeader = true + var n int + var err error + if w.snappy { + n, err = w.writer.Write([]byte(magicChunkSnappy)) + } else { + n, err = w.writer.Write([]byte(magicChunk)) + } + if err != nil { + return 0, w.err(err) + } + if n != len(magicChunk) { + return 0, w.err(io.ErrShortWrite) + } + w.written += int64(n) + } + + for len(p) > 0 { + var uncompressed []byte + if len(p) > w.blockSize { + uncompressed, p = p[:w.blockSize], p[w.blockSize:] + } else { + uncompressed, p = p, nil + } + + obuf := w.buffers.Get().([]byte)[:w.obufLen] + checksum := crc(uncompressed) + + // Set to uncompressed. + chunkType := uint8(chunkTypeUncompressedData) + chunkLen := 4 + len(uncompressed) + + // Attempt compressing. + n := binary.PutUvarint(obuf[obufHeaderLen:], uint64(len(uncompressed))) + n2 := w.encodeBlock(obuf[obufHeaderLen+n:], uncompressed) + + if n2 > 0 { + chunkType = uint8(chunkTypeCompressedData) + chunkLen = 4 + n + n2 + obuf = obuf[:obufHeaderLen+n+n2] + } else { + obuf = obuf[:8] + } + + // Fill in the per-chunk header that comes before the body. + obuf[0] = chunkType + obuf[1] = uint8(chunkLen >> 0) + obuf[2] = uint8(chunkLen >> 8) + obuf[3] = uint8(chunkLen >> 16) + obuf[4] = uint8(checksum >> 0) + obuf[5] = uint8(checksum >> 8) + obuf[6] = uint8(checksum >> 16) + obuf[7] = uint8(checksum >> 24) + + n, err := w.writer.Write(obuf) + if err != nil { + return 0, w.err(err) + } + if n != len(obuf) { + return 0, w.err(io.ErrShortWrite) + } + w.err(w.index.add(w.written, w.uncompWritten)) + w.written += int64(n) + w.uncompWritten += int64(len(uncompressed)) + + if chunkType == chunkTypeUncompressedData { + // Write uncompressed data. + n, err := w.writer.Write(uncompressed) + if err != nil { + return 0, w.err(err) + } + if n != len(uncompressed) { + return 0, w.err(io.ErrShortWrite) + } + w.written += int64(n) + } + w.buffers.Put(obuf) + // Queue final output. + nRet += len(uncompressed) + } + return nRet, nil +} + +// Flush flushes the Writer to its underlying io.Writer. +// This does not apply padding. +func (w *Writer) Flush() error { + if err := w.err(nil); err != nil { + return err + } + + // Queue any data still in input buffer. + if len(w.ibuf) != 0 { + if !w.wroteStreamHeader { + _, err := w.writeSync(w.ibuf) + w.ibuf = w.ibuf[:0] + return w.err(err) + } else { + _, err := w.write(w.ibuf) + w.ibuf = w.ibuf[:0] + err = w.err(err) + if err != nil { + return err + } + } + } + if w.output == nil { + return w.err(nil) + } + + // Send empty buffer + res := make(chan result) + w.output <- res + // Block until this has been picked up. + res <- result{b: nil, startOffset: w.uncompWritten} + // When it is closed, we have flushed. + <-res + return w.err(nil) +} + +// Close calls Flush and then closes the Writer. +// Calling Close multiple times is ok, +// but calling CloseIndex after this will make it not return the index. +func (w *Writer) Close() error { + _, err := w.closeIndex(w.appendIndex) + return err +} + +// CloseIndex calls Close and returns an index on first call. +// This is not required if you are only adding index to a stream. +func (w *Writer) CloseIndex() ([]byte, error) { + return w.closeIndex(true) +} + +func (w *Writer) closeIndex(idx bool) ([]byte, error) { + err := w.Flush() + if w.output != nil { + close(w.output) + w.writerWg.Wait() + w.output = nil + } + + var index []byte + if w.err(nil) == nil && w.writer != nil { + // Create index. + if idx { + compSize := int64(-1) + if w.pad <= 1 { + compSize = w.written + } + index = w.index.appendTo(w.ibuf[:0], w.uncompWritten, compSize) + // Count as written for padding. + if w.appendIndex { + w.written += int64(len(index)) + } + } + + if w.pad > 1 { + tmp := w.ibuf[:0] + if len(index) > 0 { + // Allocate another buffer. + tmp = w.buffers.Get().([]byte)[:0] + defer w.buffers.Put(tmp) + } + add := calcSkippableFrame(w.written, int64(w.pad)) + frame, err := skippableFrame(tmp, add, w.randSrc) + if err = w.err(err); err != nil { + return nil, err + } + n, err2 := w.writer.Write(frame) + if err2 == nil && n != len(frame) { + err2 = io.ErrShortWrite + } + _ = w.err(err2) + } + if len(index) > 0 && w.appendIndex { + n, err2 := w.writer.Write(index) + if err2 == nil && n != len(index) { + err2 = io.ErrShortWrite + } + _ = w.err(err2) + } + } + err = w.err(errClosed) + if err == errClosed { + return index, nil + } + return nil, err +} + +// calcSkippableFrame will return a total size to be added for written +// to be divisible by multiple. +// The value will always be > skippableFrameHeader. +// The function will panic if written < 0 or wantMultiple <= 0. +func calcSkippableFrame(written, wantMultiple int64) int { + if wantMultiple <= 0 { + panic("wantMultiple <= 0") + } + if written < 0 { + panic("written < 0") + } + leftOver := written % wantMultiple + if leftOver == 0 { + return 0 + } + toAdd := wantMultiple - leftOver + for toAdd < skippableFrameHeader { + toAdd += wantMultiple + } + return int(toAdd) +} + +// skippableFrame will add a skippable frame with a total size of bytes. +// total should be >= skippableFrameHeader and < maxBlockSize + skippableFrameHeader +func skippableFrame(dst []byte, total int, r io.Reader) ([]byte, error) { + if total == 0 { + return dst, nil + } + if total < skippableFrameHeader { + return dst, fmt.Errorf("s2: requested skippable frame (%d) < 4", total) + } + if int64(total) >= maxBlockSize+skippableFrameHeader { + return dst, fmt.Errorf("s2: requested skippable frame (%d) >= max 1<<24", total) + } + // Chunk type 0xfe "Section 4.4 Padding (chunk type 0xfe)" + dst = append(dst, chunkTypePadding) + f := uint32(total - skippableFrameHeader) + // Add chunk length. + dst = append(dst, uint8(f), uint8(f>>8), uint8(f>>16)) + // Add data + start := len(dst) + dst = append(dst, make([]byte, f)...) + _, err := io.ReadFull(r, dst[start:]) + return dst, err +} + +// WriterOption is an option for creating a encoder. +type WriterOption func(*Writer) error + +// WriterConcurrency will set the concurrency, +// meaning the maximum number of decoders to run concurrently. +// The value supplied must be at least 1. +// By default this will be set to GOMAXPROCS. +func WriterConcurrency(n int) WriterOption { + return func(w *Writer) error { + if n <= 0 { + return errors.New("concurrency must be at least 1") + } + w.concurrency = n + return nil + } +} + +// WriterAddIndex will append an index to the end of a stream +// when it is closed. +func WriterAddIndex() WriterOption { + return func(w *Writer) error { + w.appendIndex = true + return nil + } +} + +// WriterBetterCompression will enable better compression. +// EncodeBetter compresses better than Encode but typically with a +// 10-40% speed decrease on both compression and decompression. +func WriterBetterCompression() WriterOption { + return func(w *Writer) error { + w.level = levelBetter + return nil + } +} + +// WriterBestCompression will enable better compression. +// EncodeBetter compresses better than Encode but typically with a +// big speed decrease on compression. +func WriterBestCompression() WriterOption { + return func(w *Writer) error { + w.level = levelBest + return nil + } +} + +// WriterUncompressed will bypass compression. +// The stream will be written as uncompressed blocks only. +// If concurrency is > 1 CRC and output will still be done async. +func WriterUncompressed() WriterOption { + return func(w *Writer) error { + w.level = levelUncompressed + return nil + } +} + +// WriterBlockSize allows to override the default block size. +// Blocks will be this size or smaller. +// Minimum size is 4KB and and maximum size is 4MB. +// +// Bigger blocks may give bigger throughput on systems with many cores, +// and will increase compression slightly, but it will limit the possible +// concurrency for smaller payloads for both encoding and decoding. +// Default block size is 1MB. +// +// When writing Snappy compatible output using WriterSnappyCompat, +// the maximum block size is 64KB. +func WriterBlockSize(n int) WriterOption { + return func(w *Writer) error { + if w.snappy && n > maxSnappyBlockSize || n < minBlockSize { + return errors.New("s2: block size too large. Must be <= 64K and >=4KB on for snappy compatible output") + } + if n > maxBlockSize || n < minBlockSize { + return errors.New("s2: block size too large. Must be <= 4MB and >=4KB") + } + w.blockSize = n + return nil + } +} + +// WriterPadding will add padding to all output so the size will be a multiple of n. +// This can be used to obfuscate the exact output size or make blocks of a certain size. +// The contents will be a skippable frame, so it will be invisible by the decoder. +// n must be > 0 and <= 4MB. +// The padded area will be filled with data from crypto/rand.Reader. +// The padding will be applied whenever Close is called on the writer. +func WriterPadding(n int) WriterOption { + return func(w *Writer) error { + if n <= 0 { + return fmt.Errorf("s2: padding must be at least 1") + } + // No need to waste our time. + if n == 1 { + w.pad = 0 + } + if n > maxBlockSize { + return fmt.Errorf("s2: padding must less than 4MB") + } + w.pad = n + return nil + } +} + +// WriterPaddingSrc will get random data for padding from the supplied source. +// By default crypto/rand is used. +func WriterPaddingSrc(reader io.Reader) WriterOption { + return func(w *Writer) error { + w.randSrc = reader + return nil + } +} + +// WriterSnappyCompat will write snappy compatible output. +// The output can be decompressed using either snappy or s2. +// If block size is more than 64KB it is set to that. +func WriterSnappyCompat() WriterOption { + return func(w *Writer) error { + w.snappy = true + if w.blockSize > 64<<10 { + // We choose 8 bytes less than 64K, since that will make literal emits slightly more effective. + // And allows us to skip some size checks. + w.blockSize = (64 << 10) - 8 + } + return nil + } +} + +// WriterFlushOnWrite will compress blocks on each call to the Write function. +// +// This is quite inefficient as blocks size will depend on the write size. +// +// Use WriterConcurrency(1) to also make sure that output is flushed. +// When Write calls return, otherwise they will be written when compression is done. +func WriterFlushOnWrite() WriterOption { + return func(w *Writer) error { + w.flushOnWrite = true + return nil + } +} diff --git a/vendor/github.com/klauspost/compress/s2/encode_all.go b/vendor/github.com/klauspost/compress/s2/encode_all.go new file mode 100644 index 000000000..8b16c38a6 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode_all.go @@ -0,0 +1,456 @@ +// Copyright 2016 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "bytes" + "encoding/binary" + "math/bits" +) + +func load32(b []byte, i int) uint32 { + return binary.LittleEndian.Uint32(b[i:]) +} + +func load64(b []byte, i int) uint64 { + return binary.LittleEndian.Uint64(b[i:]) +} + +// hash6 returns the hash of the lowest 6 bytes of u to fit in a hash table with h bits. +// Preferably h should be a constant and should always be <64. +func hash6(u uint64, h uint8) uint32 { + const prime6bytes = 227718039650203 + return uint32(((u << (64 - 48)) * prime6bytes) >> ((64 - h) & 63)) +} + +func encodeGo(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); n < 0 { + panic(ErrTooLarge) + } else if len(dst) < n { + dst = make([]byte, n) + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + if len(src) == 0 { + return dst[:d] + } + if len(src) < minNonLiteralBlockSize { + d += emitLiteral(dst[d:], src) + return dst[:d] + } + n := encodeBlockGo(dst[d:], src) + if n > 0 { + d += n + return dst[:d] + } + // Not compressible + d += emitLiteral(dst[d:], src) + return dst[:d] +} + +// encodeBlockGo encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockGo(dst, src []byte) (d int) { + // Initialize the hash table. + const ( + tableBits = 14 + maxTableSize = 1 << tableBits + + debug = false + ) + + var table [maxTableSize]uint32 + + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + + // Bail if we can't compress to at least this. + dstLimit := len(src) - len(src)>>5 - 5 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We search for a repeat at -1, but don't output repeats when nextEmit == 0 + repeat := 1 + + for { + candidate := 0 + for { + // Next src position to check + nextS := s + (s-nextEmit)>>6 + 4 + if nextS > sLimit { + goto emitRemainder + } + hash0 := hash6(cv, tableBits) + hash1 := hash6(cv>>8, tableBits) + candidate = int(table[hash0]) + candidate2 := int(table[hash1]) + table[hash0] = uint32(s) + table[hash1] = uint32(s + 1) + hash2 := hash6(cv>>16, tableBits) + + // Check repeat at offset checkRep. + const checkRep = 1 + if uint32(cv>>(checkRep*8)) == load32(src, s-repeat+checkRep) { + base := s + checkRep + // Extend back + for i := base - repeat; base > nextEmit && i > 0 && src[i-1] == src[base-1]; { + i-- + base-- + } + d += emitLiteral(dst[d:], src[nextEmit:base]) + + // Extend forward + candidate := s - repeat + 4 + checkRep + s += 4 + checkRep + for s <= sLimit { + if diff := load64(src, s) ^ load64(src, candidate); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidate += 8 + } + if debug { + // Validate match. + if s <= candidate { + panic("s <= candidate") + } + a := src[base:s] + b := src[base-repeat : base-repeat+(s-base)] + if !bytes.Equal(a, b) { + panic("mismatch") + } + } + if nextEmit > 0 { + // same as `add := emitCopy(dst[d:], repeat, s-base)` but skips storing offset. + d += emitRepeat(dst[d:], repeat, s-base) + } else { + // First match, cannot be repeat. + d += emitCopy(dst[d:], repeat, s-base) + } + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + cv = load64(src, s) + continue + } + + if uint32(cv) == load32(src, candidate) { + break + } + candidate = int(table[hash2]) + if uint32(cv>>8) == load32(src, candidate2) { + table[hash2] = uint32(s + 2) + candidate = candidate2 + s++ + break + } + table[hash2] = uint32(s + 2) + if uint32(cv>>16) == load32(src, candidate) { + s += 2 + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards. + // The top bytes will be rechecked to get the full match. + for candidate > 0 && s > nextEmit && src[candidate-1] == src[s-1] { + candidate-- + s-- + } + + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + // A 4-byte match has been found. We'll later see if more than 4 bytes + // match. But, prior to the match, src[nextEmit:s] are unmatched. Emit + // them as literal bytes. + + d += emitLiteral(dst[d:], src[nextEmit:s]) + + // Call emitCopy, and then see if another emitCopy could be our next + // move. Repeat until we find no match for the input immediately after + // what was consumed by the last emitCopy call. + // + // If we exit this loop normally then we need to call emitLiteral next, + // though we don't yet know how big the literal will be. We handle that + // by proceeding to the next iteration of the main loop. We also can + // exit this loop via goto if we get close to exhausting the input. + for { + // Invariant: we have a 4-byte match at s, and no need to emit any + // literal bytes prior to s. + base := s + repeat = base - candidate + + // Extend the 4-byte match as long as possible. + s += 4 + candidate += 4 + for s <= len(src)-8 { + if diff := load64(src, s) ^ load64(src, candidate); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidate += 8 + } + + d += emitCopy(dst[d:], repeat, s-base) + if debug { + // Validate match. + if s <= candidate { + panic("s <= candidate") + } + a := src[base:s] + b := src[base-repeat : base-repeat+(s-base)] + if !bytes.Equal(a, b) { + panic("mismatch") + } + } + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Check for an immediate match, otherwise start search at s+1 + x := load64(src, s-2) + m2Hash := hash6(x, tableBits) + currHash := hash6(x>>16, tableBits) + candidate = int(table[currHash]) + table[m2Hash] = uint32(s - 2) + table[currHash] = uint32(s) + if debug && s == candidate { + panic("s == candidate") + } + if uint32(x>>16) != load32(src, candidate) { + cv = load64(src, s+1) + s++ + break + } + } + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} + +func encodeBlockSnappyGo(dst, src []byte) (d int) { + // Initialize the hash table. + const ( + tableBits = 14 + maxTableSize = 1 << tableBits + ) + + var table [maxTableSize]uint32 + + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + + // Bail if we can't compress to at least this. + dstLimit := len(src) - len(src)>>5 - 5 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We search for a repeat at -1, but don't output repeats when nextEmit == 0 + repeat := 1 + + for { + candidate := 0 + for { + // Next src position to check + nextS := s + (s-nextEmit)>>6 + 4 + if nextS > sLimit { + goto emitRemainder + } + hash0 := hash6(cv, tableBits) + hash1 := hash6(cv>>8, tableBits) + candidate = int(table[hash0]) + candidate2 := int(table[hash1]) + table[hash0] = uint32(s) + table[hash1] = uint32(s + 1) + hash2 := hash6(cv>>16, tableBits) + + // Check repeat at offset checkRep. + const checkRep = 1 + if uint32(cv>>(checkRep*8)) == load32(src, s-repeat+checkRep) { + base := s + checkRep + // Extend back + for i := base - repeat; base > nextEmit && i > 0 && src[i-1] == src[base-1]; { + i-- + base-- + } + d += emitLiteral(dst[d:], src[nextEmit:base]) + + // Extend forward + candidate := s - repeat + 4 + checkRep + s += 4 + checkRep + for s <= sLimit { + if diff := load64(src, s) ^ load64(src, candidate); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidate += 8 + } + + d += emitCopyNoRepeat(dst[d:], repeat, s-base) + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + cv = load64(src, s) + continue + } + + if uint32(cv) == load32(src, candidate) { + break + } + candidate = int(table[hash2]) + if uint32(cv>>8) == load32(src, candidate2) { + table[hash2] = uint32(s + 2) + candidate = candidate2 + s++ + break + } + table[hash2] = uint32(s + 2) + if uint32(cv>>16) == load32(src, candidate) { + s += 2 + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards + for candidate > 0 && s > nextEmit && src[candidate-1] == src[s-1] { + candidate-- + s-- + } + + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + // A 4-byte match has been found. We'll later see if more than 4 bytes + // match. But, prior to the match, src[nextEmit:s] are unmatched. Emit + // them as literal bytes. + + d += emitLiteral(dst[d:], src[nextEmit:s]) + + // Call emitCopy, and then see if another emitCopy could be our next + // move. Repeat until we find no match for the input immediately after + // what was consumed by the last emitCopy call. + // + // If we exit this loop normally then we need to call emitLiteral next, + // though we don't yet know how big the literal will be. We handle that + // by proceeding to the next iteration of the main loop. We also can + // exit this loop via goto if we get close to exhausting the input. + for { + // Invariant: we have a 4-byte match at s, and no need to emit any + // literal bytes prior to s. + base := s + repeat = base - candidate + + // Extend the 4-byte match as long as possible. + s += 4 + candidate += 4 + for s <= len(src)-8 { + if diff := load64(src, s) ^ load64(src, candidate); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidate += 8 + } + + d += emitCopyNoRepeat(dst[d:], repeat, s-base) + if false { + // Validate match. + a := src[base:s] + b := src[base-repeat : base-repeat+(s-base)] + if !bytes.Equal(a, b) { + panic("mismatch") + } + } + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Check for an immediate match, otherwise start search at s+1 + x := load64(src, s-2) + m2Hash := hash6(x, tableBits) + currHash := hash6(x>>16, tableBits) + candidate = int(table[currHash]) + table[m2Hash] = uint32(s - 2) + table[currHash] = uint32(s) + if uint32(x>>16) != load32(src, candidate) { + cv = load64(src, s+1) + s++ + break + } + } + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} diff --git a/vendor/github.com/klauspost/compress/s2/encode_amd64.go b/vendor/github.com/klauspost/compress/s2/encode_amd64.go new file mode 100644 index 000000000..e612225f4 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode_amd64.go @@ -0,0 +1,142 @@ +//go:build !appengine && !noasm && gc +// +build !appengine,!noasm,gc + +package s2 + +// encodeBlock encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlock(dst, src []byte) (d int) { + const ( + // Use 12 bit table when less than... + limit12B = 16 << 10 + // Use 10 bit table when less than... + limit10B = 4 << 10 + // Use 8 bit table when less than... + limit8B = 512 + ) + + if len(src) >= 4<<20 { + return encodeBlockAsm(dst, src) + } + if len(src) >= limit12B { + return encodeBlockAsm4MB(dst, src) + } + if len(src) >= limit10B { + return encodeBlockAsm12B(dst, src) + } + if len(src) >= limit8B { + return encodeBlockAsm10B(dst, src) + } + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeBlockAsm8B(dst, src) +} + +// encodeBlockBetter encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBetter(dst, src []byte) (d int) { + const ( + // Use 12 bit table when less than... + limit12B = 16 << 10 + // Use 10 bit table when less than... + limit10B = 4 << 10 + // Use 8 bit table when less than... + limit8B = 512 + ) + + if len(src) > 4<<20 { + return encodeBetterBlockAsm(dst, src) + } + if len(src) >= limit12B { + return encodeBetterBlockAsm4MB(dst, src) + } + if len(src) >= limit10B { + return encodeBetterBlockAsm12B(dst, src) + } + if len(src) >= limit8B { + return encodeBetterBlockAsm10B(dst, src) + } + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeBetterBlockAsm8B(dst, src) +} + +// encodeBlockSnappy encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockSnappy(dst, src []byte) (d int) { + const ( + // Use 12 bit table when less than... + limit12B = 16 << 10 + // Use 10 bit table when less than... + limit10B = 4 << 10 + // Use 8 bit table when less than... + limit8B = 512 + ) + if len(src) >= 64<<10 { + return encodeSnappyBlockAsm(dst, src) + } + if len(src) >= limit12B { + return encodeSnappyBlockAsm64K(dst, src) + } + if len(src) >= limit10B { + return encodeSnappyBlockAsm12B(dst, src) + } + if len(src) >= limit8B { + return encodeSnappyBlockAsm10B(dst, src) + } + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeSnappyBlockAsm8B(dst, src) +} + +// encodeBlockSnappy encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBetterSnappy(dst, src []byte) (d int) { + const ( + // Use 12 bit table when less than... + limit12B = 16 << 10 + // Use 10 bit table when less than... + limit10B = 4 << 10 + // Use 8 bit table when less than... + limit8B = 512 + ) + if len(src) >= 64<<10 { + return encodeSnappyBetterBlockAsm(dst, src) + } + if len(src) >= limit12B { + return encodeSnappyBetterBlockAsm64K(dst, src) + } + if len(src) >= limit10B { + return encodeSnappyBetterBlockAsm12B(dst, src) + } + if len(src) >= limit8B { + return encodeSnappyBetterBlockAsm10B(dst, src) + } + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeSnappyBetterBlockAsm8B(dst, src) +} diff --git a/vendor/github.com/klauspost/compress/s2/encode_best.go b/vendor/github.com/klauspost/compress/s2/encode_best.go new file mode 100644 index 000000000..4bc80bc6a --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode_best.go @@ -0,0 +1,630 @@ +// Copyright 2016 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "fmt" + "math/bits" +) + +// encodeBlockBest encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBest(dst, src []byte) (d int) { + // Initialize the hash tables. + const ( + // Long hash matches. + lTableBits = 19 + maxLTableSize = 1 << lTableBits + + // Short hash matches. + sTableBits = 16 + maxSTableSize = 1 << sTableBits + + inputMargin = 8 + 2 + ) + + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + if len(src) < minNonLiteralBlockSize { + return 0 + } + + var lTable [maxLTableSize]uint64 + var sTable [maxSTableSize]uint64 + + // Bail if we can't compress to at least this. + dstLimit := len(src) - 5 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We search for a repeat at -1, but don't output repeats when nextEmit == 0 + repeat := 1 + const lowbitMask = 0xffffffff + getCur := func(x uint64) int { + return int(x & lowbitMask) + } + getPrev := func(x uint64) int { + return int(x >> 32) + } + const maxSkip = 64 + + for { + type match struct { + offset int + s int + length int + score int + rep bool + } + var best match + for { + // Next src position to check + nextS := (s-nextEmit)>>8 + 1 + if nextS > maxSkip { + nextS = s + maxSkip + } else { + nextS += s + } + if nextS > sLimit { + goto emitRemainder + } + hashL := hash8(cv, lTableBits) + hashS := hash4(cv, sTableBits) + candidateL := lTable[hashL] + candidateS := sTable[hashS] + + score := func(m match) int { + // Matches that are longer forward are penalized since we must emit it as a literal. + score := m.length - m.s + if nextEmit == m.s { + // If we do not have to emit literals, we save 1 byte + score++ + } + offset := m.s - m.offset + if m.rep { + return score - emitRepeatSize(offset, m.length) + } + return score - emitCopySize(offset, m.length) + } + + matchAt := func(offset, s int, first uint32, rep bool) match { + if best.length != 0 && best.s-best.offset == s-offset { + // Don't retest if we have the same offset. + return match{offset: offset, s: s} + } + if load32(src, offset) != first { + return match{offset: offset, s: s} + } + m := match{offset: offset, s: s, length: 4 + offset, rep: rep} + s += 4 + for s <= sLimit { + if diff := load64(src, s) ^ load64(src, m.length); diff != 0 { + m.length += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + m.length += 8 + } + m.length -= offset + m.score = score(m) + if m.score <= -m.s { + // Eliminate if no savings, we might find a better one. + m.length = 0 + } + return m + } + + bestOf := func(a, b match) match { + if b.length == 0 { + return a + } + if a.length == 0 { + return b + } + as := a.score + b.s + bs := b.score + a.s + if as >= bs { + return a + } + return b + } + + best = bestOf(matchAt(getCur(candidateL), s, uint32(cv), false), matchAt(getPrev(candidateL), s, uint32(cv), false)) + best = bestOf(best, matchAt(getCur(candidateS), s, uint32(cv), false)) + best = bestOf(best, matchAt(getPrev(candidateS), s, uint32(cv), false)) + + { + best = bestOf(best, matchAt(s-repeat+1, s+1, uint32(cv>>8), true)) + if best.length > 0 { + // s+1 + nextShort := sTable[hash4(cv>>8, sTableBits)] + s := s + 1 + cv := load64(src, s) + nextLong := lTable[hash8(cv, lTableBits)] + best = bestOf(best, matchAt(getCur(nextShort), s, uint32(cv), false)) + best = bestOf(best, matchAt(getPrev(nextShort), s, uint32(cv), false)) + best = bestOf(best, matchAt(getCur(nextLong), s, uint32(cv), false)) + best = bestOf(best, matchAt(getPrev(nextLong), s, uint32(cv), false)) + // Repeat at + 2 + best = bestOf(best, matchAt(s-repeat+1, s+1, uint32(cv>>8), true)) + + // s+2 + if true { + nextShort = sTable[hash4(cv>>8, sTableBits)] + s++ + cv = load64(src, s) + nextLong = lTable[hash8(cv, lTableBits)] + best = bestOf(best, matchAt(getCur(nextShort), s, uint32(cv), false)) + best = bestOf(best, matchAt(getPrev(nextShort), s, uint32(cv), false)) + best = bestOf(best, matchAt(getCur(nextLong), s, uint32(cv), false)) + best = bestOf(best, matchAt(getPrev(nextLong), s, uint32(cv), false)) + } + // Search for a match at best match end, see if that is better. + if sAt := best.s + best.length; sAt < sLimit { + sBack := best.s + backL := best.length + // Load initial values + cv = load64(src, sBack) + // Search for mismatch + next := lTable[hash8(load64(src, sAt), lTableBits)] + //next := sTable[hash4(load64(src, sAt), sTableBits)] + + if checkAt := getCur(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) + } + if checkAt := getPrev(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) + } + } + } + } + + // Update table + lTable[hashL] = uint64(s) | candidateL<<32 + sTable[hashS] = uint64(s) | candidateS<<32 + + if best.length > 0 { + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards, not needed for repeats... + s = best.s + if !best.rep { + for best.offset > 0 && s > nextEmit && src[best.offset-1] == src[s-1] { + best.offset-- + best.length++ + s-- + } + } + if false && best.offset >= s { + panic(fmt.Errorf("t %d >= s %d", best.offset, s)) + } + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + base := s + offset := s - best.offset + + s += best.length + + if offset > 65535 && s-base <= 5 && !best.rep { + // Bail if the match is equal or worse to the encoding. + s = best.s + 1 + if s >= sLimit { + goto emitRemainder + } + cv = load64(src, s) + continue + } + d += emitLiteral(dst[d:], src[nextEmit:base]) + if best.rep { + if nextEmit > 0 { + // same as `add := emitCopy(dst[d:], repeat, s-base)` but skips storing offset. + d += emitRepeat(dst[d:], offset, best.length) + } else { + // First match, cannot be repeat. + d += emitCopy(dst[d:], offset, best.length) + } + } else { + d += emitCopy(dst[d:], offset, best.length) + } + repeat = offset + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Fill tables... + for i := best.s + 1; i < s; i++ { + cv0 := load64(src, i) + long0 := hash8(cv0, lTableBits) + short0 := hash4(cv0, sTableBits) + lTable[long0] = uint64(i) | lTable[long0]<<32 + sTable[short0] = uint64(i) | sTable[short0]<<32 + } + cv = load64(src, s) + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} + +// encodeBlockBestSnappy encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBestSnappy(dst, src []byte) (d int) { + // Initialize the hash tables. + const ( + // Long hash matches. + lTableBits = 19 + maxLTableSize = 1 << lTableBits + + // Short hash matches. + sTableBits = 16 + maxSTableSize = 1 << sTableBits + + inputMargin = 8 + 2 + ) + + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + if len(src) < minNonLiteralBlockSize { + return 0 + } + + var lTable [maxLTableSize]uint64 + var sTable [maxSTableSize]uint64 + + // Bail if we can't compress to at least this. + dstLimit := len(src) - 5 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We search for a repeat at -1, but don't output repeats when nextEmit == 0 + repeat := 1 + const lowbitMask = 0xffffffff + getCur := func(x uint64) int { + return int(x & lowbitMask) + } + getPrev := func(x uint64) int { + return int(x >> 32) + } + const maxSkip = 64 + + for { + type match struct { + offset int + s int + length int + score int + } + var best match + for { + // Next src position to check + nextS := (s-nextEmit)>>8 + 1 + if nextS > maxSkip { + nextS = s + maxSkip + } else { + nextS += s + } + if nextS > sLimit { + goto emitRemainder + } + hashL := hash8(cv, lTableBits) + hashS := hash4(cv, sTableBits) + candidateL := lTable[hashL] + candidateS := sTable[hashS] + + score := func(m match) int { + // Matches that are longer forward are penalized since we must emit it as a literal. + score := m.length - m.s + if nextEmit == m.s { + // If we do not have to emit literals, we save 1 byte + score++ + } + offset := m.s - m.offset + + return score - emitCopyNoRepeatSize(offset, m.length) + } + + matchAt := func(offset, s int, first uint32) match { + if best.length != 0 && best.s-best.offset == s-offset { + // Don't retest if we have the same offset. + return match{offset: offset, s: s} + } + if load32(src, offset) != first { + return match{offset: offset, s: s} + } + m := match{offset: offset, s: s, length: 4 + offset} + s += 4 + for s <= sLimit { + if diff := load64(src, s) ^ load64(src, m.length); diff != 0 { + m.length += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + m.length += 8 + } + m.length -= offset + m.score = score(m) + if m.score <= -m.s { + // Eliminate if no savings, we might find a better one. + m.length = 0 + } + return m + } + + bestOf := func(a, b match) match { + if b.length == 0 { + return a + } + if a.length == 0 { + return b + } + as := a.score + b.s + bs := b.score + a.s + if as >= bs { + return a + } + return b + } + + best = bestOf(matchAt(getCur(candidateL), s, uint32(cv)), matchAt(getPrev(candidateL), s, uint32(cv))) + best = bestOf(best, matchAt(getCur(candidateS), s, uint32(cv))) + best = bestOf(best, matchAt(getPrev(candidateS), s, uint32(cv))) + + { + best = bestOf(best, matchAt(s-repeat+1, s+1, uint32(cv>>8))) + if best.length > 0 { + // s+1 + nextShort := sTable[hash4(cv>>8, sTableBits)] + s := s + 1 + cv := load64(src, s) + nextLong := lTable[hash8(cv, lTableBits)] + best = bestOf(best, matchAt(getCur(nextShort), s, uint32(cv))) + best = bestOf(best, matchAt(getPrev(nextShort), s, uint32(cv))) + best = bestOf(best, matchAt(getCur(nextLong), s, uint32(cv))) + best = bestOf(best, matchAt(getPrev(nextLong), s, uint32(cv))) + // Repeat at + 2 + best = bestOf(best, matchAt(s-repeat+1, s+1, uint32(cv>>8))) + + // s+2 + if true { + nextShort = sTable[hash4(cv>>8, sTableBits)] + s++ + cv = load64(src, s) + nextLong = lTable[hash8(cv, lTableBits)] + best = bestOf(best, matchAt(getCur(nextShort), s, uint32(cv))) + best = bestOf(best, matchAt(getPrev(nextShort), s, uint32(cv))) + best = bestOf(best, matchAt(getCur(nextLong), s, uint32(cv))) + best = bestOf(best, matchAt(getPrev(nextLong), s, uint32(cv))) + } + // Search for a match at best match end, see if that is better. + if sAt := best.s + best.length; sAt < sLimit { + sBack := best.s + backL := best.length + // Load initial values + cv = load64(src, sBack) + // Search for mismatch + next := lTable[hash8(load64(src, sAt), lTableBits)] + //next := sTable[hash4(load64(src, sAt), sTableBits)] + + if checkAt := getCur(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv))) + } + if checkAt := getPrev(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv))) + } + } + } + } + + // Update table + lTable[hashL] = uint64(s) | candidateL<<32 + sTable[hashS] = uint64(s) | candidateS<<32 + + if best.length > 0 { + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards, not needed for repeats... + s = best.s + if true { + for best.offset > 0 && s > nextEmit && src[best.offset-1] == src[s-1] { + best.offset-- + best.length++ + s-- + } + } + if false && best.offset >= s { + panic(fmt.Errorf("t %d >= s %d", best.offset, s)) + } + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + base := s + offset := s - best.offset + + s += best.length + + if offset > 65535 && s-base <= 5 { + // Bail if the match is equal or worse to the encoding. + s = best.s + 1 + if s >= sLimit { + goto emitRemainder + } + cv = load64(src, s) + continue + } + d += emitLiteral(dst[d:], src[nextEmit:base]) + d += emitCopyNoRepeat(dst[d:], offset, best.length) + repeat = offset + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Fill tables... + for i := best.s + 1; i < s; i++ { + cv0 := load64(src, i) + long0 := hash8(cv0, lTableBits) + short0 := hash4(cv0, sTableBits) + lTable[long0] = uint64(i) | lTable[long0]<<32 + sTable[short0] = uint64(i) | sTable[short0]<<32 + } + cv = load64(src, s) + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} + +// emitCopySize returns the size to encode the offset+length +// +// It assumes that: +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +func emitCopySize(offset, length int) int { + if offset >= 65536 { + i := 0 + if length > 64 { + length -= 64 + if length >= 4 { + // Emit remaining as repeats + return 5 + emitRepeatSize(offset, length) + } + i = 5 + } + if length == 0 { + return i + } + return i + 5 + } + + // Offset no more than 2 bytes. + if length > 64 { + if offset < 2048 { + // Emit 8 bytes, then rest as repeats... + return 2 + emitRepeatSize(offset, length-8) + } + // Emit remaining as repeats, at least 4 bytes remain. + return 3 + emitRepeatSize(offset, length-60) + } + if length >= 12 || offset >= 2048 { + return 3 + } + // Emit the remaining copy, encoded as 2 bytes. + return 2 +} + +// emitCopyNoRepeatSize returns the size to encode the offset+length +// +// It assumes that: +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +func emitCopyNoRepeatSize(offset, length int) int { + if offset >= 65536 { + return 5 + 5*(length/64) + } + + // Offset no more than 2 bytes. + if length > 64 { + // Emit remaining as repeats, at least 4 bytes remain. + return 3 + 3*(length/60) + } + if length >= 12 || offset >= 2048 { + return 3 + } + // Emit the remaining copy, encoded as 2 bytes. + return 2 +} + +// emitRepeatSize returns the number of bytes required to encode a repeat. +// Length must be at least 4 and < 1<<24 +func emitRepeatSize(offset, length int) int { + // Repeat offset, make length cheaper + if length <= 4+4 || (length < 8+4 && offset < 2048) { + return 2 + } + if length < (1<<8)+4+4 { + return 3 + } + if length < (1<<16)+(1<<8)+4 { + return 4 + } + const maxRepeat = (1 << 24) - 1 + length -= (1 << 16) - 4 + left := 0 + if length > maxRepeat { + left = length - maxRepeat + 4 + length = maxRepeat - 4 + } + if left > 0 { + return 5 + emitRepeatSize(offset, left) + } + return 5 +} diff --git a/vendor/github.com/klauspost/compress/s2/encode_better.go b/vendor/github.com/klauspost/compress/s2/encode_better.go new file mode 100644 index 000000000..943215b8a --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode_better.go @@ -0,0 +1,431 @@ +// Copyright 2016 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "math/bits" +) + +// hash4 returns the hash of the lowest 4 bytes of u to fit in a hash table with h bits. +// Preferably h should be a constant and should always be <32. +func hash4(u uint64, h uint8) uint32 { + const prime4bytes = 2654435761 + return (uint32(u) * prime4bytes) >> ((32 - h) & 31) +} + +// hash5 returns the hash of the lowest 5 bytes of u to fit in a hash table with h bits. +// Preferably h should be a constant and should always be <64. +func hash5(u uint64, h uint8) uint32 { + const prime5bytes = 889523592379 + return uint32(((u << (64 - 40)) * prime5bytes) >> ((64 - h) & 63)) +} + +// hash7 returns the hash of the lowest 7 bytes of u to fit in a hash table with h bits. +// Preferably h should be a constant and should always be <64. +func hash7(u uint64, h uint8) uint32 { + const prime7bytes = 58295818150454627 + return uint32(((u << (64 - 56)) * prime7bytes) >> ((64 - h) & 63)) +} + +// hash8 returns the hash of u to fit in a hash table with h bits. +// Preferably h should be a constant and should always be <64. +func hash8(u uint64, h uint8) uint32 { + const prime8bytes = 0xcf1bbcdcb7a56463 + return uint32((u * prime8bytes) >> ((64 - h) & 63)) +} + +// encodeBlockBetter encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBetterGo(dst, src []byte) (d int) { + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + if len(src) < minNonLiteralBlockSize { + return 0 + } + + // Initialize the hash tables. + const ( + // Long hash matches. + lTableBits = 16 + maxLTableSize = 1 << lTableBits + + // Short hash matches. + sTableBits = 14 + maxSTableSize = 1 << sTableBits + ) + + var lTable [maxLTableSize]uint32 + var sTable [maxSTableSize]uint32 + + // Bail if we can't compress to at least this. + dstLimit := len(src) - len(src)>>5 - 6 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We initialize repeat to 0, so we never match on first attempt + repeat := 0 + + for { + candidateL := 0 + nextS := 0 + for { + // Next src position to check + nextS = s + (s-nextEmit)>>7 + 1 + if nextS > sLimit { + goto emitRemainder + } + hashL := hash7(cv, lTableBits) + hashS := hash4(cv, sTableBits) + candidateL = int(lTable[hashL]) + candidateS := int(sTable[hashS]) + lTable[hashL] = uint32(s) + sTable[hashS] = uint32(s) + + // Check repeat at offset checkRep. + const checkRep = 1 + if false && uint32(cv>>(checkRep*8)) == load32(src, s-repeat+checkRep) { + base := s + checkRep + // Extend back + for i := base - repeat; base > nextEmit && i > 0 && src[i-1] == src[base-1]; { + i-- + base-- + } + d += emitLiteral(dst[d:], src[nextEmit:base]) + + // Extend forward + candidate := s - repeat + 4 + checkRep + s += 4 + checkRep + for s < len(src) { + if len(src)-s < 8 { + if src[s] == src[candidate] { + s++ + candidate++ + continue + } + break + } + if diff := load64(src, s) ^ load64(src, candidate); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidate += 8 + } + if nextEmit > 0 { + // same as `add := emitCopy(dst[d:], repeat, s-base)` but skips storing offset. + d += emitRepeat(dst[d:], repeat, s-base) + } else { + // First match, cannot be repeat. + d += emitCopy(dst[d:], repeat, s-base) + } + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + cv = load64(src, s) + continue + } + + if uint32(cv) == load32(src, candidateL) { + break + } + + // Check our short candidate + if uint32(cv) == load32(src, candidateS) { + // Try a long candidate at s+1 + hashL = hash7(cv>>8, lTableBits) + candidateL = int(lTable[hashL]) + lTable[hashL] = uint32(s + 1) + if uint32(cv>>8) == load32(src, candidateL) { + s++ + break + } + // Use our short candidate. + candidateL = candidateS + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards + for candidateL > 0 && s > nextEmit && src[candidateL-1] == src[s-1] { + candidateL-- + s-- + } + + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + base := s + offset := base - candidateL + + // Extend the 4-byte match as long as possible. + s += 4 + candidateL += 4 + for s < len(src) { + if len(src)-s < 8 { + if src[s] == src[candidateL] { + s++ + candidateL++ + continue + } + break + } + if diff := load64(src, s) ^ load64(src, candidateL); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidateL += 8 + } + + if offset > 65535 && s-base <= 5 && repeat != offset { + // Bail if the match is equal or worse to the encoding. + s = nextS + 1 + if s >= sLimit { + goto emitRemainder + } + cv = load64(src, s) + continue + } + + d += emitLiteral(dst[d:], src[nextEmit:base]) + if repeat == offset { + d += emitRepeat(dst[d:], offset, s-base) + } else { + d += emitCopy(dst[d:], offset, s-base) + repeat = offset + } + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Index match start+1 (long) and start+2 (short) + index0 := base + 1 + // Index match end-2 (long) and end-1 (short) + index1 := s - 2 + + cv0 := load64(src, index0) + cv1 := load64(src, index1) + cv = load64(src, s) + lTable[hash7(cv0, lTableBits)] = uint32(index0) + lTable[hash7(cv0>>8, lTableBits)] = uint32(index0 + 1) + lTable[hash7(cv1, lTableBits)] = uint32(index1) + lTable[hash7(cv1>>8, lTableBits)] = uint32(index1 + 1) + sTable[hash4(cv0>>8, sTableBits)] = uint32(index0 + 1) + sTable[hash4(cv0>>16, sTableBits)] = uint32(index0 + 2) + sTable[hash4(cv1>>8, sTableBits)] = uint32(index1 + 1) + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} + +// encodeBlockBetterSnappyGo encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) && +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +func encodeBlockBetterSnappyGo(dst, src []byte) (d int) { + // sLimit is when to stop looking for offset/length copies. The inputMargin + // lets us use a fast path for emitLiteral in the main loop, while we are + // looking for copies. + sLimit := len(src) - inputMargin + if len(src) < minNonLiteralBlockSize { + return 0 + } + + // Initialize the hash tables. + const ( + // Long hash matches. + lTableBits = 16 + maxLTableSize = 1 << lTableBits + + // Short hash matches. + sTableBits = 14 + maxSTableSize = 1 << sTableBits + ) + + var lTable [maxLTableSize]uint32 + var sTable [maxSTableSize]uint32 + + // Bail if we can't compress to at least this. + dstLimit := len(src) - len(src)>>5 - 6 + + // nextEmit is where in src the next emitLiteral should start from. + nextEmit := 0 + + // The encoded form must start with a literal, as there are no previous + // bytes to copy, so we start looking for hash matches at s == 1. + s := 1 + cv := load64(src, s) + + // We initialize repeat to 0, so we never match on first attempt + repeat := 0 + const maxSkip = 100 + + for { + candidateL := 0 + nextS := 0 + for { + // Next src position to check + nextS = (s-nextEmit)>>7 + 1 + if nextS > maxSkip { + nextS = s + maxSkip + } else { + nextS += s + } + + if nextS > sLimit { + goto emitRemainder + } + hashL := hash7(cv, lTableBits) + hashS := hash4(cv, sTableBits) + candidateL = int(lTable[hashL]) + candidateS := int(sTable[hashS]) + lTable[hashL] = uint32(s) + sTable[hashS] = uint32(s) + + if uint32(cv) == load32(src, candidateL) { + break + } + + // Check our short candidate + if uint32(cv) == load32(src, candidateS) { + // Try a long candidate at s+1 + hashL = hash7(cv>>8, lTableBits) + candidateL = int(lTable[hashL]) + lTable[hashL] = uint32(s + 1) + if uint32(cv>>8) == load32(src, candidateL) { + s++ + break + } + // Use our short candidate. + candidateL = candidateS + break + } + + cv = load64(src, nextS) + s = nextS + } + + // Extend backwards + for candidateL > 0 && s > nextEmit && src[candidateL-1] == src[s-1] { + candidateL-- + s-- + } + + // Bail if we exceed the maximum size. + if d+(s-nextEmit) > dstLimit { + return 0 + } + + base := s + offset := base - candidateL + + // Extend the 4-byte match as long as possible. + s += 4 + candidateL += 4 + for s < len(src) { + if len(src)-s < 8 { + if src[s] == src[candidateL] { + s++ + candidateL++ + continue + } + break + } + if diff := load64(src, s) ^ load64(src, candidateL); diff != 0 { + s += bits.TrailingZeros64(diff) >> 3 + break + } + s += 8 + candidateL += 8 + } + + if offset > 65535 && s-base <= 5 && repeat != offset { + // Bail if the match is equal or worse to the encoding. + s = nextS + 1 + if s >= sLimit { + goto emitRemainder + } + cv = load64(src, s) + continue + } + + d += emitLiteral(dst[d:], src[nextEmit:base]) + d += emitCopyNoRepeat(dst[d:], offset, s-base) + repeat = offset + + nextEmit = s + if s >= sLimit { + goto emitRemainder + } + + if d > dstLimit { + // Do we have space for more, if not bail. + return 0 + } + // Index match start+1 (long) and start+2 (short) + index0 := base + 1 + // Index match end-2 (long) and end-1 (short) + index1 := s - 2 + + cv0 := load64(src, index0) + cv1 := load64(src, index1) + cv = load64(src, s) + lTable[hash7(cv0, lTableBits)] = uint32(index0) + lTable[hash7(cv0>>8, lTableBits)] = uint32(index0 + 1) + lTable[hash7(cv1, lTableBits)] = uint32(index1) + lTable[hash7(cv1>>8, lTableBits)] = uint32(index1 + 1) + sTable[hash4(cv0>>8, sTableBits)] = uint32(index0 + 1) + sTable[hash4(cv0>>16, sTableBits)] = uint32(index0 + 2) + sTable[hash4(cv1>>8, sTableBits)] = uint32(index1 + 1) + } + +emitRemainder: + if nextEmit < len(src) { + // Bail if we exceed the maximum size. + if d+len(src)-nextEmit > dstLimit { + return 0 + } + d += emitLiteral(dst[d:], src[nextEmit:]) + } + return d +} diff --git a/vendor/github.com/klauspost/compress/s2/encode_go.go b/vendor/github.com/klauspost/compress/s2/encode_go.go new file mode 100644 index 000000000..94784b82a --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encode_go.go @@ -0,0 +1,307 @@ +//go:build !amd64 || appengine || !gc || noasm +// +build !amd64 appengine !gc noasm + +package s2 + +import ( + "math/bits" +) + +// encodeBlock encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) +func encodeBlock(dst, src []byte) (d int) { + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeBlockGo(dst, src) +} + +// encodeBlockBetter encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) +func encodeBlockBetter(dst, src []byte) (d int) { + return encodeBlockBetterGo(dst, src) +} + +// encodeBlockBetter encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) +func encodeBlockBetterSnappy(dst, src []byte) (d int) { + return encodeBlockBetterSnappyGo(dst, src) +} + +// encodeBlock encodes a non-empty src to a guaranteed-large-enough dst. It +// assumes that the varint-encoded length of the decompressed bytes has already +// been written. +// +// It also assumes that: +// len(dst) >= MaxEncodedLen(len(src)) +func encodeBlockSnappy(dst, src []byte) (d int) { + if len(src) < minNonLiteralBlockSize { + return 0 + } + return encodeBlockSnappyGo(dst, src) +} + +// emitLiteral writes a literal chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes +// 0 <= len(lit) && len(lit) <= math.MaxUint32 +func emitLiteral(dst, lit []byte) int { + if len(lit) == 0 { + return 0 + } + const num = 63<<2 | tagLiteral + i, n := 0, uint(len(lit)-1) + switch { + case n < 60: + dst[0] = uint8(n)<<2 | tagLiteral + i = 1 + case n < 1<<8: + dst[1] = uint8(n) + dst[0] = 60<<2 | tagLiteral + i = 2 + case n < 1<<16: + dst[2] = uint8(n >> 8) + dst[1] = uint8(n) + dst[0] = 61<<2 | tagLiteral + i = 3 + case n < 1<<24: + dst[3] = uint8(n >> 16) + dst[2] = uint8(n >> 8) + dst[1] = uint8(n) + dst[0] = 62<<2 | tagLiteral + i = 4 + default: + dst[4] = uint8(n >> 24) + dst[3] = uint8(n >> 16) + dst[2] = uint8(n >> 8) + dst[1] = uint8(n) + dst[0] = 63<<2 | tagLiteral + i = 5 + } + return i + copy(dst[i:], lit) +} + +// emitRepeat writes a repeat chunk and returns the number of bytes written. +// Length must be at least 4 and < 1<<24 +func emitRepeat(dst []byte, offset, length int) int { + // Repeat offset, make length cheaper + length -= 4 + if length <= 4 { + dst[0] = uint8(length)<<2 | tagCopy1 + dst[1] = 0 + return 2 + } + if length < 8 && offset < 2048 { + // Encode WITH offset + dst[1] = uint8(offset) + dst[0] = uint8(offset>>8)<<5 | uint8(length)<<2 | tagCopy1 + return 2 + } + if length < (1<<8)+4 { + length -= 4 + dst[2] = uint8(length) + dst[1] = 0 + dst[0] = 5<<2 | tagCopy1 + return 3 + } + if length < (1<<16)+(1<<8) { + length -= 1 << 8 + dst[3] = uint8(length >> 8) + dst[2] = uint8(length >> 0) + dst[1] = 0 + dst[0] = 6<<2 | tagCopy1 + return 4 + } + const maxRepeat = (1 << 24) - 1 + length -= 1 << 16 + left := 0 + if length > maxRepeat { + left = length - maxRepeat + 4 + length = maxRepeat - 4 + } + dst[4] = uint8(length >> 16) + dst[3] = uint8(length >> 8) + dst[2] = uint8(length >> 0) + dst[1] = 0 + dst[0] = 7<<2 | tagCopy1 + if left > 0 { + return 5 + emitRepeat(dst[5:], offset, left) + } + return 5 +} + +// emitCopy writes a copy chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +func emitCopy(dst []byte, offset, length int) int { + if offset >= 65536 { + i := 0 + if length > 64 { + // Emit a length 64 copy, encoded as 5 bytes. + dst[4] = uint8(offset >> 24) + dst[3] = uint8(offset >> 16) + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = 63<<2 | tagCopy4 + length -= 64 + if length >= 4 { + // Emit remaining as repeats + return 5 + emitRepeat(dst[5:], offset, length) + } + i = 5 + } + if length == 0 { + return i + } + // Emit a copy, offset encoded as 4 bytes. + dst[i+0] = uint8(length-1)<<2 | tagCopy4 + dst[i+1] = uint8(offset) + dst[i+2] = uint8(offset >> 8) + dst[i+3] = uint8(offset >> 16) + dst[i+4] = uint8(offset >> 24) + return i + 5 + } + + // Offset no more than 2 bytes. + if length > 64 { + off := 3 + if offset < 2048 { + // emit 8 bytes as tagCopy1, rest as repeats. + dst[1] = uint8(offset) + dst[0] = uint8(offset>>8)<<5 | uint8(8-4)<<2 | tagCopy1 + length -= 8 + off = 2 + } else { + // Emit a length 60 copy, encoded as 3 bytes. + // Emit remaining as repeat value (minimum 4 bytes). + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = 59<<2 | tagCopy2 + length -= 60 + } + // Emit remaining as repeats, at least 4 bytes remain. + return off + emitRepeat(dst[off:], offset, length) + } + if length >= 12 || offset >= 2048 { + // Emit the remaining copy, encoded as 3 bytes. + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = uint8(length-1)<<2 | tagCopy2 + return 3 + } + // Emit the remaining copy, encoded as 2 bytes. + dst[1] = uint8(offset) + dst[0] = uint8(offset>>8)<<5 | uint8(length-4)<<2 | tagCopy1 + return 2 +} + +// emitCopyNoRepeat writes a copy chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +func emitCopyNoRepeat(dst []byte, offset, length int) int { + if offset >= 65536 { + i := 0 + if length > 64 { + // Emit a length 64 copy, encoded as 5 bytes. + dst[4] = uint8(offset >> 24) + dst[3] = uint8(offset >> 16) + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = 63<<2 | tagCopy4 + length -= 64 + if length >= 4 { + // Emit remaining as repeats + return 5 + emitCopyNoRepeat(dst[5:], offset, length) + } + i = 5 + } + if length == 0 { + return i + } + // Emit a copy, offset encoded as 4 bytes. + dst[i+0] = uint8(length-1)<<2 | tagCopy4 + dst[i+1] = uint8(offset) + dst[i+2] = uint8(offset >> 8) + dst[i+3] = uint8(offset >> 16) + dst[i+4] = uint8(offset >> 24) + return i + 5 + } + + // Offset no more than 2 bytes. + if length > 64 { + // Emit a length 60 copy, encoded as 3 bytes. + // Emit remaining as repeat value (minimum 4 bytes). + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = 59<<2 | tagCopy2 + length -= 60 + // Emit remaining as repeats, at least 4 bytes remain. + return 3 + emitCopyNoRepeat(dst[3:], offset, length) + } + if length >= 12 || offset >= 2048 { + // Emit the remaining copy, encoded as 3 bytes. + dst[2] = uint8(offset >> 8) + dst[1] = uint8(offset) + dst[0] = uint8(length-1)<<2 | tagCopy2 + return 3 + } + // Emit the remaining copy, encoded as 2 bytes. + dst[1] = uint8(offset) + dst[0] = uint8(offset>>8)<<5 | uint8(length-4)<<2 | tagCopy1 + return 2 +} + +// matchLen returns how many bytes match in a and b +// +// It assumes that: +// len(a) <= len(b) +// +func matchLen(a []byte, b []byte) int { + b = b[:len(a)] + var checked int + if len(a) > 4 { + // Try 4 bytes first + if diff := load32(a, 0) ^ load32(b, 0); diff != 0 { + return bits.TrailingZeros32(diff) >> 3 + } + // Switch to 8 byte matching. + checked = 4 + a = a[4:] + b = b[4:] + for len(a) >= 8 { + b = b[:len(a)] + if diff := load64(a, 0) ^ load64(b, 0); diff != 0 { + return checked + (bits.TrailingZeros64(diff) >> 3) + } + checked += 8 + a = a[8:] + b = b[8:] + } + } + b = b[:len(a)] + for i := range a { + if a[i] != b[i] { + return int(i) + checked + } + } + return len(a) + checked +} diff --git a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go new file mode 100644 index 000000000..88f27c099 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go @@ -0,0 +1,191 @@ +// Code generated by command: go run gen.go -out ../encodeblock_amd64.s -stubs ../encodeblock_amd64.go -pkg=s2. DO NOT EDIT. + +//go:build !appengine && !noasm && gc && !noasm +// +build !appengine,!noasm,gc,!noasm + +package s2 + +func _dummy_() + +// encodeBlockAsm encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4294967295 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBlockAsm(dst []byte, src []byte) int + +// encodeBlockAsm4MB encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4194304 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBlockAsm4MB(dst []byte, src []byte) int + +// encodeBlockAsm12B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 16383 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBlockAsm12B(dst []byte, src []byte) int + +// encodeBlockAsm10B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4095 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBlockAsm10B(dst []byte, src []byte) int + +// encodeBlockAsm8B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 511 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBlockAsm8B(dst []byte, src []byte) int + +// encodeBetterBlockAsm encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4294967295 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBetterBlockAsm(dst []byte, src []byte) int + +// encodeBetterBlockAsm4MB encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4194304 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBetterBlockAsm4MB(dst []byte, src []byte) int + +// encodeBetterBlockAsm12B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 16383 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBetterBlockAsm12B(dst []byte, src []byte) int + +// encodeBetterBlockAsm10B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4095 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBetterBlockAsm10B(dst []byte, src []byte) int + +// encodeBetterBlockAsm8B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 511 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeBetterBlockAsm8B(dst []byte, src []byte) int + +// encodeSnappyBlockAsm encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4294967295 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBlockAsm(dst []byte, src []byte) int + +// encodeSnappyBlockAsm64K encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 65535 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBlockAsm64K(dst []byte, src []byte) int + +// encodeSnappyBlockAsm12B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 16383 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBlockAsm12B(dst []byte, src []byte) int + +// encodeSnappyBlockAsm10B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4095 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBlockAsm10B(dst []byte, src []byte) int + +// encodeSnappyBlockAsm8B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 511 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBlockAsm8B(dst []byte, src []byte) int + +// encodeSnappyBetterBlockAsm encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4294967295 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBetterBlockAsm(dst []byte, src []byte) int + +// encodeSnappyBetterBlockAsm64K encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 65535 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBetterBlockAsm64K(dst []byte, src []byte) int + +// encodeSnappyBetterBlockAsm12B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 16383 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBetterBlockAsm12B(dst []byte, src []byte) int + +// encodeSnappyBetterBlockAsm10B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 4095 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBetterBlockAsm10B(dst []byte, src []byte) int + +// encodeSnappyBetterBlockAsm8B encodes a non-empty src to a guaranteed-large-enough dst. +// Maximum input 511 bytes. +// It assumes that the varint-encoded length of the decompressed bytes has already been written. +// +//go:noescape +func encodeSnappyBetterBlockAsm8B(dst []byte, src []byte) int + +// emitLiteral writes a literal chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes with margin of 0 bytes +// 0 <= len(lit) && len(lit) <= math.MaxUint32 +// +//go:noescape +func emitLiteral(dst []byte, lit []byte) int + +// emitRepeat writes a repeat chunk and returns the number of bytes written. +// Length must be at least 4 and < 1<<32 +// +//go:noescape +func emitRepeat(dst []byte, offset int, length int) int + +// emitCopy writes a copy chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +// +//go:noescape +func emitCopy(dst []byte, offset int, length int) int + +// emitCopyNoRepeat writes a copy chunk and returns the number of bytes written. +// +// It assumes that: +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 +// +//go:noescape +func emitCopyNoRepeat(dst []byte, offset int, length int) int + +// matchLen returns how many bytes match in a and b +// +// It assumes that: +// len(a) <= len(b) +// +//go:noescape +func matchLen(a []byte, b []byte) int diff --git a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s new file mode 100644 index 000000000..36915d949 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s @@ -0,0 +1,17779 @@ +// Code generated by command: go run gen.go -out ../encodeblock_amd64.s -stubs ../encodeblock_amd64.go -pkg=s2. DO NOT EDIT. + +//go:build !appengine && !noasm && gc && !noasm +// +build !appengine,!noasm,gc,!noasm + +#include "textflag.h" + +// func _dummy_() +TEXT ·_dummy_(SB), $0 +#ifdef GOAMD64_v4 +#ifndef GOAMD64_v3 +#define GOAMD64_v3 +#endif +#endif + RET + +// func encodeBlockAsm(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBlockAsm(SB), $65560-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000200, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBlockAsm: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBlockAsm + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBlockAsm: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBlockAsm + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R11 + IMULQ R9, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeBlockAsm + LEAL 1(CX), DI + MOVL 12(SP), R8 + MOVL DI, SI + SUBL 16(SP), SI + JZ repeat_extend_back_end_encodeBlockAsm + +repeat_extend_back_loop_encodeBlockAsm: + CMPL DI, R8 + JLE repeat_extend_back_end_encodeBlockAsm + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeBlockAsm + LEAL -1(DI), DI + DECL SI + JNZ repeat_extend_back_loop_encodeBlockAsm + +repeat_extend_back_end_encodeBlockAsm: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeBlockAsm + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeBlockAsm + CMPL SI, $0x00010000 + JLT three_bytes_repeat_emit_encodeBlockAsm + CMPL SI, $0x01000000 + JLT four_bytes_repeat_emit_encodeBlockAsm + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_repeat_emit_encodeBlockAsm + +four_bytes_repeat_emit_encodeBlockAsm: + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_repeat_emit_encodeBlockAsm + +three_bytes_repeat_emit_encodeBlockAsm: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeBlockAsm + +two_bytes_repeat_emit_encodeBlockAsm: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeBlockAsm + JMP memmove_long_repeat_emit_encodeBlockAsm + +one_byte_repeat_emit_encodeBlockAsm: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm + +emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm + +emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm + +emit_lit_memmove_repeat_emit_encodeBlockAsm_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_repeat_emit_encodeBlockAsm: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeBlockAsm + +memmove_long_repeat_emit_encodeBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R12 + SHRQ $0x05, R12 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R13 + SUBQ R11, R13 + DECQ R12 + JA emit_lit_memmove_long_repeat_emit_encodeBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(R10)(R13*1), R11 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_repeat_emit_encodeBlockAsmlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R11 + ADDQ $0x20, R13 + DECQ R12 + JNA emit_lit_memmove_long_repeat_emit_encodeBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(R10)(R13*1), X4 + MOVOU -16(R10)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R9, R13 + JAE emit_lit_memmove_long_repeat_emit_encodeBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeBlockAsm: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R9 + SUBL CX, R9 + LEAQ (DX)(CX*1), R10 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R12, R12 + CMPL R9, $0x08 + JL matchlen_match4_repeat_extend_encodeBlockAsm + +matchlen_loopback_repeat_extend_encodeBlockAsm: + MOVQ (R10)(R12*1), R11 + XORQ (SI)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_repeat_extend_encodeBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP repeat_extend_forward_end_encodeBlockAsm + +matchlen_loop_repeat_extend_encodeBlockAsm: + LEAL -8(R9), R9 + LEAL 8(R12), R12 + CMPL R9, $0x08 + JGE matchlen_loopback_repeat_extend_encodeBlockAsm + JZ repeat_extend_forward_end_encodeBlockAsm + +matchlen_match4_repeat_extend_encodeBlockAsm: + CMPL R9, $0x04 + JL matchlen_match2_repeat_extend_encodeBlockAsm + MOVL (R10)(R12*1), R11 + CMPL (SI)(R12*1), R11 + JNE matchlen_match2_repeat_extend_encodeBlockAsm + SUBL $0x04, R9 + LEAL 4(R12), R12 + +matchlen_match2_repeat_extend_encodeBlockAsm: + CMPL R9, $0x02 + JL matchlen_match1_repeat_extend_encodeBlockAsm + MOVW (R10)(R12*1), R11 + CMPW (SI)(R12*1), R11 + JNE matchlen_match1_repeat_extend_encodeBlockAsm + SUBL $0x02, R9 + LEAL 2(R12), R12 + +matchlen_match1_repeat_extend_encodeBlockAsm: + CMPL R9, $0x01 + JL repeat_extend_forward_end_encodeBlockAsm + MOVB (R10)(R12*1), R11 + CMPB (SI)(R12*1), R11 + JNE repeat_extend_forward_end_encodeBlockAsm + LEAL 1(R12), R12 + +repeat_extend_forward_end_encodeBlockAsm: + ADDL R12, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + TESTL R8, R8 + JZ repeat_as_copy_encodeBlockAsm + + // emitRepeat +emit_repeat_again_match_repeat_encodeBlockAsm: + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_match_repeat_encodeBlockAsm + CMPL R8, $0x0c + JGE cant_repeat_two_offset_match_repeat_encodeBlockAsm + CMPL DI, $0x00000800 + JLT repeat_two_offset_match_repeat_encodeBlockAsm + +cant_repeat_two_offset_match_repeat_encodeBlockAsm: + CMPL SI, $0x00000104 + JLT repeat_three_match_repeat_encodeBlockAsm + CMPL SI, $0x00010100 + JLT repeat_four_match_repeat_encodeBlockAsm + CMPL SI, $0x0100ffff + JLT repeat_five_match_repeat_encodeBlockAsm + LEAL -16842747(SI), SI + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_repeat_encodeBlockAsm + +repeat_five_match_repeat_encodeBlockAsm: + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_four_match_repeat_encodeBlockAsm: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_three_match_repeat_encodeBlockAsm: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_match_repeat_encodeBlockAsm: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_offset_match_repeat_encodeBlockAsm: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_as_copy_encodeBlockAsm: + // emitCopy + CMPL DI, $0x00010000 + JL two_byte_offset_repeat_as_copy_encodeBlockAsm + +four_bytes_loop_back_repeat_as_copy_encodeBlockAsm: + CMPL SI, $0x40 + JLE four_bytes_remain_repeat_as_copy_encodeBlockAsm + MOVB $0xff, (AX) + MOVL DI, 1(AX) + LEAL -64(SI), SI + ADDQ $0x05, AX + CMPL SI, $0x04 + JL four_bytes_remain_repeat_as_copy_encodeBlockAsm + + // emitRepeat +emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy: + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy + CMPL SI, $0x0100ffff + JLT repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy + LEAL -16842747(SI), SI + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy + +repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy: + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + JMP four_bytes_loop_back_repeat_as_copy_encodeBlockAsm + +four_bytes_remain_repeat_as_copy_encodeBlockAsm: + TESTL SI, SI + JZ repeat_end_emit_encodeBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVL DI, 1(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm + +two_byte_offset_repeat_as_copy_encodeBlockAsm: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeBlockAsm + CMPL DI, $0x00000800 + JAE long_offset_short_repeat_as_copy_encodeBlockAsm + MOVL $0x00000001, R8 + LEAL 16(R8), R8 + MOVB DI, 1(AX) + MOVL DI, R9 + SHRL $0x08, R9 + SHLL $0x05, R9 + ORL R9, R8 + MOVB R8, (AX) + ADDQ $0x02, AX + SUBL $0x08, SI + + // emitRepeat + LEAL -4(SI), SI + JMP cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + +emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + CMPL SI, $0x0100ffff + JLT repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + LEAL -16842747(SI), SI + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b + +repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short_2b: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +long_offset_short_repeat_as_copy_encodeBlockAsm: + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + + // emitRepeat +emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy_short: + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy_short + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy_short + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy_short + CMPL SI, $0x0100ffff + JLT repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy_short + LEAL -16842747(SI), SI + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_repeat_as_copy_encodeBlockAsm_emit_copy_short + +repeat_five_repeat_as_copy_encodeBlockAsm_emit_copy_short: + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_four_repeat_as_copy_encodeBlockAsm_emit_copy_short: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_three_repeat_as_copy_encodeBlockAsm_emit_copy_short: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_repeat_as_copy_encodeBlockAsm_emit_copy_short: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +repeat_two_offset_repeat_as_copy_encodeBlockAsm_emit_copy_short: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + JMP two_byte_offset_repeat_as_copy_encodeBlockAsm + +two_byte_offset_short_repeat_as_copy_encodeBlockAsm: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm + +emit_copy_three_repeat_as_copy_encodeBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeBlockAsm: + MOVL CX, 12(SP) + JMP search_loop_encodeBlockAsm + +no_repeat_found_encodeBlockAsm: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBlockAsm + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeBlockAsm + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeBlockAsm + +candidate3_match_encodeBlockAsm: + ADDL $0x02, CX + JMP candidate_match_encodeBlockAsm + +candidate2_match_encodeBlockAsm: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeBlockAsm: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBlockAsm + +match_extend_back_loop_encodeBlockAsm: + CMPL CX, DI + JLE match_extend_back_end_encodeBlockAsm + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBlockAsm + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBlockAsm + JMP match_extend_back_loop_encodeBlockAsm + +match_extend_back_end_encodeBlockAsm: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 5(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBlockAsm: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeBlockAsm + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeBlockAsm + CMPL R8, $0x00010000 + JLT three_bytes_match_emit_encodeBlockAsm + CMPL R8, $0x01000000 + JLT four_bytes_match_emit_encodeBlockAsm + MOVB $0xfc, (AX) + MOVL R8, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_match_emit_encodeBlockAsm + +four_bytes_match_emit_encodeBlockAsm: + MOVL R8, R10 + SHRL $0x10, R10 + MOVB $0xf8, (AX) + MOVW R8, 1(AX) + MOVB R10, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeBlockAsm + +three_bytes_match_emit_encodeBlockAsm: + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBlockAsm + +two_bytes_match_emit_encodeBlockAsm: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeBlockAsm + JMP memmove_long_match_emit_encodeBlockAsm + +one_byte_match_emit_encodeBlockAsm: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBlockAsm: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeBlockAsm + +emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm + +emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm + +emit_lit_memmove_match_emit_encodeBlockAsm_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBlockAsm: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeBlockAsm + +memmove_long_match_emit_encodeBlockAsm: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeBlockAsmlarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeBlockAsm: +match_nolit_loop_encodeBlockAsm: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeBlockAsm + +matchlen_loopback_match_nolit_encodeBlockAsm: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeBlockAsm + +matchlen_loop_match_nolit_encodeBlockAsm: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeBlockAsm + JZ match_nolit_end_encodeBlockAsm + +matchlen_match4_match_nolit_encodeBlockAsm: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeBlockAsm + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeBlockAsm + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeBlockAsm: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeBlockAsm + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeBlockAsm + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeBlockAsm: + CMPL DI, $0x01 + JL match_nolit_end_encodeBlockAsm + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeBlockAsm + LEAL 1(R10), R10 + +match_nolit_end_encodeBlockAsm: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy + CMPL SI, $0x00010000 + JL two_byte_offset_match_nolit_encodeBlockAsm + +four_bytes_loop_back_match_nolit_encodeBlockAsm: + CMPL R10, $0x40 + JLE four_bytes_remain_match_nolit_encodeBlockAsm + MOVB $0xff, (AX) + MOVL SI, 1(AX) + LEAL -64(R10), R10 + ADDQ $0x05, AX + CMPL R10, $0x04 + JL four_bytes_remain_match_nolit_encodeBlockAsm + + // emitRepeat +emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy: + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm_emit_copy + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy + +cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm_emit_copy + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm_emit_copy + CMPL R10, $0x0100ffff + JLT repeat_five_match_nolit_encodeBlockAsm_emit_copy + LEAL -16842747(R10), R10 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy + +repeat_five_match_nolit_encodeBlockAsm_emit_copy: + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_four_match_nolit_encodeBlockAsm_emit_copy: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_three_match_nolit_encodeBlockAsm_emit_copy: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_match_nolit_encodeBlockAsm_emit_copy: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + JMP four_bytes_loop_back_match_nolit_encodeBlockAsm + +four_bytes_remain_match_nolit_encodeBlockAsm: + TESTL R10, R10 + JZ match_nolit_emitcopy_end_encodeBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +two_byte_offset_match_nolit_encodeBlockAsm: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBlockAsm + CMPL SI, $0x00000800 + JAE long_offset_short_match_nolit_encodeBlockAsm + MOVL $0x00000001, DI + LEAL 16(DI), DI + MOVB SI, 1(AX) + MOVL SI, R8 + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, DI + MOVB DI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R10 + + // emitRepeat + LEAL -4(R10), R10 + JMP cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short_2b + +emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy_short_2b: + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm_emit_copy_short_2b + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short_2b + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short_2b: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm_emit_copy_short_2b + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm_emit_copy_short_2b + CMPL R10, $0x0100ffff + JLT repeat_five_match_nolit_encodeBlockAsm_emit_copy_short_2b + LEAL -16842747(R10), R10 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy_short_2b + +repeat_five_match_nolit_encodeBlockAsm_emit_copy_short_2b: + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_four_match_nolit_encodeBlockAsm_emit_copy_short_2b: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_three_match_nolit_encodeBlockAsm_emit_copy_short_2b: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_match_nolit_encodeBlockAsm_emit_copy_short_2b: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short_2b: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +long_offset_short_match_nolit_encodeBlockAsm: + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + + // emitRepeat +emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy_short: + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm_emit_copy_short + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm_emit_copy_short + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm_emit_copy_short + CMPL R10, $0x0100ffff + JLT repeat_five_match_nolit_encodeBlockAsm_emit_copy_short + LEAL -16842747(R10), R10 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBlockAsm_emit_copy_short + +repeat_five_match_nolit_encodeBlockAsm_emit_copy_short: + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_four_match_nolit_encodeBlockAsm_emit_copy_short: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_three_match_nolit_encodeBlockAsm_emit_copy_short: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_match_nolit_encodeBlockAsm_emit_copy_short: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +repeat_two_offset_match_nolit_encodeBlockAsm_emit_copy_short: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + JMP two_byte_offset_match_nolit_encodeBlockAsm + +two_byte_offset_short_match_nolit_encodeBlockAsm: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeBlockAsm + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm + +emit_copy_three_match_nolit_encodeBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeBlockAsm: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBlockAsm + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBlockAsm: + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x10, R8 + IMULQ R9, R8 + SHRQ $0x32, R8 + SHLQ $0x10, SI + IMULQ R9, SI + SHRQ $0x32, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeBlockAsm + INCL CX + JMP search_loop_encodeBlockAsm + +emit_remainder_encodeBlockAsm: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 5(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBlockAsm: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBlockAsm + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBlockAsm + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBlockAsm + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeBlockAsm + CMPL DX, $0x01000000 + JLT four_bytes_emit_remainder_encodeBlockAsm + MOVB $0xfc, (AX) + MOVL DX, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_emit_remainder_encodeBlockAsm + +four_bytes_emit_remainder_encodeBlockAsm: + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeBlockAsm + +three_bytes_emit_remainder_encodeBlockAsm: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBlockAsm + +two_bytes_emit_remainder_encodeBlockAsm: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBlockAsm + JMP memmove_long_emit_remainder_encodeBlockAsm + +one_byte_emit_remainder_encodeBlockAsm: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm + +emit_lit_memmove_emit_remainder_encodeBlockAsm_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBlockAsm: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBlockAsm + +memmove_long_emit_remainder_encodeBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBlockAsmlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBlockAsm: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBlockAsm4MB(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBlockAsm4MB(SB), $65560-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000200, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBlockAsm4MB: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBlockAsm4MB + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBlockAsm4MB: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBlockAsm4MB + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R11 + IMULQ R9, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeBlockAsm4MB + LEAL 1(CX), DI + MOVL 12(SP), R8 + MOVL DI, SI + SUBL 16(SP), SI + JZ repeat_extend_back_end_encodeBlockAsm4MB + +repeat_extend_back_loop_encodeBlockAsm4MB: + CMPL DI, R8 + JLE repeat_extend_back_end_encodeBlockAsm4MB + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeBlockAsm4MB + LEAL -1(DI), DI + DECL SI + JNZ repeat_extend_back_loop_encodeBlockAsm4MB + +repeat_extend_back_end_encodeBlockAsm4MB: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeBlockAsm4MB + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeBlockAsm4MB + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeBlockAsm4MB + CMPL SI, $0x00010000 + JLT three_bytes_repeat_emit_encodeBlockAsm4MB + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_repeat_emit_encodeBlockAsm4MB + +three_bytes_repeat_emit_encodeBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeBlockAsm4MB + +two_bytes_repeat_emit_encodeBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeBlockAsm4MB + JMP memmove_long_repeat_emit_encodeBlockAsm4MB + +one_byte_repeat_emit_encodeBlockAsm4MB: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm4MB + +emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm4MB + +emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm4MB + +emit_lit_memmove_repeat_emit_encodeBlockAsm4MB_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_repeat_emit_encodeBlockAsm4MB: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeBlockAsm4MB + +memmove_long_repeat_emit_encodeBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R12 + SHRQ $0x05, R12 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R13 + SUBQ R11, R13 + DECQ R12 + JA emit_lit_memmove_long_repeat_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(R10)(R13*1), R11 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm4MBlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R11 + ADDQ $0x20, R13 + DECQ R12 + JNA emit_lit_memmove_long_repeat_emit_encodeBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(R10)(R13*1), X4 + MOVOU -16(R10)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R9, R13 + JAE emit_lit_memmove_long_repeat_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeBlockAsm4MB: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R9 + SUBL CX, R9 + LEAQ (DX)(CX*1), R10 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R12, R12 + CMPL R9, $0x08 + JL matchlen_match4_repeat_extend_encodeBlockAsm4MB + +matchlen_loopback_repeat_extend_encodeBlockAsm4MB: + MOVQ (R10)(R12*1), R11 + XORQ (SI)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_repeat_extend_encodeBlockAsm4MB + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP repeat_extend_forward_end_encodeBlockAsm4MB + +matchlen_loop_repeat_extend_encodeBlockAsm4MB: + LEAL -8(R9), R9 + LEAL 8(R12), R12 + CMPL R9, $0x08 + JGE matchlen_loopback_repeat_extend_encodeBlockAsm4MB + JZ repeat_extend_forward_end_encodeBlockAsm4MB + +matchlen_match4_repeat_extend_encodeBlockAsm4MB: + CMPL R9, $0x04 + JL matchlen_match2_repeat_extend_encodeBlockAsm4MB + MOVL (R10)(R12*1), R11 + CMPL (SI)(R12*1), R11 + JNE matchlen_match2_repeat_extend_encodeBlockAsm4MB + SUBL $0x04, R9 + LEAL 4(R12), R12 + +matchlen_match2_repeat_extend_encodeBlockAsm4MB: + CMPL R9, $0x02 + JL matchlen_match1_repeat_extend_encodeBlockAsm4MB + MOVW (R10)(R12*1), R11 + CMPW (SI)(R12*1), R11 + JNE matchlen_match1_repeat_extend_encodeBlockAsm4MB + SUBL $0x02, R9 + LEAL 2(R12), R12 + +matchlen_match1_repeat_extend_encodeBlockAsm4MB: + CMPL R9, $0x01 + JL repeat_extend_forward_end_encodeBlockAsm4MB + MOVB (R10)(R12*1), R11 + CMPB (SI)(R12*1), R11 + JNE repeat_extend_forward_end_encodeBlockAsm4MB + LEAL 1(R12), R12 + +repeat_extend_forward_end_encodeBlockAsm4MB: + ADDL R12, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + TESTL R8, R8 + JZ repeat_as_copy_encodeBlockAsm4MB + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_match_repeat_encodeBlockAsm4MB + CMPL R8, $0x0c + JGE cant_repeat_two_offset_match_repeat_encodeBlockAsm4MB + CMPL DI, $0x00000800 + JLT repeat_two_offset_match_repeat_encodeBlockAsm4MB + +cant_repeat_two_offset_match_repeat_encodeBlockAsm4MB: + CMPL SI, $0x00000104 + JLT repeat_three_match_repeat_encodeBlockAsm4MB + CMPL SI, $0x00010100 + JLT repeat_four_match_repeat_encodeBlockAsm4MB + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_four_match_repeat_encodeBlockAsm4MB: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_three_match_repeat_encodeBlockAsm4MB: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_match_repeat_encodeBlockAsm4MB: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_offset_match_repeat_encodeBlockAsm4MB: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_as_copy_encodeBlockAsm4MB: + // emitCopy + CMPL DI, $0x00010000 + JL two_byte_offset_repeat_as_copy_encodeBlockAsm4MB + +four_bytes_loop_back_repeat_as_copy_encodeBlockAsm4MB: + CMPL SI, $0x40 + JLE four_bytes_remain_repeat_as_copy_encodeBlockAsm4MB + MOVB $0xff, (AX) + MOVL DI, 1(AX) + LEAL -64(SI), SI + ADDQ $0x05, AX + CMPL SI, $0x04 + JL four_bytes_remain_repeat_as_copy_encodeBlockAsm4MB + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + JMP four_bytes_loop_back_repeat_as_copy_encodeBlockAsm4MB + +four_bytes_remain_repeat_as_copy_encodeBlockAsm4MB: + TESTL SI, SI + JZ repeat_end_emit_encodeBlockAsm4MB + MOVB $0x03, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVL DI, 1(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +two_byte_offset_repeat_as_copy_encodeBlockAsm4MB: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeBlockAsm4MB + CMPL DI, $0x00000800 + JAE long_offset_short_repeat_as_copy_encodeBlockAsm4MB + MOVL $0x00000001, R8 + LEAL 16(R8), R8 + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, R8 + MOVB R8, (AX) + ADDQ $0x02, AX + SUBL $0x08, SI + + // emitRepeat + LEAL -4(SI), SI + JMP cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short_2b: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +long_offset_short_repeat_as_copy_encodeBlockAsm4MB: + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short + CMPL SI, $0x00010100 + JLT repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short + LEAL -65536(SI), SI + MOVL SI, DI + MOVW $0x001d, (AX) + MOVW SI, 2(AX) + SARL $0x10, DI + MOVB DI, 4(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_four_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short: + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_three_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +repeat_two_offset_repeat_as_copy_encodeBlockAsm4MB_emit_copy_short: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + JMP two_byte_offset_repeat_as_copy_encodeBlockAsm4MB + +two_byte_offset_short_repeat_as_copy_encodeBlockAsm4MB: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm4MB + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm4MB + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm4MB + +emit_copy_three_repeat_as_copy_encodeBlockAsm4MB: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeBlockAsm4MB: + MOVL CX, 12(SP) + JMP search_loop_encodeBlockAsm4MB + +no_repeat_found_encodeBlockAsm4MB: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBlockAsm4MB + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeBlockAsm4MB + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeBlockAsm4MB + MOVL 20(SP), CX + JMP search_loop_encodeBlockAsm4MB + +candidate3_match_encodeBlockAsm4MB: + ADDL $0x02, CX + JMP candidate_match_encodeBlockAsm4MB + +candidate2_match_encodeBlockAsm4MB: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeBlockAsm4MB: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBlockAsm4MB + +match_extend_back_loop_encodeBlockAsm4MB: + CMPL CX, DI + JLE match_extend_back_end_encodeBlockAsm4MB + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBlockAsm4MB + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBlockAsm4MB + JMP match_extend_back_loop_encodeBlockAsm4MB + +match_extend_back_end_encodeBlockAsm4MB: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 4(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBlockAsm4MB: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeBlockAsm4MB + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeBlockAsm4MB + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeBlockAsm4MB + CMPL R8, $0x00010000 + JLT three_bytes_match_emit_encodeBlockAsm4MB + MOVL R8, R10 + SHRL $0x10, R10 + MOVB $0xf8, (AX) + MOVW R8, 1(AX) + MOVB R10, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeBlockAsm4MB + +three_bytes_match_emit_encodeBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBlockAsm4MB + +two_bytes_match_emit_encodeBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeBlockAsm4MB + JMP memmove_long_match_emit_encodeBlockAsm4MB + +one_byte_match_emit_encodeBlockAsm4MB: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBlockAsm4MB: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBlockAsm4MB_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBlockAsm4MB: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeBlockAsm4MB + +memmove_long_match_emit_encodeBlockAsm4MB: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeBlockAsm4MBlarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeBlockAsm4MB: +match_nolit_loop_encodeBlockAsm4MB: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeBlockAsm4MB + +matchlen_loopback_match_nolit_encodeBlockAsm4MB: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeBlockAsm4MB + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeBlockAsm4MB + +matchlen_loop_match_nolit_encodeBlockAsm4MB: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeBlockAsm4MB + JZ match_nolit_end_encodeBlockAsm4MB + +matchlen_match4_match_nolit_encodeBlockAsm4MB: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeBlockAsm4MB + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeBlockAsm4MB + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeBlockAsm4MB: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeBlockAsm4MB + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeBlockAsm4MB + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeBlockAsm4MB: + CMPL DI, $0x01 + JL match_nolit_end_encodeBlockAsm4MB + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeBlockAsm4MB + LEAL 1(R10), R10 + +match_nolit_end_encodeBlockAsm4MB: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy + CMPL SI, $0x00010000 + JL two_byte_offset_match_nolit_encodeBlockAsm4MB + +four_bytes_loop_back_match_nolit_encodeBlockAsm4MB: + CMPL R10, $0x40 + JLE four_bytes_remain_match_nolit_encodeBlockAsm4MB + MOVB $0xff, (AX) + MOVL SI, 1(AX) + LEAL -64(R10), R10 + ADDQ $0x05, AX + CMPL R10, $0x04 + JL four_bytes_remain_match_nolit_encodeBlockAsm4MB + + // emitRepeat + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy + +cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + JMP four_bytes_loop_back_match_nolit_encodeBlockAsm4MB + +four_bytes_remain_match_nolit_encodeBlockAsm4MB: + TESTL R10, R10 + JZ match_nolit_emitcopy_end_encodeBlockAsm4MB + MOVB $0x03, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +two_byte_offset_match_nolit_encodeBlockAsm4MB: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBlockAsm4MB + CMPL SI, $0x00000800 + JAE long_offset_short_match_nolit_encodeBlockAsm4MB + MOVL $0x00000001, DI + LEAL 16(DI), DI + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, DI + MOVB DI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R10 + + // emitRepeat + LEAL -4(R10), R10 + JMP cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short_2b: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +long_offset_short_match_nolit_encodeBlockAsm4MB: + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + + // emitRepeat + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy_short + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy_short + CMPL R10, $0x00010100 + JLT repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy_short + LEAL -65536(R10), R10 + MOVL R10, SI + MOVW $0x001d, (AX) + MOVW R10, 2(AX) + SARL $0x10, SI + MOVB SI, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_four_match_nolit_encodeBlockAsm4MB_emit_copy_short: + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_three_match_nolit_encodeBlockAsm4MB_emit_copy_short: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_match_nolit_encodeBlockAsm4MB_emit_copy_short: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBlockAsm4MB_emit_copy_short: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + JMP two_byte_offset_match_nolit_encodeBlockAsm4MB + +two_byte_offset_short_match_nolit_encodeBlockAsm4MB: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeBlockAsm4MB + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBlockAsm4MB + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm4MB + +emit_copy_three_match_nolit_encodeBlockAsm4MB: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeBlockAsm4MB: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBlockAsm4MB + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBlockAsm4MB: + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x10, R8 + IMULQ R9, R8 + SHRQ $0x32, R8 + SHLQ $0x10, SI + IMULQ R9, SI + SHRQ $0x32, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeBlockAsm4MB + INCL CX + JMP search_loop_encodeBlockAsm4MB + +emit_remainder_encodeBlockAsm4MB: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 4(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBlockAsm4MB: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBlockAsm4MB + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBlockAsm4MB + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBlockAsm4MB + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeBlockAsm4MB + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeBlockAsm4MB + +three_bytes_emit_remainder_encodeBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBlockAsm4MB + +two_bytes_emit_remainder_encodeBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBlockAsm4MB + JMP memmove_long_emit_remainder_encodeBlockAsm4MB + +one_byte_emit_remainder_encodeBlockAsm4MB: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBlockAsm4MB: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBlockAsm4MB_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBlockAsm4MB: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBlockAsm4MB + +memmove_long_emit_remainder_encodeBlockAsm4MB: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm4MBlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBlockAsm4MB: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBlockAsm12B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBlockAsm12B(SB), $16408-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000080, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBlockAsm12B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBlockAsm12B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBlockAsm12B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBlockAsm12B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x000000cf1bbcdcbb, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x18, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + SHLQ $0x18, R11 + IMULQ R9, R11 + SHRQ $0x34, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x18, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeBlockAsm12B + LEAL 1(CX), DI + MOVL 12(SP), R8 + MOVL DI, SI + SUBL 16(SP), SI + JZ repeat_extend_back_end_encodeBlockAsm12B + +repeat_extend_back_loop_encodeBlockAsm12B: + CMPL DI, R8 + JLE repeat_extend_back_end_encodeBlockAsm12B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeBlockAsm12B + LEAL -1(DI), DI + DECL SI + JNZ repeat_extend_back_loop_encodeBlockAsm12B + +repeat_extend_back_end_encodeBlockAsm12B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeBlockAsm12B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeBlockAsm12B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeBlockAsm12B + +two_bytes_repeat_emit_encodeBlockAsm12B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeBlockAsm12B + JMP memmove_long_repeat_emit_encodeBlockAsm12B + +one_byte_repeat_emit_encodeBlockAsm12B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeBlockAsm12B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_repeat_emit_encodeBlockAsm12B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeBlockAsm12B + +memmove_long_repeat_emit_encodeBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R12 + SHRQ $0x05, R12 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R13 + SUBQ R11, R13 + DECQ R12 + JA emit_lit_memmove_long_repeat_emit_encodeBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R13*1), R11 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm12Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R11 + ADDQ $0x20, R13 + DECQ R12 + JNA emit_lit_memmove_long_repeat_emit_encodeBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R13*1), X4 + MOVOU -16(R10)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R9, R13 + JAE emit_lit_memmove_long_repeat_emit_encodeBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeBlockAsm12B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R9 + SUBL CX, R9 + LEAQ (DX)(CX*1), R10 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R12, R12 + CMPL R9, $0x08 + JL matchlen_match4_repeat_extend_encodeBlockAsm12B + +matchlen_loopback_repeat_extend_encodeBlockAsm12B: + MOVQ (R10)(R12*1), R11 + XORQ (SI)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_repeat_extend_encodeBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP repeat_extend_forward_end_encodeBlockAsm12B + +matchlen_loop_repeat_extend_encodeBlockAsm12B: + LEAL -8(R9), R9 + LEAL 8(R12), R12 + CMPL R9, $0x08 + JGE matchlen_loopback_repeat_extend_encodeBlockAsm12B + JZ repeat_extend_forward_end_encodeBlockAsm12B + +matchlen_match4_repeat_extend_encodeBlockAsm12B: + CMPL R9, $0x04 + JL matchlen_match2_repeat_extend_encodeBlockAsm12B + MOVL (R10)(R12*1), R11 + CMPL (SI)(R12*1), R11 + JNE matchlen_match2_repeat_extend_encodeBlockAsm12B + SUBL $0x04, R9 + LEAL 4(R12), R12 + +matchlen_match2_repeat_extend_encodeBlockAsm12B: + CMPL R9, $0x02 + JL matchlen_match1_repeat_extend_encodeBlockAsm12B + MOVW (R10)(R12*1), R11 + CMPW (SI)(R12*1), R11 + JNE matchlen_match1_repeat_extend_encodeBlockAsm12B + SUBL $0x02, R9 + LEAL 2(R12), R12 + +matchlen_match1_repeat_extend_encodeBlockAsm12B: + CMPL R9, $0x01 + JL repeat_extend_forward_end_encodeBlockAsm12B + MOVB (R10)(R12*1), R11 + CMPB (SI)(R12*1), R11 + JNE repeat_extend_forward_end_encodeBlockAsm12B + LEAL 1(R12), R12 + +repeat_extend_forward_end_encodeBlockAsm12B: + ADDL R12, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + TESTL R8, R8 + JZ repeat_as_copy_encodeBlockAsm12B + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_match_repeat_encodeBlockAsm12B + CMPL R8, $0x0c + JGE cant_repeat_two_offset_match_repeat_encodeBlockAsm12B + CMPL DI, $0x00000800 + JLT repeat_two_offset_match_repeat_encodeBlockAsm12B + +cant_repeat_two_offset_match_repeat_encodeBlockAsm12B: + CMPL SI, $0x00000104 + JLT repeat_three_match_repeat_encodeBlockAsm12B + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_three_match_repeat_encodeBlockAsm12B: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_match_repeat_encodeBlockAsm12B: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_offset_match_repeat_encodeBlockAsm12B: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_as_copy_encodeBlockAsm12B: + // emitCopy +two_byte_offset_repeat_as_copy_encodeBlockAsm12B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeBlockAsm12B + CMPL DI, $0x00000800 + JAE long_offset_short_repeat_as_copy_encodeBlockAsm12B + MOVL $0x00000001, R8 + LEAL 16(R8), R8 + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, R8 + MOVB R8, (AX) + ADDQ $0x02, AX + SUBL $0x08, SI + + // emitRepeat + LEAL -4(SI), SI + JMP cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_three_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short_2b: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +long_offset_short_repeat_as_copy_encodeBlockAsm12B: + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm12B_emit_copy_short + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm12B_emit_copy_short + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_three_repeat_as_copy_encodeBlockAsm12B_emit_copy_short: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_repeat_as_copy_encodeBlockAsm12B_emit_copy_short: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +repeat_two_offset_repeat_as_copy_encodeBlockAsm12B_emit_copy_short: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + JMP two_byte_offset_repeat_as_copy_encodeBlockAsm12B + +two_byte_offset_short_repeat_as_copy_encodeBlockAsm12B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm12B + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm12B + +emit_copy_three_repeat_as_copy_encodeBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeBlockAsm12B: + MOVL CX, 12(SP) + JMP search_loop_encodeBlockAsm12B + +no_repeat_found_encodeBlockAsm12B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBlockAsm12B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeBlockAsm12B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeBlockAsm12B + +candidate3_match_encodeBlockAsm12B: + ADDL $0x02, CX + JMP candidate_match_encodeBlockAsm12B + +candidate2_match_encodeBlockAsm12B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeBlockAsm12B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBlockAsm12B + +match_extend_back_loop_encodeBlockAsm12B: + CMPL CX, DI + JLE match_extend_back_end_encodeBlockAsm12B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBlockAsm12B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBlockAsm12B + JMP match_extend_back_loop_encodeBlockAsm12B + +match_extend_back_end_encodeBlockAsm12B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBlockAsm12B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeBlockAsm12B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeBlockAsm12B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBlockAsm12B + +two_bytes_match_emit_encodeBlockAsm12B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeBlockAsm12B + JMP memmove_long_match_emit_encodeBlockAsm12B + +one_byte_match_emit_encodeBlockAsm12B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBlockAsm12B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeBlockAsm12B + +emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm12B + +emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm12B + +emit_lit_memmove_match_emit_encodeBlockAsm12B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBlockAsm12B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeBlockAsm12B + +memmove_long_match_emit_encodeBlockAsm12B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeBlockAsm12Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeBlockAsm12B: +match_nolit_loop_encodeBlockAsm12B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeBlockAsm12B + +matchlen_loopback_match_nolit_encodeBlockAsm12B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeBlockAsm12B + +matchlen_loop_match_nolit_encodeBlockAsm12B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeBlockAsm12B + JZ match_nolit_end_encodeBlockAsm12B + +matchlen_match4_match_nolit_encodeBlockAsm12B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeBlockAsm12B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeBlockAsm12B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeBlockAsm12B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeBlockAsm12B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeBlockAsm12B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeBlockAsm12B: + CMPL DI, $0x01 + JL match_nolit_end_encodeBlockAsm12B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeBlockAsm12B + LEAL 1(R10), R10 + +match_nolit_end_encodeBlockAsm12B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBlockAsm12B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBlockAsm12B + CMPL SI, $0x00000800 + JAE long_offset_short_match_nolit_encodeBlockAsm12B + MOVL $0x00000001, DI + LEAL 16(DI), DI + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, DI + MOVB DI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R10 + + // emitRepeat + LEAL -4(R10), R10 + JMP cant_repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short_2b + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm12B_emit_copy_short_2b + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short_2b + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short_2b: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm12B_emit_copy_short_2b + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_three_match_nolit_encodeBlockAsm12B_emit_copy_short_2b: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_two_match_nolit_encodeBlockAsm12B_emit_copy_short_2b: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short_2b: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +long_offset_short_match_nolit_encodeBlockAsm12B: + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + + // emitRepeat + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm12B_emit_copy_short + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm12B_emit_copy_short + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_three_match_nolit_encodeBlockAsm12B_emit_copy_short: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_two_match_nolit_encodeBlockAsm12B_emit_copy_short: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +repeat_two_offset_match_nolit_encodeBlockAsm12B_emit_copy_short: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + JMP two_byte_offset_match_nolit_encodeBlockAsm12B + +two_byte_offset_short_match_nolit_encodeBlockAsm12B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeBlockAsm12B + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm12B + +emit_copy_three_match_nolit_encodeBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeBlockAsm12B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBlockAsm12B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBlockAsm12B: + MOVQ $0x000000cf1bbcdcbb, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x18, R8 + IMULQ R9, R8 + SHRQ $0x34, R8 + SHLQ $0x18, SI + IMULQ R9, SI + SHRQ $0x34, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeBlockAsm12B + INCL CX + JMP search_loop_encodeBlockAsm12B + +emit_remainder_encodeBlockAsm12B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBlockAsm12B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBlockAsm12B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBlockAsm12B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBlockAsm12B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBlockAsm12B + +two_bytes_emit_remainder_encodeBlockAsm12B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBlockAsm12B + JMP memmove_long_emit_remainder_encodeBlockAsm12B + +one_byte_emit_remainder_encodeBlockAsm12B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBlockAsm12B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBlockAsm12B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBlockAsm12B + +memmove_long_emit_remainder_encodeBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm12Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBlockAsm12B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBlockAsm10B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBlockAsm10B(SB), $4120-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000020, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBlockAsm10B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBlockAsm10B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBlockAsm10B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBlockAsm10B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x9e3779b1, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ R9, R11 + SHRQ $0x36, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeBlockAsm10B + LEAL 1(CX), DI + MOVL 12(SP), R8 + MOVL DI, SI + SUBL 16(SP), SI + JZ repeat_extend_back_end_encodeBlockAsm10B + +repeat_extend_back_loop_encodeBlockAsm10B: + CMPL DI, R8 + JLE repeat_extend_back_end_encodeBlockAsm10B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeBlockAsm10B + LEAL -1(DI), DI + DECL SI + JNZ repeat_extend_back_loop_encodeBlockAsm10B + +repeat_extend_back_end_encodeBlockAsm10B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeBlockAsm10B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeBlockAsm10B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeBlockAsm10B + +two_bytes_repeat_emit_encodeBlockAsm10B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeBlockAsm10B + JMP memmove_long_repeat_emit_encodeBlockAsm10B + +one_byte_repeat_emit_encodeBlockAsm10B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeBlockAsm10B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_repeat_emit_encodeBlockAsm10B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeBlockAsm10B + +memmove_long_repeat_emit_encodeBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R12 + SHRQ $0x05, R12 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R13 + SUBQ R11, R13 + DECQ R12 + JA emit_lit_memmove_long_repeat_emit_encodeBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R13*1), R11 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm10Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R11 + ADDQ $0x20, R13 + DECQ R12 + JNA emit_lit_memmove_long_repeat_emit_encodeBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R13*1), X4 + MOVOU -16(R10)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R9, R13 + JAE emit_lit_memmove_long_repeat_emit_encodeBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeBlockAsm10B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R9 + SUBL CX, R9 + LEAQ (DX)(CX*1), R10 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R12, R12 + CMPL R9, $0x08 + JL matchlen_match4_repeat_extend_encodeBlockAsm10B + +matchlen_loopback_repeat_extend_encodeBlockAsm10B: + MOVQ (R10)(R12*1), R11 + XORQ (SI)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_repeat_extend_encodeBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP repeat_extend_forward_end_encodeBlockAsm10B + +matchlen_loop_repeat_extend_encodeBlockAsm10B: + LEAL -8(R9), R9 + LEAL 8(R12), R12 + CMPL R9, $0x08 + JGE matchlen_loopback_repeat_extend_encodeBlockAsm10B + JZ repeat_extend_forward_end_encodeBlockAsm10B + +matchlen_match4_repeat_extend_encodeBlockAsm10B: + CMPL R9, $0x04 + JL matchlen_match2_repeat_extend_encodeBlockAsm10B + MOVL (R10)(R12*1), R11 + CMPL (SI)(R12*1), R11 + JNE matchlen_match2_repeat_extend_encodeBlockAsm10B + SUBL $0x04, R9 + LEAL 4(R12), R12 + +matchlen_match2_repeat_extend_encodeBlockAsm10B: + CMPL R9, $0x02 + JL matchlen_match1_repeat_extend_encodeBlockAsm10B + MOVW (R10)(R12*1), R11 + CMPW (SI)(R12*1), R11 + JNE matchlen_match1_repeat_extend_encodeBlockAsm10B + SUBL $0x02, R9 + LEAL 2(R12), R12 + +matchlen_match1_repeat_extend_encodeBlockAsm10B: + CMPL R9, $0x01 + JL repeat_extend_forward_end_encodeBlockAsm10B + MOVB (R10)(R12*1), R11 + CMPB (SI)(R12*1), R11 + JNE repeat_extend_forward_end_encodeBlockAsm10B + LEAL 1(R12), R12 + +repeat_extend_forward_end_encodeBlockAsm10B: + ADDL R12, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + TESTL R8, R8 + JZ repeat_as_copy_encodeBlockAsm10B + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_match_repeat_encodeBlockAsm10B + CMPL R8, $0x0c + JGE cant_repeat_two_offset_match_repeat_encodeBlockAsm10B + CMPL DI, $0x00000800 + JLT repeat_two_offset_match_repeat_encodeBlockAsm10B + +cant_repeat_two_offset_match_repeat_encodeBlockAsm10B: + CMPL SI, $0x00000104 + JLT repeat_three_match_repeat_encodeBlockAsm10B + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_three_match_repeat_encodeBlockAsm10B: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_match_repeat_encodeBlockAsm10B: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_offset_match_repeat_encodeBlockAsm10B: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_as_copy_encodeBlockAsm10B: + // emitCopy +two_byte_offset_repeat_as_copy_encodeBlockAsm10B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeBlockAsm10B + CMPL DI, $0x00000800 + JAE long_offset_short_repeat_as_copy_encodeBlockAsm10B + MOVL $0x00000001, R8 + LEAL 16(R8), R8 + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, R8 + MOVB R8, (AX) + ADDQ $0x02, AX + SUBL $0x08, SI + + // emitRepeat + LEAL -4(SI), SI + JMP cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_three_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short_2b: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +long_offset_short_repeat_as_copy_encodeBlockAsm10B: + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + + // emitRepeat + MOVL SI, R8 + LEAL -4(SI), SI + CMPL R8, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm10B_emit_copy_short + CMPL R8, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short + CMPL DI, $0x00000800 + JLT repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm10B_emit_copy_short + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_three_repeat_as_copy_encodeBlockAsm10B_emit_copy_short: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_repeat_as_copy_encodeBlockAsm10B_emit_copy_short: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +repeat_two_offset_repeat_as_copy_encodeBlockAsm10B_emit_copy_short: + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + JMP two_byte_offset_repeat_as_copy_encodeBlockAsm10B + +two_byte_offset_short_repeat_as_copy_encodeBlockAsm10B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm10B + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm10B + +emit_copy_three_repeat_as_copy_encodeBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeBlockAsm10B: + MOVL CX, 12(SP) + JMP search_loop_encodeBlockAsm10B + +no_repeat_found_encodeBlockAsm10B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBlockAsm10B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeBlockAsm10B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeBlockAsm10B + +candidate3_match_encodeBlockAsm10B: + ADDL $0x02, CX + JMP candidate_match_encodeBlockAsm10B + +candidate2_match_encodeBlockAsm10B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeBlockAsm10B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBlockAsm10B + +match_extend_back_loop_encodeBlockAsm10B: + CMPL CX, DI + JLE match_extend_back_end_encodeBlockAsm10B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBlockAsm10B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBlockAsm10B + JMP match_extend_back_loop_encodeBlockAsm10B + +match_extend_back_end_encodeBlockAsm10B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBlockAsm10B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeBlockAsm10B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeBlockAsm10B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBlockAsm10B + +two_bytes_match_emit_encodeBlockAsm10B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeBlockAsm10B + JMP memmove_long_match_emit_encodeBlockAsm10B + +one_byte_match_emit_encodeBlockAsm10B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBlockAsm10B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeBlockAsm10B + +emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm10B + +emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm10B + +emit_lit_memmove_match_emit_encodeBlockAsm10B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBlockAsm10B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeBlockAsm10B + +memmove_long_match_emit_encodeBlockAsm10B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeBlockAsm10Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeBlockAsm10B: +match_nolit_loop_encodeBlockAsm10B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeBlockAsm10B + +matchlen_loopback_match_nolit_encodeBlockAsm10B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeBlockAsm10B + +matchlen_loop_match_nolit_encodeBlockAsm10B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeBlockAsm10B + JZ match_nolit_end_encodeBlockAsm10B + +matchlen_match4_match_nolit_encodeBlockAsm10B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeBlockAsm10B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeBlockAsm10B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeBlockAsm10B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeBlockAsm10B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeBlockAsm10B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeBlockAsm10B: + CMPL DI, $0x01 + JL match_nolit_end_encodeBlockAsm10B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeBlockAsm10B + LEAL 1(R10), R10 + +match_nolit_end_encodeBlockAsm10B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBlockAsm10B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBlockAsm10B + CMPL SI, $0x00000800 + JAE long_offset_short_match_nolit_encodeBlockAsm10B + MOVL $0x00000001, DI + LEAL 16(DI), DI + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, DI + MOVB DI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R10 + + // emitRepeat + LEAL -4(R10), R10 + JMP cant_repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short_2b + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm10B_emit_copy_short_2b + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short_2b + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short_2b: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm10B_emit_copy_short_2b + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_three_match_nolit_encodeBlockAsm10B_emit_copy_short_2b: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_two_match_nolit_encodeBlockAsm10B_emit_copy_short_2b: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short_2b: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +long_offset_short_match_nolit_encodeBlockAsm10B: + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + + // emitRepeat + MOVL R10, DI + LEAL -4(R10), R10 + CMPL DI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm10B_emit_copy_short + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short + CMPL SI, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm10B_emit_copy_short + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_three_match_nolit_encodeBlockAsm10B_emit_copy_short: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_two_match_nolit_encodeBlockAsm10B_emit_copy_short: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +repeat_two_offset_match_nolit_encodeBlockAsm10B_emit_copy_short: + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + JMP two_byte_offset_match_nolit_encodeBlockAsm10B + +two_byte_offset_short_match_nolit_encodeBlockAsm10B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeBlockAsm10B + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm10B + +emit_copy_three_match_nolit_encodeBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeBlockAsm10B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBlockAsm10B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBlockAsm10B: + MOVQ $0x9e3779b1, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x20, R8 + IMULQ R9, R8 + SHRQ $0x36, R8 + SHLQ $0x20, SI + IMULQ R9, SI + SHRQ $0x36, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeBlockAsm10B + INCL CX + JMP search_loop_encodeBlockAsm10B + +emit_remainder_encodeBlockAsm10B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBlockAsm10B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBlockAsm10B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBlockAsm10B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBlockAsm10B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBlockAsm10B + +two_bytes_emit_remainder_encodeBlockAsm10B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBlockAsm10B + JMP memmove_long_emit_remainder_encodeBlockAsm10B + +one_byte_emit_remainder_encodeBlockAsm10B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBlockAsm10B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBlockAsm10B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBlockAsm10B + +memmove_long_emit_remainder_encodeBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm10Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBlockAsm10B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBlockAsm8B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBlockAsm8B(SB), $1048-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000008, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBlockAsm8B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBlockAsm8B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBlockAsm8B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x04, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBlockAsm8B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x9e3779b1, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x38, R10 + SHLQ $0x20, R11 + IMULQ R9, R11 + SHRQ $0x38, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x38, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeBlockAsm8B + LEAL 1(CX), DI + MOVL 12(SP), R8 + MOVL DI, SI + SUBL 16(SP), SI + JZ repeat_extend_back_end_encodeBlockAsm8B + +repeat_extend_back_loop_encodeBlockAsm8B: + CMPL DI, R8 + JLE repeat_extend_back_end_encodeBlockAsm8B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeBlockAsm8B + LEAL -1(DI), DI + DECL SI + JNZ repeat_extend_back_loop_encodeBlockAsm8B + +repeat_extend_back_end_encodeBlockAsm8B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeBlockAsm8B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeBlockAsm8B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeBlockAsm8B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeBlockAsm8B + +two_bytes_repeat_emit_encodeBlockAsm8B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeBlockAsm8B + JMP memmove_long_repeat_emit_encodeBlockAsm8B + +one_byte_repeat_emit_encodeBlockAsm8B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_repeat_emit_encodeBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeBlockAsm8B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_repeat_emit_encodeBlockAsm8B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeBlockAsm8B + +memmove_long_repeat_emit_encodeBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R12 + SHRQ $0x05, R12 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R13 + SUBQ R11, R13 + DECQ R12 + JA emit_lit_memmove_long_repeat_emit_encodeBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R13*1), R11 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm8Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R11 + ADDQ $0x20, R13 + DECQ R12 + JNA emit_lit_memmove_long_repeat_emit_encodeBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R13*1), X4 + MOVOU -16(R10)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R9, R13 + JAE emit_lit_memmove_long_repeat_emit_encodeBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeBlockAsm8B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R9 + SUBL CX, R9 + LEAQ (DX)(CX*1), R10 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R12, R12 + CMPL R9, $0x08 + JL matchlen_match4_repeat_extend_encodeBlockAsm8B + +matchlen_loopback_repeat_extend_encodeBlockAsm8B: + MOVQ (R10)(R12*1), R11 + XORQ (SI)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_repeat_extend_encodeBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP repeat_extend_forward_end_encodeBlockAsm8B + +matchlen_loop_repeat_extend_encodeBlockAsm8B: + LEAL -8(R9), R9 + LEAL 8(R12), R12 + CMPL R9, $0x08 + JGE matchlen_loopback_repeat_extend_encodeBlockAsm8B + JZ repeat_extend_forward_end_encodeBlockAsm8B + +matchlen_match4_repeat_extend_encodeBlockAsm8B: + CMPL R9, $0x04 + JL matchlen_match2_repeat_extend_encodeBlockAsm8B + MOVL (R10)(R12*1), R11 + CMPL (SI)(R12*1), R11 + JNE matchlen_match2_repeat_extend_encodeBlockAsm8B + SUBL $0x04, R9 + LEAL 4(R12), R12 + +matchlen_match2_repeat_extend_encodeBlockAsm8B: + CMPL R9, $0x02 + JL matchlen_match1_repeat_extend_encodeBlockAsm8B + MOVW (R10)(R12*1), R11 + CMPW (SI)(R12*1), R11 + JNE matchlen_match1_repeat_extend_encodeBlockAsm8B + SUBL $0x02, R9 + LEAL 2(R12), R12 + +matchlen_match1_repeat_extend_encodeBlockAsm8B: + CMPL R9, $0x01 + JL repeat_extend_forward_end_encodeBlockAsm8B + MOVB (R10)(R12*1), R11 + CMPB (SI)(R12*1), R11 + JNE repeat_extend_forward_end_encodeBlockAsm8B + LEAL 1(R12), R12 + +repeat_extend_forward_end_encodeBlockAsm8B: + ADDL R12, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + TESTL R8, R8 + JZ repeat_as_copy_encodeBlockAsm8B + + // emitRepeat + MOVL SI, DI + LEAL -4(SI), SI + CMPL DI, $0x08 + JLE repeat_two_match_repeat_encodeBlockAsm8B + CMPL DI, $0x0c + JGE cant_repeat_two_offset_match_repeat_encodeBlockAsm8B + +cant_repeat_two_offset_match_repeat_encodeBlockAsm8B: + CMPL SI, $0x00000104 + JLT repeat_three_match_repeat_encodeBlockAsm8B + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_three_match_repeat_encodeBlockAsm8B: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_two_match_repeat_encodeBlockAsm8B: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_as_copy_encodeBlockAsm8B: + // emitCopy +two_byte_offset_repeat_as_copy_encodeBlockAsm8B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeBlockAsm8B + CMPL DI, $0x00000800 + JAE long_offset_short_repeat_as_copy_encodeBlockAsm8B + MOVL $0x00000001, R8 + LEAL 16(R8), R8 + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, R8 + MOVB R8, (AX) + ADDQ $0x02, AX + SUBL $0x08, SI + + // emitRepeat + LEAL -4(SI), SI + JMP cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b + MOVL SI, DI + LEAL -4(SI), SI + CMPL DI, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b + CMPL DI, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_three_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_two_repeat_as_copy_encodeBlockAsm8B_emit_copy_short_2b: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + +long_offset_short_repeat_as_copy_encodeBlockAsm8B: + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + + // emitRepeat + MOVL SI, DI + LEAL -4(SI), SI + CMPL DI, $0x08 + JLE repeat_two_repeat_as_copy_encodeBlockAsm8B_emit_copy_short + CMPL DI, $0x0c + JGE cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm8B_emit_copy_short + +cant_repeat_two_offset_repeat_as_copy_encodeBlockAsm8B_emit_copy_short: + CMPL SI, $0x00000104 + JLT repeat_three_repeat_as_copy_encodeBlockAsm8B_emit_copy_short + LEAL -256(SI), SI + MOVW $0x0019, (AX) + MOVW SI, 2(AX) + ADDQ $0x04, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_three_repeat_as_copy_encodeBlockAsm8B_emit_copy_short: + LEAL -4(SI), SI + MOVW $0x0015, (AX) + MOVB SI, 2(AX) + ADDQ $0x03, AX + JMP repeat_end_emit_encodeBlockAsm8B + +repeat_two_repeat_as_copy_encodeBlockAsm8B_emit_copy_short: + SHLL $0x02, SI + ORL $0x01, SI + MOVW SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + XORQ R8, R8 + LEAL 1(R8)(SI*4), SI + MOVB DI, 1(AX) + SARL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + JMP two_byte_offset_repeat_as_copy_encodeBlockAsm8B + +two_byte_offset_short_repeat_as_copy_encodeBlockAsm8B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeBlockAsm8B + +emit_copy_three_repeat_as_copy_encodeBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeBlockAsm8B: + MOVL CX, 12(SP) + JMP search_loop_encodeBlockAsm8B + +no_repeat_found_encodeBlockAsm8B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBlockAsm8B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeBlockAsm8B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeBlockAsm8B + +candidate3_match_encodeBlockAsm8B: + ADDL $0x02, CX + JMP candidate_match_encodeBlockAsm8B + +candidate2_match_encodeBlockAsm8B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeBlockAsm8B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBlockAsm8B + +match_extend_back_loop_encodeBlockAsm8B: + CMPL CX, DI + JLE match_extend_back_end_encodeBlockAsm8B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBlockAsm8B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBlockAsm8B + JMP match_extend_back_loop_encodeBlockAsm8B + +match_extend_back_end_encodeBlockAsm8B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBlockAsm8B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeBlockAsm8B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeBlockAsm8B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeBlockAsm8B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBlockAsm8B + +two_bytes_match_emit_encodeBlockAsm8B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeBlockAsm8B + JMP memmove_long_match_emit_encodeBlockAsm8B + +one_byte_match_emit_encodeBlockAsm8B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBlockAsm8B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeBlockAsm8B + +emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm8B + +emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBlockAsm8B + +emit_lit_memmove_match_emit_encodeBlockAsm8B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBlockAsm8B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeBlockAsm8B + +memmove_long_match_emit_encodeBlockAsm8B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeBlockAsm8Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeBlockAsm8B: +match_nolit_loop_encodeBlockAsm8B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeBlockAsm8B + +matchlen_loopback_match_nolit_encodeBlockAsm8B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeBlockAsm8B + +matchlen_loop_match_nolit_encodeBlockAsm8B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeBlockAsm8B + JZ match_nolit_end_encodeBlockAsm8B + +matchlen_match4_match_nolit_encodeBlockAsm8B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeBlockAsm8B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeBlockAsm8B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeBlockAsm8B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeBlockAsm8B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeBlockAsm8B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeBlockAsm8B: + CMPL DI, $0x01 + JL match_nolit_end_encodeBlockAsm8B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeBlockAsm8B + LEAL 1(R10), R10 + +match_nolit_end_encodeBlockAsm8B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBlockAsm8B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBlockAsm8B + CMPL SI, $0x00000800 + JAE long_offset_short_match_nolit_encodeBlockAsm8B + MOVL $0x00000001, DI + LEAL 16(DI), DI + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, DI + MOVB DI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R10 + + // emitRepeat + LEAL -4(R10), R10 + JMP cant_repeat_two_offset_match_nolit_encodeBlockAsm8B_emit_copy_short_2b + MOVL R10, SI + LEAL -4(R10), R10 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm8B_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm8B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBlockAsm8B_emit_copy_short_2b: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm8B_emit_copy_short_2b + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +repeat_three_match_nolit_encodeBlockAsm8B_emit_copy_short_2b: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +repeat_two_match_nolit_encodeBlockAsm8B_emit_copy_short_2b: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +long_offset_short_match_nolit_encodeBlockAsm8B: + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + + // emitRepeat + MOVL R10, SI + LEAL -4(R10), R10 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBlockAsm8B_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBlockAsm8B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBlockAsm8B_emit_copy_short: + CMPL R10, $0x00000104 + JLT repeat_three_match_nolit_encodeBlockAsm8B_emit_copy_short + LEAL -256(R10), R10 + MOVW $0x0019, (AX) + MOVW R10, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +repeat_three_match_nolit_encodeBlockAsm8B_emit_copy_short: + LEAL -4(R10), R10 + MOVW $0x0015, (AX) + MOVB R10, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +repeat_two_match_nolit_encodeBlockAsm8B_emit_copy_short: + SHLL $0x02, R10 + ORL $0x01, R10 + MOVW R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + XORQ DI, DI + LEAL 1(DI)(R10*4), R10 + MOVB SI, 1(AX) + SARL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + JMP two_byte_offset_match_nolit_encodeBlockAsm8B + +two_byte_offset_short_match_nolit_encodeBlockAsm8B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBlockAsm8B + +emit_copy_three_match_nolit_encodeBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeBlockAsm8B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBlockAsm8B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBlockAsm8B: + MOVQ $0x9e3779b1, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x20, R8 + IMULQ R9, R8 + SHRQ $0x38, R8 + SHLQ $0x20, SI + IMULQ R9, SI + SHRQ $0x38, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeBlockAsm8B + INCL CX + JMP search_loop_encodeBlockAsm8B + +emit_remainder_encodeBlockAsm8B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBlockAsm8B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBlockAsm8B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBlockAsm8B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBlockAsm8B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBlockAsm8B + +two_bytes_emit_remainder_encodeBlockAsm8B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBlockAsm8B + JMP memmove_long_emit_remainder_encodeBlockAsm8B + +one_byte_emit_remainder_encodeBlockAsm8B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBlockAsm8B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBlockAsm8B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBlockAsm8B + +memmove_long_emit_remainder_encodeBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm8Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBlockAsm8B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBetterBlockAsm(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBetterBlockAsm(SB), $327704-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000a00, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBetterBlockAsm: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBetterBlockAsm + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -6(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBetterBlockAsm: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x07, SI + CMPL SI, $0x63 + JLE check_maxskip_ok_encodeBetterBlockAsm + LEAL 100(CX), SI + JMP check_maxskip_cont_encodeBetterBlockAsm + +check_maxskip_ok_encodeBetterBlockAsm: + LEAL 1(CX)(SI*1), SI + +check_maxskip_cont_encodeBetterBlockAsm: + CMPL SI, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x00cf1bbcdcbfa563, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 262168(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 262168(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeBetterBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm + +candidateS_match_encodeBetterBlockAsm: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm + DECL CX + MOVL R8, SI + +candidate_match_encodeBetterBlockAsm: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBetterBlockAsm + +match_extend_back_loop_encodeBetterBlockAsm: + CMPL CX, DI + JLE match_extend_back_end_encodeBetterBlockAsm + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBetterBlockAsm + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBetterBlockAsm + JMP match_extend_back_loop_encodeBetterBlockAsm + +match_extend_back_end_encodeBetterBlockAsm: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 5(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBetterBlockAsm: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeBetterBlockAsm + +matchlen_loopback_match_nolit_encodeBetterBlockAsm: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeBetterBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeBetterBlockAsm + +matchlen_loop_match_nolit_encodeBetterBlockAsm: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeBetterBlockAsm + JZ match_nolit_end_encodeBetterBlockAsm + +matchlen_match4_match_nolit_encodeBetterBlockAsm: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeBetterBlockAsm + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeBetterBlockAsm + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeBetterBlockAsm: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeBetterBlockAsm + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeBetterBlockAsm + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeBetterBlockAsm: + CMPL R8, $0x01 + JL match_nolit_end_encodeBetterBlockAsm + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeBetterBlockAsm + LEAL 1(R12), R12 + +match_nolit_end_encodeBetterBlockAsm: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL 16(SP), R8 + JEQ match_is_repeat_encodeBetterBlockAsm + CMPL R12, $0x01 + JG match_length_ok_encodeBetterBlockAsm + CMPL R8, $0x0000ffff + JLE match_length_ok_encodeBetterBlockAsm + MOVL 20(SP), CX + INCL CX + JMP search_loop_encodeBetterBlockAsm + +match_length_ok_encodeBetterBlockAsm: + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeBetterBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeBetterBlockAsm + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeBetterBlockAsm + CMPL SI, $0x00010000 + JLT three_bytes_match_emit_encodeBetterBlockAsm + CMPL SI, $0x01000000 + JLT four_bytes_match_emit_encodeBetterBlockAsm + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm + +four_bytes_match_emit_encodeBetterBlockAsm: + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm + +three_bytes_match_emit_encodeBetterBlockAsm: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm + +two_bytes_match_emit_encodeBetterBlockAsm: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeBetterBlockAsm + JMP memmove_long_match_emit_encodeBetterBlockAsm + +one_byte_match_emit_encodeBetterBlockAsm: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_encodeBetterBlockAsm_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBetterBlockAsm: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeBetterBlockAsm + +memmove_long_match_emit_encodeBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeBetterBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsmlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeBetterBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeBetterBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeBetterBlockAsm: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy + CMPL R8, $0x00010000 + JL two_byte_offset_match_nolit_encodeBetterBlockAsm + +four_bytes_loop_back_match_nolit_encodeBetterBlockAsm: + CMPL R12, $0x40 + JLE four_bytes_remain_match_nolit_encodeBetterBlockAsm + MOVB $0xff, (AX) + MOVL R8, 1(AX) + LEAL -64(R12), R12 + ADDQ $0x05, AX + CMPL R12, $0x04 + JL four_bytes_remain_match_nolit_encodeBetterBlockAsm + + // emitRepeat +emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy: + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy + CMPL R12, $0x0100ffff + JLT repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy + LEAL -16842747(R12), R12 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy + +repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy: + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + JMP four_bytes_loop_back_match_nolit_encodeBetterBlockAsm + +four_bytes_remain_match_nolit_encodeBetterBlockAsm: + TESTL R12, R12 + JZ match_nolit_emitcopy_end_encodeBetterBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVL R8, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +two_byte_offset_match_nolit_encodeBetterBlockAsm: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBetterBlockAsm + CMPL R8, $0x00000800 + JAE long_offset_short_match_nolit_encodeBetterBlockAsm + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB R8, 1(AX) + MOVL R8, R9 + SHRL $0x08, R9 + SHLL $0x05, R9 + ORL R9, SI + MOVB SI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R12 + + // emitRepeat + LEAL -4(R12), R12 + JMP cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + +emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + CMPL R12, $0x0100ffff + JLT repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + LEAL -16842747(R12), R12 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b + +repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short_2b: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +long_offset_short_match_nolit_encodeBetterBlockAsm: + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + + // emitRepeat +emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy_short: + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy_short + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy_short + CMPL R12, $0x0100ffff + JLT repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy_short + LEAL -16842747(R12), R12 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_encodeBetterBlockAsm_emit_copy_short + +repeat_five_match_nolit_encodeBetterBlockAsm_emit_copy_short: + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_four_match_nolit_encodeBetterBlockAsm_emit_copy_short: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_three_match_nolit_encodeBetterBlockAsm_emit_copy_short: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_match_nolit_encodeBetterBlockAsm_emit_copy_short: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_offset_match_nolit_encodeBetterBlockAsm_emit_copy_short: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + JMP two_byte_offset_match_nolit_encodeBetterBlockAsm + +two_byte_offset_short_match_nolit_encodeBetterBlockAsm: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +emit_copy_three_match_nolit_encodeBetterBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +match_is_repeat_encodeBetterBlockAsm: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_repeat_encodeBetterBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_repeat_encodeBetterBlockAsm + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_repeat_encodeBetterBlockAsm + CMPL SI, $0x00010000 + JLT three_bytes_match_emit_repeat_encodeBetterBlockAsm + CMPL SI, $0x01000000 + JLT four_bytes_match_emit_repeat_encodeBetterBlockAsm + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm + +four_bytes_match_emit_repeat_encodeBetterBlockAsm: + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm + +three_bytes_match_emit_repeat_encodeBetterBlockAsm: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm + +two_bytes_match_emit_repeat_encodeBetterBlockAsm: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_repeat_encodeBetterBlockAsm + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm + +one_byte_match_emit_repeat_encodeBetterBlockAsm: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_repeat_encodeBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_33through64 + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm: + MOVQ SI, AX + JMP emit_literal_done_match_emit_repeat_encodeBetterBlockAsm + +memmove_long_match_emit_repeat_encodeBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsmlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_repeat_encodeBetterBlockAsm: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitRepeat +emit_repeat_again_match_nolit_repeat_encodeBetterBlockAsm: + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_repeat_encodeBetterBlockAsm + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm + +cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_repeat_encodeBetterBlockAsm + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_repeat_encodeBetterBlockAsm + CMPL R12, $0x0100ffff + JLT repeat_five_match_nolit_repeat_encodeBetterBlockAsm + LEAL -16842747(R12), R12 + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + JMP emit_repeat_again_match_nolit_repeat_encodeBetterBlockAsm + +repeat_five_match_nolit_repeat_encodeBetterBlockAsm: + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_four_match_nolit_repeat_encodeBetterBlockAsm: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_three_match_nolit_repeat_encodeBetterBlockAsm: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_match_nolit_repeat_encodeBetterBlockAsm: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm + +repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + +match_nolit_emitcopy_end_encodeBetterBlockAsm: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBetterBlockAsm: + MOVQ $0x00cf1bbcdcbfa563, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x32, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 262168(SP)(R11*4) + MOVL R15, 262168(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 262168(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeBetterBlockAsm + +emit_remainder_encodeBetterBlockAsm: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 5(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBetterBlockAsm: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBetterBlockAsm + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBetterBlockAsm + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBetterBlockAsm + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeBetterBlockAsm + CMPL DX, $0x01000000 + JLT four_bytes_emit_remainder_encodeBetterBlockAsm + MOVB $0xfc, (AX) + MOVL DX, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm + +four_bytes_emit_remainder_encodeBetterBlockAsm: + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm + +three_bytes_emit_remainder_encodeBetterBlockAsm: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm + +two_bytes_emit_remainder_encodeBetterBlockAsm: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBetterBlockAsm + JMP memmove_long_emit_remainder_encodeBetterBlockAsm + +one_byte_emit_remainder_encodeBetterBlockAsm: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBetterBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBetterBlockAsm: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBetterBlockAsm + +memmove_long_emit_remainder_encodeBetterBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsmlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBetterBlockAsm: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBetterBlockAsm4MB(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBetterBlockAsm4MB(SB), $327704-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000a00, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBetterBlockAsm4MB: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBetterBlockAsm4MB + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -6(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBetterBlockAsm4MB: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x07, SI + CMPL SI, $0x63 + JLE check_maxskip_ok_encodeBetterBlockAsm4MB + LEAL 100(CX), SI + JMP check_maxskip_cont_encodeBetterBlockAsm4MB + +check_maxskip_ok_encodeBetterBlockAsm4MB: + LEAL 1(CX)(SI*1), SI + +check_maxskip_cont_encodeBetterBlockAsm4MB: + CMPL SI, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm4MB + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x00cf1bbcdcbfa563, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 262168(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 262168(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm4MB + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeBetterBlockAsm4MB + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm4MB + +candidateS_match_encodeBetterBlockAsm4MB: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm4MB + DECL CX + MOVL R8, SI + +candidate_match_encodeBetterBlockAsm4MB: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBetterBlockAsm4MB + +match_extend_back_loop_encodeBetterBlockAsm4MB: + CMPL CX, DI + JLE match_extend_back_end_encodeBetterBlockAsm4MB + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBetterBlockAsm4MB + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBetterBlockAsm4MB + JMP match_extend_back_loop_encodeBetterBlockAsm4MB + +match_extend_back_end_encodeBetterBlockAsm4MB: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 4(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBetterBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBetterBlockAsm4MB: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeBetterBlockAsm4MB + +matchlen_loopback_match_nolit_encodeBetterBlockAsm4MB: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeBetterBlockAsm4MB + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeBetterBlockAsm4MB + +matchlen_loop_match_nolit_encodeBetterBlockAsm4MB: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeBetterBlockAsm4MB + JZ match_nolit_end_encodeBetterBlockAsm4MB + +matchlen_match4_match_nolit_encodeBetterBlockAsm4MB: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeBetterBlockAsm4MB + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeBetterBlockAsm4MB + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeBetterBlockAsm4MB: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeBetterBlockAsm4MB + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeBetterBlockAsm4MB + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeBetterBlockAsm4MB: + CMPL R8, $0x01 + JL match_nolit_end_encodeBetterBlockAsm4MB + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeBetterBlockAsm4MB + LEAL 1(R12), R12 + +match_nolit_end_encodeBetterBlockAsm4MB: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL 16(SP), R8 + JEQ match_is_repeat_encodeBetterBlockAsm4MB + CMPL R12, $0x01 + JG match_length_ok_encodeBetterBlockAsm4MB + CMPL R8, $0x0000ffff + JLE match_length_ok_encodeBetterBlockAsm4MB + MOVL 20(SP), CX + INCL CX + JMP search_loop_encodeBetterBlockAsm4MB + +match_length_ok_encodeBetterBlockAsm4MB: + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeBetterBlockAsm4MB + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeBetterBlockAsm4MB + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeBetterBlockAsm4MB + CMPL SI, $0x00010000 + JLT three_bytes_match_emit_encodeBetterBlockAsm4MB + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm4MB + +three_bytes_match_emit_encodeBetterBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm4MB + +two_bytes_match_emit_encodeBetterBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeBetterBlockAsm4MB + JMP memmove_long_match_emit_encodeBetterBlockAsm4MB + +one_byte_match_emit_encodeBetterBlockAsm4MB: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBetterBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_encodeBetterBlockAsm4MB_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBetterBlockAsm4MB: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeBetterBlockAsm4MB + +memmove_long_match_emit_encodeBetterBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm4MBlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeBetterBlockAsm4MB: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy + CMPL R8, $0x00010000 + JL two_byte_offset_match_nolit_encodeBetterBlockAsm4MB + +four_bytes_loop_back_match_nolit_encodeBetterBlockAsm4MB: + CMPL R12, $0x40 + JLE four_bytes_remain_match_nolit_encodeBetterBlockAsm4MB + MOVB $0xff, (AX) + MOVL R8, 1(AX) + LEAL -64(R12), R12 + ADDQ $0x05, AX + CMPL R12, $0x04 + JL four_bytes_remain_match_nolit_encodeBetterBlockAsm4MB + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + JMP four_bytes_loop_back_match_nolit_encodeBetterBlockAsm4MB + +four_bytes_remain_match_nolit_encodeBetterBlockAsm4MB: + TESTL R12, R12 + JZ match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + MOVB $0x03, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVL R8, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +two_byte_offset_match_nolit_encodeBetterBlockAsm4MB: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBetterBlockAsm4MB + CMPL R8, $0x00000800 + JAE long_offset_short_match_nolit_encodeBetterBlockAsm4MB + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, SI + MOVB SI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R12 + + // emitRepeat + LEAL -4(R12), R12 + JMP cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short_2b: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +long_offset_short_match_nolit_encodeBetterBlockAsm4MB: + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_four_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_three_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_offset_match_nolit_encodeBetterBlockAsm4MB_emit_copy_short: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + JMP two_byte_offset_match_nolit_encodeBetterBlockAsm4MB + +two_byte_offset_short_match_nolit_encodeBetterBlockAsm4MB: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm4MB + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm4MB + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +emit_copy_three_match_nolit_encodeBetterBlockAsm4MB: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +match_is_repeat_encodeBetterBlockAsm4MB: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_repeat_encodeBetterBlockAsm4MB + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_repeat_encodeBetterBlockAsm4MB + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_repeat_encodeBetterBlockAsm4MB + CMPL SI, $0x00010000 + JLT three_bytes_match_emit_repeat_encodeBetterBlockAsm4MB + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm4MB + +three_bytes_match_emit_repeat_encodeBetterBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm4MB + +two_bytes_match_emit_repeat_encodeBetterBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_repeat_encodeBetterBlockAsm4MB + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm4MB + +one_byte_match_emit_repeat_encodeBetterBlockAsm4MB: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_repeat_encodeBetterBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm4MB + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm4MB_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm4MB: + MOVQ SI, AX + JMP emit_literal_done_match_emit_repeat_encodeBetterBlockAsm4MB + +memmove_long_match_emit_repeat_encodeBetterBlockAsm4MB: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm4MBlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_repeat_encodeBetterBlockAsm4MB: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_repeat_encodeBetterBlockAsm4MB + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm4MB + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm4MB + +cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm4MB: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_repeat_encodeBetterBlockAsm4MB + CMPL R12, $0x00010100 + JLT repeat_four_match_nolit_repeat_encodeBetterBlockAsm4MB + LEAL -65536(R12), R12 + MOVL R12, R8 + MOVW $0x001d, (AX) + MOVW R12, 2(AX) + SARL $0x10, R8 + MOVB R8, 4(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_four_match_nolit_repeat_encodeBetterBlockAsm4MB: + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_three_match_nolit_repeat_encodeBetterBlockAsm4MB: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_match_nolit_repeat_encodeBetterBlockAsm4MB: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm4MB + +repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm4MB: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + +match_nolit_emitcopy_end_encodeBetterBlockAsm4MB: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm4MB + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBetterBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBetterBlockAsm4MB: + MOVQ $0x00cf1bbcdcbfa563, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x32, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 262168(SP)(R11*4) + MOVL R15, 262168(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 262168(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeBetterBlockAsm4MB + +emit_remainder_encodeBetterBlockAsm4MB: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 4(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBetterBlockAsm4MB + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBetterBlockAsm4MB: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBetterBlockAsm4MB + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBetterBlockAsm4MB + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBetterBlockAsm4MB + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeBetterBlockAsm4MB + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm4MB + +three_bytes_emit_remainder_encodeBetterBlockAsm4MB: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm4MB + +two_bytes_emit_remainder_encodeBetterBlockAsm4MB: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBetterBlockAsm4MB + JMP memmove_long_emit_remainder_encodeBetterBlockAsm4MB + +one_byte_emit_remainder_encodeBetterBlockAsm4MB: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBetterBlockAsm4MB: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm4MB_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBetterBlockAsm4MB: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBetterBlockAsm4MB + +memmove_long_emit_remainder_encodeBetterBlockAsm4MB: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm4MBlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm4MBlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm4MBlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBetterBlockAsm4MB: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBetterBlockAsm12B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBetterBlockAsm12B(SB), $81944-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000280, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBetterBlockAsm12B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBetterBlockAsm12B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -6(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBetterBlockAsm12B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm12B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x34, R11 + MOVL 24(SP)(R10*4), SI + MOVL 65560(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 65560(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm12B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeBetterBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm12B + +candidateS_match_encodeBetterBlockAsm12B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm12B + DECL CX + MOVL R8, SI + +candidate_match_encodeBetterBlockAsm12B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBetterBlockAsm12B + +match_extend_back_loop_encodeBetterBlockAsm12B: + CMPL CX, DI + JLE match_extend_back_end_encodeBetterBlockAsm12B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBetterBlockAsm12B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBetterBlockAsm12B + JMP match_extend_back_loop_encodeBetterBlockAsm12B + +match_extend_back_end_encodeBetterBlockAsm12B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBetterBlockAsm12B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeBetterBlockAsm12B + +matchlen_loopback_match_nolit_encodeBetterBlockAsm12B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeBetterBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeBetterBlockAsm12B + +matchlen_loop_match_nolit_encodeBetterBlockAsm12B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeBetterBlockAsm12B + JZ match_nolit_end_encodeBetterBlockAsm12B + +matchlen_match4_match_nolit_encodeBetterBlockAsm12B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeBetterBlockAsm12B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeBetterBlockAsm12B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeBetterBlockAsm12B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeBetterBlockAsm12B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeBetterBlockAsm12B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeBetterBlockAsm12B: + CMPL R8, $0x01 + JL match_nolit_end_encodeBetterBlockAsm12B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeBetterBlockAsm12B + LEAL 1(R12), R12 + +match_nolit_end_encodeBetterBlockAsm12B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL 16(SP), R8 + JEQ match_is_repeat_encodeBetterBlockAsm12B + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeBetterBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeBetterBlockAsm12B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeBetterBlockAsm12B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm12B + +two_bytes_match_emit_encodeBetterBlockAsm12B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeBetterBlockAsm12B + JMP memmove_long_match_emit_encodeBetterBlockAsm12B + +one_byte_match_emit_encodeBetterBlockAsm12B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm12B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBetterBlockAsm12B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeBetterBlockAsm12B + +memmove_long_match_emit_encodeBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm12Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeBetterBlockAsm12B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBetterBlockAsm12B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBetterBlockAsm12B + CMPL R8, $0x00000800 + JAE long_offset_short_match_nolit_encodeBetterBlockAsm12B + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, SI + MOVB SI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R12 + + // emitRepeat + LEAL -4(R12), R12 + JMP cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_three_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short_2b: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +long_offset_short_match_nolit_encodeBetterBlockAsm12B: + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm12B_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm12B_emit_copy_short + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_three_match_nolit_encodeBetterBlockAsm12B_emit_copy_short: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_match_nolit_encodeBetterBlockAsm12B_emit_copy_short: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_offset_match_nolit_encodeBetterBlockAsm12B_emit_copy_short: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + JMP two_byte_offset_match_nolit_encodeBetterBlockAsm12B + +two_byte_offset_short_match_nolit_encodeBetterBlockAsm12B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm12B + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +emit_copy_three_match_nolit_encodeBetterBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +match_is_repeat_encodeBetterBlockAsm12B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_repeat_encodeBetterBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_repeat_encodeBetterBlockAsm12B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_repeat_encodeBetterBlockAsm12B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm12B + +two_bytes_match_emit_repeat_encodeBetterBlockAsm12B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_repeat_encodeBetterBlockAsm12B + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm12B + +one_byte_match_emit_repeat_encodeBetterBlockAsm12B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_repeat_encodeBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm12B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm12B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm12B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_repeat_encodeBetterBlockAsm12B + +memmove_long_match_emit_repeat_encodeBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm12Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_repeat_encodeBetterBlockAsm12B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_repeat_encodeBetterBlockAsm12B + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm12B + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm12B + +cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm12B: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_repeat_encodeBetterBlockAsm12B + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_three_match_nolit_repeat_encodeBetterBlockAsm12B: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_match_nolit_repeat_encodeBetterBlockAsm12B: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm12B + +repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm12B: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + +match_nolit_emitcopy_end_encodeBetterBlockAsm12B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm12B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBetterBlockAsm12B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x32, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x34, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x34, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 65560(SP)(R11*4) + MOVL R15, 65560(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x32, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x34, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x32, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 65560(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeBetterBlockAsm12B + +emit_remainder_encodeBetterBlockAsm12B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBetterBlockAsm12B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBetterBlockAsm12B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBetterBlockAsm12B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBetterBlockAsm12B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm12B + +two_bytes_emit_remainder_encodeBetterBlockAsm12B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBetterBlockAsm12B + JMP memmove_long_emit_remainder_encodeBetterBlockAsm12B + +one_byte_emit_remainder_encodeBetterBlockAsm12B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBetterBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm12B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBetterBlockAsm12B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBetterBlockAsm12B + +memmove_long_emit_remainder_encodeBetterBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm12Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBetterBlockAsm12B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBetterBlockAsm10B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBetterBlockAsm10B(SB), $20504-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x000000a0, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBetterBlockAsm10B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBetterBlockAsm10B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -6(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBetterBlockAsm10B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm10B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x36, R11 + MOVL 24(SP)(R10*4), SI + MOVL 16408(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 16408(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm10B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeBetterBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm10B + +candidateS_match_encodeBetterBlockAsm10B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm10B + DECL CX + MOVL R8, SI + +candidate_match_encodeBetterBlockAsm10B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBetterBlockAsm10B + +match_extend_back_loop_encodeBetterBlockAsm10B: + CMPL CX, DI + JLE match_extend_back_end_encodeBetterBlockAsm10B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBetterBlockAsm10B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBetterBlockAsm10B + JMP match_extend_back_loop_encodeBetterBlockAsm10B + +match_extend_back_end_encodeBetterBlockAsm10B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBetterBlockAsm10B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeBetterBlockAsm10B + +matchlen_loopback_match_nolit_encodeBetterBlockAsm10B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeBetterBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeBetterBlockAsm10B + +matchlen_loop_match_nolit_encodeBetterBlockAsm10B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeBetterBlockAsm10B + JZ match_nolit_end_encodeBetterBlockAsm10B + +matchlen_match4_match_nolit_encodeBetterBlockAsm10B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeBetterBlockAsm10B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeBetterBlockAsm10B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeBetterBlockAsm10B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeBetterBlockAsm10B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeBetterBlockAsm10B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeBetterBlockAsm10B: + CMPL R8, $0x01 + JL match_nolit_end_encodeBetterBlockAsm10B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeBetterBlockAsm10B + LEAL 1(R12), R12 + +match_nolit_end_encodeBetterBlockAsm10B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL 16(SP), R8 + JEQ match_is_repeat_encodeBetterBlockAsm10B + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeBetterBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeBetterBlockAsm10B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeBetterBlockAsm10B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm10B + +two_bytes_match_emit_encodeBetterBlockAsm10B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeBetterBlockAsm10B + JMP memmove_long_match_emit_encodeBetterBlockAsm10B + +one_byte_match_emit_encodeBetterBlockAsm10B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm10B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBetterBlockAsm10B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeBetterBlockAsm10B + +memmove_long_match_emit_encodeBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm10Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeBetterBlockAsm10B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBetterBlockAsm10B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBetterBlockAsm10B + CMPL R8, $0x00000800 + JAE long_offset_short_match_nolit_encodeBetterBlockAsm10B + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, SI + MOVB SI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R12 + + // emitRepeat + LEAL -4(R12), R12 + JMP cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_three_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short_2b: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +long_offset_short_match_nolit_encodeBetterBlockAsm10B: + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm10B_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm10B_emit_copy_short + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_three_match_nolit_encodeBetterBlockAsm10B_emit_copy_short: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_match_nolit_encodeBetterBlockAsm10B_emit_copy_short: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_offset_match_nolit_encodeBetterBlockAsm10B_emit_copy_short: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + JMP two_byte_offset_match_nolit_encodeBetterBlockAsm10B + +two_byte_offset_short_match_nolit_encodeBetterBlockAsm10B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm10B + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +emit_copy_three_match_nolit_encodeBetterBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +match_is_repeat_encodeBetterBlockAsm10B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_repeat_encodeBetterBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_repeat_encodeBetterBlockAsm10B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_repeat_encodeBetterBlockAsm10B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm10B + +two_bytes_match_emit_repeat_encodeBetterBlockAsm10B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_repeat_encodeBetterBlockAsm10B + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm10B + +one_byte_match_emit_repeat_encodeBetterBlockAsm10B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_repeat_encodeBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm10B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm10B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm10B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_repeat_encodeBetterBlockAsm10B + +memmove_long_match_emit_repeat_encodeBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm10Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_repeat_encodeBetterBlockAsm10B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_repeat_encodeBetterBlockAsm10B + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm10B + CMPL R8, $0x00000800 + JLT repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm10B + +cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm10B: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_repeat_encodeBetterBlockAsm10B + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_three_match_nolit_repeat_encodeBetterBlockAsm10B: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_match_nolit_repeat_encodeBetterBlockAsm10B: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm10B + +repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm10B: + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + +match_nolit_emitcopy_end_encodeBetterBlockAsm10B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm10B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBetterBlockAsm10B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x34, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x34, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x36, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x36, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 16408(SP)(R11*4) + MOVL R15, 16408(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x34, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x36, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x34, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 16408(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeBetterBlockAsm10B + +emit_remainder_encodeBetterBlockAsm10B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBetterBlockAsm10B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBetterBlockAsm10B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBetterBlockAsm10B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBetterBlockAsm10B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm10B + +two_bytes_emit_remainder_encodeBetterBlockAsm10B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBetterBlockAsm10B + JMP memmove_long_emit_remainder_encodeBetterBlockAsm10B + +one_byte_emit_remainder_encodeBetterBlockAsm10B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBetterBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm10B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBetterBlockAsm10B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBetterBlockAsm10B + +memmove_long_emit_remainder_encodeBetterBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm10Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBetterBlockAsm10B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeBetterBlockAsm8B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeBetterBlockAsm8B(SB), $5144-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000028, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeBetterBlockAsm8B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeBetterBlockAsm8B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -6(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeBetterBlockAsm8B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x04, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm8B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x38, R11 + MOVL 24(SP)(R10*4), SI + MOVL 4120(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 4120(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm8B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeBetterBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm8B + +candidateS_match_encodeBetterBlockAsm8B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeBetterBlockAsm8B + DECL CX + MOVL R8, SI + +candidate_match_encodeBetterBlockAsm8B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeBetterBlockAsm8B + +match_extend_back_loop_encodeBetterBlockAsm8B: + CMPL CX, DI + JLE match_extend_back_end_encodeBetterBlockAsm8B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeBetterBlockAsm8B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeBetterBlockAsm8B + JMP match_extend_back_loop_encodeBetterBlockAsm8B + +match_extend_back_end_encodeBetterBlockAsm8B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeBetterBlockAsm8B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeBetterBlockAsm8B + +matchlen_loopback_match_nolit_encodeBetterBlockAsm8B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeBetterBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeBetterBlockAsm8B + +matchlen_loop_match_nolit_encodeBetterBlockAsm8B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeBetterBlockAsm8B + JZ match_nolit_end_encodeBetterBlockAsm8B + +matchlen_match4_match_nolit_encodeBetterBlockAsm8B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeBetterBlockAsm8B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeBetterBlockAsm8B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeBetterBlockAsm8B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeBetterBlockAsm8B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeBetterBlockAsm8B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeBetterBlockAsm8B: + CMPL R8, $0x01 + JL match_nolit_end_encodeBetterBlockAsm8B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeBetterBlockAsm8B + LEAL 1(R12), R12 + +match_nolit_end_encodeBetterBlockAsm8B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL 16(SP), R8 + JEQ match_is_repeat_encodeBetterBlockAsm8B + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeBetterBlockAsm8B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeBetterBlockAsm8B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeBetterBlockAsm8B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeBetterBlockAsm8B + +two_bytes_match_emit_encodeBetterBlockAsm8B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeBetterBlockAsm8B + JMP memmove_long_match_emit_encodeBetterBlockAsm8B + +one_byte_match_emit_encodeBetterBlockAsm8B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeBetterBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x04 + JLE emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_4 + CMPQ R9, $0x08 + JB emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_4through7 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_4: + MOVL (R10), R11 + MOVL R11, (AX) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_4through7: + MOVL (R10), R11 + MOVL -4(R10)(R9*1), R10 + MOVL R11, (AX) + MOVL R10, -4(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeBetterBlockAsm8B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeBetterBlockAsm8B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeBetterBlockAsm8B + +memmove_long_match_emit_encodeBetterBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm8Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeBetterBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeBetterBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeBetterBlockAsm8B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeBetterBlockAsm8B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeBetterBlockAsm8B + CMPL R8, $0x00000800 + JAE long_offset_short_match_nolit_encodeBetterBlockAsm8B + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, SI + MOVB SI, (AX) + ADDQ $0x02, AX + SUBL $0x08, R12 + + // emitRepeat + LEAL -4(R12), R12 + JMP cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_three_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_two_match_nolit_encodeBetterBlockAsm8B_emit_copy_short_2b: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +long_offset_short_match_nolit_encodeBetterBlockAsm8B: + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_encodeBetterBlockAsm8B_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm8B_emit_copy_short + +cant_repeat_two_offset_match_nolit_encodeBetterBlockAsm8B_emit_copy_short: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_encodeBetterBlockAsm8B_emit_copy_short + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_three_match_nolit_encodeBetterBlockAsm8B_emit_copy_short: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_two_match_nolit_encodeBetterBlockAsm8B_emit_copy_short: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + JMP two_byte_offset_match_nolit_encodeBetterBlockAsm8B + +two_byte_offset_short_match_nolit_encodeBetterBlockAsm8B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeBetterBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +emit_copy_three_match_nolit_encodeBetterBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +match_is_repeat_encodeBetterBlockAsm8B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_repeat_encodeBetterBlockAsm8B + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_repeat_encodeBetterBlockAsm8B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_repeat_encodeBetterBlockAsm8B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm8B + +two_bytes_match_emit_repeat_encodeBetterBlockAsm8B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_repeat_encodeBetterBlockAsm8B + JMP memmove_long_match_emit_repeat_encodeBetterBlockAsm8B + +one_byte_match_emit_repeat_encodeBetterBlockAsm8B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_repeat_encodeBetterBlockAsm8B: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x04 + JLE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_4 + CMPQ R8, $0x08 + JB emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_4through7 + CMPQ R8, $0x10 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_4: + MOVL (R9), R10 + MOVL R10, (AX) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_4through7: + MOVL (R9), R10 + MOVL -4(R9)(R8*1), R9 + MOVL R10, (AX) + MOVL R9, -4(AX)(R8*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm8B + +emit_lit_memmove_match_emit_repeat_encodeBetterBlockAsm8B_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_match_emit_repeat_encodeBetterBlockAsm8B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_repeat_encodeBetterBlockAsm8B + +memmove_long_match_emit_repeat_encodeBetterBlockAsm8B: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R13 + SUBQ R10, R13 + DECQ R11 + JA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(R9)(R13*1), R10 + LEAQ -32(AX)(R13*1), R14 + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm8Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R14) + MOVOA X5, 16(R14) + ADDQ $0x20, R14 + ADDQ $0x20, R10 + ADDQ $0x20, R13 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(R9)(R13*1), X4 + MOVOU -16(R9)(R13*1), X5 + MOVOA X4, -32(AX)(R13*1) + MOVOA X5, -16(AX)(R13*1) + ADDQ $0x20, R13 + CMPQ R8, R13 + JAE emit_lit_memmove_long_match_emit_repeat_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_match_emit_repeat_encodeBetterBlockAsm8B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitRepeat + MOVL R12, SI + LEAL -4(R12), R12 + CMPL SI, $0x08 + JLE repeat_two_match_nolit_repeat_encodeBetterBlockAsm8B + CMPL SI, $0x0c + JGE cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm8B + +cant_repeat_two_offset_match_nolit_repeat_encodeBetterBlockAsm8B: + CMPL R12, $0x00000104 + JLT repeat_three_match_nolit_repeat_encodeBetterBlockAsm8B + LEAL -256(R12), R12 + MOVW $0x0019, (AX) + MOVW R12, 2(AX) + ADDQ $0x04, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_three_match_nolit_repeat_encodeBetterBlockAsm8B: + LEAL -4(R12), R12 + MOVW $0x0015, (AX) + MOVB R12, 2(AX) + ADDQ $0x03, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + +repeat_two_match_nolit_repeat_encodeBetterBlockAsm8B: + SHLL $0x02, R12 + ORL $0x01, R12 + MOVW R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeBetterBlockAsm8B + XORQ SI, SI + LEAL 1(SI)(R12*4), R12 + MOVB R8, 1(AX) + SARL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + +match_nolit_emitcopy_end_encodeBetterBlockAsm8B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeBetterBlockAsm8B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeBetterBlockAsm8B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x36, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x36, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x38, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x38, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 4120(SP)(R11*4) + MOVL R15, 4120(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x38, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x36, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 4120(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeBetterBlockAsm8B + +emit_remainder_encodeBetterBlockAsm8B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeBetterBlockAsm8B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeBetterBlockAsm8B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeBetterBlockAsm8B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeBetterBlockAsm8B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeBetterBlockAsm8B + +two_bytes_emit_remainder_encodeBetterBlockAsm8B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeBetterBlockAsm8B + JMP memmove_long_emit_remainder_encodeBetterBlockAsm8B + +one_byte_emit_remainder_encodeBetterBlockAsm8B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeBetterBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeBetterBlockAsm8B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeBetterBlockAsm8B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeBetterBlockAsm8B + +memmove_long_emit_remainder_encodeBetterBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm8Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeBetterBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeBetterBlockAsm8B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBlockAsm(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBlockAsm(SB), $65560-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000200, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBlockAsm: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBlockAsm + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBlockAsm: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R11 + IMULQ R9, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeSnappyBlockAsm + LEAL 1(CX), DI + MOVL 12(SP), SI + MOVL DI, R8 + SUBL 16(SP), R8 + JZ repeat_extend_back_end_encodeSnappyBlockAsm + +repeat_extend_back_loop_encodeSnappyBlockAsm: + CMPL DI, SI + JLE repeat_extend_back_end_encodeSnappyBlockAsm + MOVB -1(DX)(R8*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeSnappyBlockAsm + LEAL -1(DI), DI + DECL R8 + JNZ repeat_extend_back_loop_encodeSnappyBlockAsm + +repeat_extend_back_end_encodeSnappyBlockAsm: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeSnappyBlockAsm + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeSnappyBlockAsm + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeSnappyBlockAsm + CMPL SI, $0x00010000 + JLT three_bytes_repeat_emit_encodeSnappyBlockAsm + CMPL SI, $0x01000000 + JLT four_bytes_repeat_emit_encodeSnappyBlockAsm + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm + +four_bytes_repeat_emit_encodeSnappyBlockAsm: + MOVL SI, R10 + SHRL $0x10, R10 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R10, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm + +three_bytes_repeat_emit_encodeSnappyBlockAsm: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm + +two_bytes_repeat_emit_encodeSnappyBlockAsm: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeSnappyBlockAsm + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm + +one_byte_repeat_emit_encodeSnappyBlockAsm: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeSnappyBlockAsm: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_8 + CMPQ R8, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_8: + MOVQ (R9), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_repeat_emit_encodeSnappyBlockAsm: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeSnappyBlockAsm + +memmove_long_repeat_emit_encodeSnappyBlockAsm: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(R9)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsmlarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(R9)(R12*1), X4 + MOVOU -16(R9)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R8, R12 + JAE emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeSnappyBlockAsm: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R11, R11 + CMPL R8, $0x08 + JL matchlen_match4_repeat_extend_encodeSnappyBlockAsm + +matchlen_loopback_repeat_extend_encodeSnappyBlockAsm: + MOVQ (R9)(R11*1), R10 + XORQ (SI)(R11*1), R10 + TESTQ R10, R10 + JZ matchlen_loop_repeat_extend_encodeSnappyBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R10, R10 + +#else + BSFQ R10, R10 + +#endif + SARQ $0x03, R10 + LEAL (R11)(R10*1), R11 + JMP repeat_extend_forward_end_encodeSnappyBlockAsm + +matchlen_loop_repeat_extend_encodeSnappyBlockAsm: + LEAL -8(R8), R8 + LEAL 8(R11), R11 + CMPL R8, $0x08 + JGE matchlen_loopback_repeat_extend_encodeSnappyBlockAsm + JZ repeat_extend_forward_end_encodeSnappyBlockAsm + +matchlen_match4_repeat_extend_encodeSnappyBlockAsm: + CMPL R8, $0x04 + JL matchlen_match2_repeat_extend_encodeSnappyBlockAsm + MOVL (R9)(R11*1), R10 + CMPL (SI)(R11*1), R10 + JNE matchlen_match2_repeat_extend_encodeSnappyBlockAsm + SUBL $0x04, R8 + LEAL 4(R11), R11 + +matchlen_match2_repeat_extend_encodeSnappyBlockAsm: + CMPL R8, $0x02 + JL matchlen_match1_repeat_extend_encodeSnappyBlockAsm + MOVW (R9)(R11*1), R10 + CMPW (SI)(R11*1), R10 + JNE matchlen_match1_repeat_extend_encodeSnappyBlockAsm + SUBL $0x02, R8 + LEAL 2(R11), R11 + +matchlen_match1_repeat_extend_encodeSnappyBlockAsm: + CMPL R8, $0x01 + JL repeat_extend_forward_end_encodeSnappyBlockAsm + MOVB (R9)(R11*1), R10 + CMPB (SI)(R11*1), R10 + JNE repeat_extend_forward_end_encodeSnappyBlockAsm + LEAL 1(R11), R11 + +repeat_extend_forward_end_encodeSnappyBlockAsm: + ADDL R11, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + + // emitCopy + CMPL DI, $0x00010000 + JL two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm + +four_bytes_loop_back_repeat_as_copy_encodeSnappyBlockAsm: + CMPL SI, $0x40 + JLE four_bytes_remain_repeat_as_copy_encodeSnappyBlockAsm + MOVB $0xff, (AX) + MOVL DI, 1(AX) + LEAL -64(SI), SI + ADDQ $0x05, AX + CMPL SI, $0x04 + JL four_bytes_remain_repeat_as_copy_encodeSnappyBlockAsm + JMP four_bytes_loop_back_repeat_as_copy_encodeSnappyBlockAsm + +four_bytes_remain_repeat_as_copy_encodeSnappyBlockAsm: + TESTL SI, SI + JZ repeat_end_emit_encodeSnappyBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVL DI, 1(AX) + ADDQ $0x05, AX + JMP repeat_end_emit_encodeSnappyBlockAsm + +two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + JMP two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm + +two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeSnappyBlockAsm + +emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeSnappyBlockAsm: + MOVL CX, 12(SP) + JMP search_loop_encodeSnappyBlockAsm + +no_repeat_found_encodeSnappyBlockAsm: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBlockAsm + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeSnappyBlockAsm + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeSnappyBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBlockAsm + +candidate3_match_encodeSnappyBlockAsm: + ADDL $0x02, CX + JMP candidate_match_encodeSnappyBlockAsm + +candidate2_match_encodeSnappyBlockAsm: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeSnappyBlockAsm: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBlockAsm + +match_extend_back_loop_encodeSnappyBlockAsm: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBlockAsm + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBlockAsm + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBlockAsm + JMP match_extend_back_loop_encodeSnappyBlockAsm + +match_extend_back_end_encodeSnappyBlockAsm: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 5(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBlockAsm: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeSnappyBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeSnappyBlockAsm + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBlockAsm + CMPL R8, $0x00010000 + JLT three_bytes_match_emit_encodeSnappyBlockAsm + CMPL R8, $0x01000000 + JLT four_bytes_match_emit_encodeSnappyBlockAsm + MOVB $0xfc, (AX) + MOVL R8, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm + +four_bytes_match_emit_encodeSnappyBlockAsm: + MOVL R8, R10 + SHRL $0x10, R10 + MOVB $0xf8, (AX) + MOVW R8, 1(AX) + MOVB R10, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm + +three_bytes_match_emit_encodeSnappyBlockAsm: + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm + +two_bytes_match_emit_encodeSnappyBlockAsm: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeSnappyBlockAsm + JMP memmove_long_match_emit_encodeSnappyBlockAsm + +one_byte_match_emit_encodeSnappyBlockAsm: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBlockAsm: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBlockAsm: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeSnappyBlockAsm + +memmove_long_match_emit_encodeSnappyBlockAsm: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsmlarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeSnappyBlockAsm: +match_nolit_loop_encodeSnappyBlockAsm: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBlockAsm + +matchlen_loopback_match_nolit_encodeSnappyBlockAsm: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeSnappyBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeSnappyBlockAsm + +matchlen_loop_match_nolit_encodeSnappyBlockAsm: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBlockAsm + JZ match_nolit_end_encodeSnappyBlockAsm + +matchlen_match4_match_nolit_encodeSnappyBlockAsm: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBlockAsm + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeSnappyBlockAsm + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeSnappyBlockAsm: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBlockAsm + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeSnappyBlockAsm + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeSnappyBlockAsm: + CMPL DI, $0x01 + JL match_nolit_end_encodeSnappyBlockAsm + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeSnappyBlockAsm + LEAL 1(R10), R10 + +match_nolit_end_encodeSnappyBlockAsm: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy + CMPL SI, $0x00010000 + JL two_byte_offset_match_nolit_encodeSnappyBlockAsm + +four_bytes_loop_back_match_nolit_encodeSnappyBlockAsm: + CMPL R10, $0x40 + JLE four_bytes_remain_match_nolit_encodeSnappyBlockAsm + MOVB $0xff, (AX) + MOVL SI, 1(AX) + LEAL -64(R10), R10 + ADDQ $0x05, AX + CMPL R10, $0x04 + JL four_bytes_remain_match_nolit_encodeSnappyBlockAsm + JMP four_bytes_loop_back_match_nolit_encodeSnappyBlockAsm + +four_bytes_remain_match_nolit_encodeSnappyBlockAsm: + TESTL R10, R10 + JZ match_nolit_emitcopy_end_encodeSnappyBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm + +two_byte_offset_match_nolit_encodeSnappyBlockAsm: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBlockAsm + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBlockAsm + +two_byte_offset_short_match_nolit_encodeSnappyBlockAsm: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm + +emit_copy_three_match_nolit_encodeSnappyBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBlockAsm: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBlockAsm: + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x10, R8 + IMULQ R9, R8 + SHRQ $0x32, R8 + SHLQ $0x10, SI + IMULQ R9, SI + SHRQ $0x32, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeSnappyBlockAsm + INCL CX + JMP search_loop_encodeSnappyBlockAsm + +emit_remainder_encodeSnappyBlockAsm: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 5(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBlockAsm: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBlockAsm + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBlockAsm + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBlockAsm + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeSnappyBlockAsm + CMPL DX, $0x01000000 + JLT four_bytes_emit_remainder_encodeSnappyBlockAsm + MOVB $0xfc, (AX) + MOVL DX, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm + +four_bytes_emit_remainder_encodeSnappyBlockAsm: + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm + +three_bytes_emit_remainder_encodeSnappyBlockAsm: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm + +two_bytes_emit_remainder_encodeSnappyBlockAsm: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBlockAsm + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm + +one_byte_emit_remainder_encodeSnappyBlockAsm: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBlockAsm: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBlockAsm + +memmove_long_emit_remainder_encodeSnappyBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsmlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBlockAsm: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBlockAsm64K(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBlockAsm64K(SB), $65560-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000200, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBlockAsm64K: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBlockAsm64K + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBlockAsm64K: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm64K + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R11 + IMULQ R9, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeSnappyBlockAsm64K + LEAL 1(CX), DI + MOVL 12(SP), SI + MOVL DI, R8 + SUBL 16(SP), R8 + JZ repeat_extend_back_end_encodeSnappyBlockAsm64K + +repeat_extend_back_loop_encodeSnappyBlockAsm64K: + CMPL DI, SI + JLE repeat_extend_back_end_encodeSnappyBlockAsm64K + MOVB -1(DX)(R8*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeSnappyBlockAsm64K + LEAL -1(DI), DI + DECL R8 + JNZ repeat_extend_back_loop_encodeSnappyBlockAsm64K + +repeat_extend_back_end_encodeSnappyBlockAsm64K: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeSnappyBlockAsm64K + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeSnappyBlockAsm64K + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeSnappyBlockAsm64K + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm64K + +two_bytes_repeat_emit_encodeSnappyBlockAsm64K: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeSnappyBlockAsm64K + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm64K + +one_byte_repeat_emit_encodeSnappyBlockAsm64K: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeSnappyBlockAsm64K: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_8 + CMPQ R8, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_8: + MOVQ (R9), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm64K_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_repeat_emit_encodeSnappyBlockAsm64K: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeSnappyBlockAsm64K + +memmove_long_repeat_emit_encodeSnappyBlockAsm64K: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + LEAQ -32(R9)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm64Klarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm64Klarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32: + MOVOU -32(R9)(R12*1), X4 + MOVOU -16(R9)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R8, R12 + JAE emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeSnappyBlockAsm64K: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R11, R11 + CMPL R8, $0x08 + JL matchlen_match4_repeat_extend_encodeSnappyBlockAsm64K + +matchlen_loopback_repeat_extend_encodeSnappyBlockAsm64K: + MOVQ (R9)(R11*1), R10 + XORQ (SI)(R11*1), R10 + TESTQ R10, R10 + JZ matchlen_loop_repeat_extend_encodeSnappyBlockAsm64K + +#ifdef GOAMD64_v3 + TZCNTQ R10, R10 + +#else + BSFQ R10, R10 + +#endif + SARQ $0x03, R10 + LEAL (R11)(R10*1), R11 + JMP repeat_extend_forward_end_encodeSnappyBlockAsm64K + +matchlen_loop_repeat_extend_encodeSnappyBlockAsm64K: + LEAL -8(R8), R8 + LEAL 8(R11), R11 + CMPL R8, $0x08 + JGE matchlen_loopback_repeat_extend_encodeSnappyBlockAsm64K + JZ repeat_extend_forward_end_encodeSnappyBlockAsm64K + +matchlen_match4_repeat_extend_encodeSnappyBlockAsm64K: + CMPL R8, $0x04 + JL matchlen_match2_repeat_extend_encodeSnappyBlockAsm64K + MOVL (R9)(R11*1), R10 + CMPL (SI)(R11*1), R10 + JNE matchlen_match2_repeat_extend_encodeSnappyBlockAsm64K + SUBL $0x04, R8 + LEAL 4(R11), R11 + +matchlen_match2_repeat_extend_encodeSnappyBlockAsm64K: + CMPL R8, $0x02 + JL matchlen_match1_repeat_extend_encodeSnappyBlockAsm64K + MOVW (R9)(R11*1), R10 + CMPW (SI)(R11*1), R10 + JNE matchlen_match1_repeat_extend_encodeSnappyBlockAsm64K + SUBL $0x02, R8 + LEAL 2(R11), R11 + +matchlen_match1_repeat_extend_encodeSnappyBlockAsm64K: + CMPL R8, $0x01 + JL repeat_extend_forward_end_encodeSnappyBlockAsm64K + MOVB (R9)(R11*1), R10 + CMPB (SI)(R11*1), R10 + JNE repeat_extend_forward_end_encodeSnappyBlockAsm64K + LEAL 1(R11), R11 + +repeat_extend_forward_end_encodeSnappyBlockAsm64K: + ADDL R11, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + + // emitCopy +two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm64K: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm64K + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + JMP two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm64K + +two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm64K: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm64K + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm64K + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeSnappyBlockAsm64K + +emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm64K: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeSnappyBlockAsm64K: + MOVL CX, 12(SP) + JMP search_loop_encodeSnappyBlockAsm64K + +no_repeat_found_encodeSnappyBlockAsm64K: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBlockAsm64K + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeSnappyBlockAsm64K + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeSnappyBlockAsm64K + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBlockAsm64K + +candidate3_match_encodeSnappyBlockAsm64K: + ADDL $0x02, CX + JMP candidate_match_encodeSnappyBlockAsm64K + +candidate2_match_encodeSnappyBlockAsm64K: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeSnappyBlockAsm64K: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBlockAsm64K + +match_extend_back_loop_encodeSnappyBlockAsm64K: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBlockAsm64K + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBlockAsm64K + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBlockAsm64K + JMP match_extend_back_loop_encodeSnappyBlockAsm64K + +match_extend_back_end_encodeSnappyBlockAsm64K: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBlockAsm64K: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeSnappyBlockAsm64K + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeSnappyBlockAsm64K + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBlockAsm64K + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm64K + +two_bytes_match_emit_encodeSnappyBlockAsm64K: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeSnappyBlockAsm64K + JMP memmove_long_match_emit_encodeSnappyBlockAsm64K + +one_byte_match_emit_encodeSnappyBlockAsm64K: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBlockAsm64K: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm64K_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBlockAsm64K: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeSnappyBlockAsm64K + +memmove_long_match_emit_encodeSnappyBlockAsm64K: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm64Klarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm64Klarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeSnappyBlockAsm64K: +match_nolit_loop_encodeSnappyBlockAsm64K: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBlockAsm64K + +matchlen_loopback_match_nolit_encodeSnappyBlockAsm64K: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeSnappyBlockAsm64K + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeSnappyBlockAsm64K + +matchlen_loop_match_nolit_encodeSnappyBlockAsm64K: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBlockAsm64K + JZ match_nolit_end_encodeSnappyBlockAsm64K + +matchlen_match4_match_nolit_encodeSnappyBlockAsm64K: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBlockAsm64K + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeSnappyBlockAsm64K + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeSnappyBlockAsm64K: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBlockAsm64K + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeSnappyBlockAsm64K + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeSnappyBlockAsm64K: + CMPL DI, $0x01 + JL match_nolit_end_encodeSnappyBlockAsm64K + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeSnappyBlockAsm64K + LEAL 1(R10), R10 + +match_nolit_end_encodeSnappyBlockAsm64K: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBlockAsm64K: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBlockAsm64K + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBlockAsm64K + +two_byte_offset_short_match_nolit_encodeSnappyBlockAsm64K: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm64K + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm64K + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm64K + +emit_copy_three_match_nolit_encodeSnappyBlockAsm64K: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBlockAsm64K: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm64K + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBlockAsm64K: + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x10, R8 + IMULQ R9, R8 + SHRQ $0x32, R8 + SHLQ $0x10, SI + IMULQ R9, SI + SHRQ $0x32, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeSnappyBlockAsm64K + INCL CX + JMP search_loop_encodeSnappyBlockAsm64K + +emit_remainder_encodeSnappyBlockAsm64K: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBlockAsm64K: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBlockAsm64K + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBlockAsm64K + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBlockAsm64K + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm64K + +two_bytes_emit_remainder_encodeSnappyBlockAsm64K: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBlockAsm64K + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm64K + +one_byte_emit_remainder_encodeSnappyBlockAsm64K: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBlockAsm64K: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm64K_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBlockAsm64K: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBlockAsm64K + +memmove_long_emit_remainder_encodeSnappyBlockAsm64K: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm64Klarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm64Klarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm64Klarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBlockAsm64K: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBlockAsm12B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBlockAsm12B(SB), $16408-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000080, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBlockAsm12B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBlockAsm12B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBlockAsm12B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm12B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x000000cf1bbcdcbb, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x18, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + SHLQ $0x18, R11 + IMULQ R9, R11 + SHRQ $0x34, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x18, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeSnappyBlockAsm12B + LEAL 1(CX), DI + MOVL 12(SP), SI + MOVL DI, R8 + SUBL 16(SP), R8 + JZ repeat_extend_back_end_encodeSnappyBlockAsm12B + +repeat_extend_back_loop_encodeSnappyBlockAsm12B: + CMPL DI, SI + JLE repeat_extend_back_end_encodeSnappyBlockAsm12B + MOVB -1(DX)(R8*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeSnappyBlockAsm12B + LEAL -1(DI), DI + DECL R8 + JNZ repeat_extend_back_loop_encodeSnappyBlockAsm12B + +repeat_extend_back_end_encodeSnappyBlockAsm12B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeSnappyBlockAsm12B + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeSnappyBlockAsm12B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeSnappyBlockAsm12B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm12B + +two_bytes_repeat_emit_encodeSnappyBlockAsm12B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeSnappyBlockAsm12B + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm12B + +one_byte_repeat_emit_encodeSnappyBlockAsm12B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeSnappyBlockAsm12B: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_8 + CMPQ R8, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_8: + MOVQ (R9), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm12B_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_repeat_emit_encodeSnappyBlockAsm12B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeSnappyBlockAsm12B + +memmove_long_repeat_emit_encodeSnappyBlockAsm12B: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(R9)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm12Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(R9)(R12*1), X4 + MOVOU -16(R9)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R8, R12 + JAE emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeSnappyBlockAsm12B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R11, R11 + CMPL R8, $0x08 + JL matchlen_match4_repeat_extend_encodeSnappyBlockAsm12B + +matchlen_loopback_repeat_extend_encodeSnappyBlockAsm12B: + MOVQ (R9)(R11*1), R10 + XORQ (SI)(R11*1), R10 + TESTQ R10, R10 + JZ matchlen_loop_repeat_extend_encodeSnappyBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R10, R10 + +#else + BSFQ R10, R10 + +#endif + SARQ $0x03, R10 + LEAL (R11)(R10*1), R11 + JMP repeat_extend_forward_end_encodeSnappyBlockAsm12B + +matchlen_loop_repeat_extend_encodeSnappyBlockAsm12B: + LEAL -8(R8), R8 + LEAL 8(R11), R11 + CMPL R8, $0x08 + JGE matchlen_loopback_repeat_extend_encodeSnappyBlockAsm12B + JZ repeat_extend_forward_end_encodeSnappyBlockAsm12B + +matchlen_match4_repeat_extend_encodeSnappyBlockAsm12B: + CMPL R8, $0x04 + JL matchlen_match2_repeat_extend_encodeSnappyBlockAsm12B + MOVL (R9)(R11*1), R10 + CMPL (SI)(R11*1), R10 + JNE matchlen_match2_repeat_extend_encodeSnappyBlockAsm12B + SUBL $0x04, R8 + LEAL 4(R11), R11 + +matchlen_match2_repeat_extend_encodeSnappyBlockAsm12B: + CMPL R8, $0x02 + JL matchlen_match1_repeat_extend_encodeSnappyBlockAsm12B + MOVW (R9)(R11*1), R10 + CMPW (SI)(R11*1), R10 + JNE matchlen_match1_repeat_extend_encodeSnappyBlockAsm12B + SUBL $0x02, R8 + LEAL 2(R11), R11 + +matchlen_match1_repeat_extend_encodeSnappyBlockAsm12B: + CMPL R8, $0x01 + JL repeat_extend_forward_end_encodeSnappyBlockAsm12B + MOVB (R9)(R11*1), R10 + CMPB (SI)(R11*1), R10 + JNE repeat_extend_forward_end_encodeSnappyBlockAsm12B + LEAL 1(R11), R11 + +repeat_extend_forward_end_encodeSnappyBlockAsm12B: + ADDL R11, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + + // emitCopy +two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm12B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm12B + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + JMP two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm12B + +two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm12B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm12B + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeSnappyBlockAsm12B + +emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeSnappyBlockAsm12B: + MOVL CX, 12(SP) + JMP search_loop_encodeSnappyBlockAsm12B + +no_repeat_found_encodeSnappyBlockAsm12B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBlockAsm12B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeSnappyBlockAsm12B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeSnappyBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBlockAsm12B + +candidate3_match_encodeSnappyBlockAsm12B: + ADDL $0x02, CX + JMP candidate_match_encodeSnappyBlockAsm12B + +candidate2_match_encodeSnappyBlockAsm12B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeSnappyBlockAsm12B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBlockAsm12B + +match_extend_back_loop_encodeSnappyBlockAsm12B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBlockAsm12B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBlockAsm12B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBlockAsm12B + JMP match_extend_back_loop_encodeSnappyBlockAsm12B + +match_extend_back_end_encodeSnappyBlockAsm12B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBlockAsm12B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeSnappyBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeSnappyBlockAsm12B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBlockAsm12B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm12B + +two_bytes_match_emit_encodeSnappyBlockAsm12B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeSnappyBlockAsm12B + JMP memmove_long_match_emit_encodeSnappyBlockAsm12B + +one_byte_match_emit_encodeSnappyBlockAsm12B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBlockAsm12B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm12B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBlockAsm12B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeSnappyBlockAsm12B + +memmove_long_match_emit_encodeSnappyBlockAsm12B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm12Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeSnappyBlockAsm12B: +match_nolit_loop_encodeSnappyBlockAsm12B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBlockAsm12B + +matchlen_loopback_match_nolit_encodeSnappyBlockAsm12B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeSnappyBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeSnappyBlockAsm12B + +matchlen_loop_match_nolit_encodeSnappyBlockAsm12B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBlockAsm12B + JZ match_nolit_end_encodeSnappyBlockAsm12B + +matchlen_match4_match_nolit_encodeSnappyBlockAsm12B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBlockAsm12B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeSnappyBlockAsm12B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeSnappyBlockAsm12B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBlockAsm12B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeSnappyBlockAsm12B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeSnappyBlockAsm12B: + CMPL DI, $0x01 + JL match_nolit_end_encodeSnappyBlockAsm12B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeSnappyBlockAsm12B + LEAL 1(R10), R10 + +match_nolit_end_encodeSnappyBlockAsm12B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBlockAsm12B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBlockAsm12B + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBlockAsm12B + +two_byte_offset_short_match_nolit_encodeSnappyBlockAsm12B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm12B + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm12B + +emit_copy_three_match_nolit_encodeSnappyBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBlockAsm12B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm12B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBlockAsm12B: + MOVQ $0x000000cf1bbcdcbb, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x18, R8 + IMULQ R9, R8 + SHRQ $0x34, R8 + SHLQ $0x18, SI + IMULQ R9, SI + SHRQ $0x34, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeSnappyBlockAsm12B + INCL CX + JMP search_loop_encodeSnappyBlockAsm12B + +emit_remainder_encodeSnappyBlockAsm12B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBlockAsm12B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBlockAsm12B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBlockAsm12B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBlockAsm12B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm12B + +two_bytes_emit_remainder_encodeSnappyBlockAsm12B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBlockAsm12B + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm12B + +one_byte_emit_remainder_encodeSnappyBlockAsm12B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm12B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBlockAsm12B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBlockAsm12B + +memmove_long_emit_remainder_encodeSnappyBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm12Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBlockAsm12B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBlockAsm10B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBlockAsm10B(SB), $4120-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000020, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBlockAsm10B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBlockAsm10B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBlockAsm10B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm10B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x9e3779b1, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ R9, R11 + SHRQ $0x36, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeSnappyBlockAsm10B + LEAL 1(CX), DI + MOVL 12(SP), SI + MOVL DI, R8 + SUBL 16(SP), R8 + JZ repeat_extend_back_end_encodeSnappyBlockAsm10B + +repeat_extend_back_loop_encodeSnappyBlockAsm10B: + CMPL DI, SI + JLE repeat_extend_back_end_encodeSnappyBlockAsm10B + MOVB -1(DX)(R8*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeSnappyBlockAsm10B + LEAL -1(DI), DI + DECL R8 + JNZ repeat_extend_back_loop_encodeSnappyBlockAsm10B + +repeat_extend_back_end_encodeSnappyBlockAsm10B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeSnappyBlockAsm10B + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeSnappyBlockAsm10B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeSnappyBlockAsm10B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm10B + +two_bytes_repeat_emit_encodeSnappyBlockAsm10B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeSnappyBlockAsm10B + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm10B + +one_byte_repeat_emit_encodeSnappyBlockAsm10B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeSnappyBlockAsm10B: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_8 + CMPQ R8, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_8: + MOVQ (R9), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm10B_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_repeat_emit_encodeSnappyBlockAsm10B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeSnappyBlockAsm10B + +memmove_long_repeat_emit_encodeSnappyBlockAsm10B: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(R9)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm10Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(R9)(R12*1), X4 + MOVOU -16(R9)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R8, R12 + JAE emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeSnappyBlockAsm10B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R11, R11 + CMPL R8, $0x08 + JL matchlen_match4_repeat_extend_encodeSnappyBlockAsm10B + +matchlen_loopback_repeat_extend_encodeSnappyBlockAsm10B: + MOVQ (R9)(R11*1), R10 + XORQ (SI)(R11*1), R10 + TESTQ R10, R10 + JZ matchlen_loop_repeat_extend_encodeSnappyBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R10, R10 + +#else + BSFQ R10, R10 + +#endif + SARQ $0x03, R10 + LEAL (R11)(R10*1), R11 + JMP repeat_extend_forward_end_encodeSnappyBlockAsm10B + +matchlen_loop_repeat_extend_encodeSnappyBlockAsm10B: + LEAL -8(R8), R8 + LEAL 8(R11), R11 + CMPL R8, $0x08 + JGE matchlen_loopback_repeat_extend_encodeSnappyBlockAsm10B + JZ repeat_extend_forward_end_encodeSnappyBlockAsm10B + +matchlen_match4_repeat_extend_encodeSnappyBlockAsm10B: + CMPL R8, $0x04 + JL matchlen_match2_repeat_extend_encodeSnappyBlockAsm10B + MOVL (R9)(R11*1), R10 + CMPL (SI)(R11*1), R10 + JNE matchlen_match2_repeat_extend_encodeSnappyBlockAsm10B + SUBL $0x04, R8 + LEAL 4(R11), R11 + +matchlen_match2_repeat_extend_encodeSnappyBlockAsm10B: + CMPL R8, $0x02 + JL matchlen_match1_repeat_extend_encodeSnappyBlockAsm10B + MOVW (R9)(R11*1), R10 + CMPW (SI)(R11*1), R10 + JNE matchlen_match1_repeat_extend_encodeSnappyBlockAsm10B + SUBL $0x02, R8 + LEAL 2(R11), R11 + +matchlen_match1_repeat_extend_encodeSnappyBlockAsm10B: + CMPL R8, $0x01 + JL repeat_extend_forward_end_encodeSnappyBlockAsm10B + MOVB (R9)(R11*1), R10 + CMPB (SI)(R11*1), R10 + JNE repeat_extend_forward_end_encodeSnappyBlockAsm10B + LEAL 1(R11), R11 + +repeat_extend_forward_end_encodeSnappyBlockAsm10B: + ADDL R11, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + + // emitCopy +two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm10B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm10B + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + JMP two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm10B + +two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm10B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm10B + CMPL DI, $0x00000800 + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeSnappyBlockAsm10B + +emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeSnappyBlockAsm10B: + MOVL CX, 12(SP) + JMP search_loop_encodeSnappyBlockAsm10B + +no_repeat_found_encodeSnappyBlockAsm10B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBlockAsm10B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeSnappyBlockAsm10B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeSnappyBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBlockAsm10B + +candidate3_match_encodeSnappyBlockAsm10B: + ADDL $0x02, CX + JMP candidate_match_encodeSnappyBlockAsm10B + +candidate2_match_encodeSnappyBlockAsm10B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeSnappyBlockAsm10B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBlockAsm10B + +match_extend_back_loop_encodeSnappyBlockAsm10B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBlockAsm10B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBlockAsm10B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBlockAsm10B + JMP match_extend_back_loop_encodeSnappyBlockAsm10B + +match_extend_back_end_encodeSnappyBlockAsm10B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBlockAsm10B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeSnappyBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeSnappyBlockAsm10B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBlockAsm10B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm10B + +two_bytes_match_emit_encodeSnappyBlockAsm10B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeSnappyBlockAsm10B + JMP memmove_long_match_emit_encodeSnappyBlockAsm10B + +one_byte_match_emit_encodeSnappyBlockAsm10B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBlockAsm10B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm10B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBlockAsm10B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeSnappyBlockAsm10B + +memmove_long_match_emit_encodeSnappyBlockAsm10B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm10Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeSnappyBlockAsm10B: +match_nolit_loop_encodeSnappyBlockAsm10B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBlockAsm10B + +matchlen_loopback_match_nolit_encodeSnappyBlockAsm10B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeSnappyBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeSnappyBlockAsm10B + +matchlen_loop_match_nolit_encodeSnappyBlockAsm10B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBlockAsm10B + JZ match_nolit_end_encodeSnappyBlockAsm10B + +matchlen_match4_match_nolit_encodeSnappyBlockAsm10B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBlockAsm10B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeSnappyBlockAsm10B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeSnappyBlockAsm10B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBlockAsm10B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeSnappyBlockAsm10B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeSnappyBlockAsm10B: + CMPL DI, $0x01 + JL match_nolit_end_encodeSnappyBlockAsm10B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeSnappyBlockAsm10B + LEAL 1(R10), R10 + +match_nolit_end_encodeSnappyBlockAsm10B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBlockAsm10B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBlockAsm10B + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBlockAsm10B + +two_byte_offset_short_match_nolit_encodeSnappyBlockAsm10B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm10B + CMPL SI, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm10B + +emit_copy_three_match_nolit_encodeSnappyBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBlockAsm10B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm10B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBlockAsm10B: + MOVQ $0x9e3779b1, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x20, R8 + IMULQ R9, R8 + SHRQ $0x36, R8 + SHLQ $0x20, SI + IMULQ R9, SI + SHRQ $0x36, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeSnappyBlockAsm10B + INCL CX + JMP search_loop_encodeSnappyBlockAsm10B + +emit_remainder_encodeSnappyBlockAsm10B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBlockAsm10B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBlockAsm10B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBlockAsm10B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBlockAsm10B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm10B + +two_bytes_emit_remainder_encodeSnappyBlockAsm10B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBlockAsm10B + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm10B + +one_byte_emit_remainder_encodeSnappyBlockAsm10B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm10B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBlockAsm10B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBlockAsm10B + +memmove_long_emit_remainder_encodeSnappyBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm10Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBlockAsm10B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBlockAsm8B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBlockAsm8B(SB), $1048-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000008, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBlockAsm8B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBlockAsm8B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL CX, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBlockAsm8B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x04, SI + LEAL 4(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm8B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x9e3779b1, R9 + MOVQ DI, R10 + MOVQ DI, R11 + SHRQ $0x08, R11 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x38, R10 + SHLQ $0x20, R11 + IMULQ R9, R11 + SHRQ $0x38, R11 + MOVL 24(SP)(R10*4), SI + MOVL 24(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + LEAL 1(CX), R10 + MOVL R10, 24(SP)(R11*4) + MOVQ DI, R10 + SHRQ $0x10, R10 + SHLQ $0x20, R10 + IMULQ R9, R10 + SHRQ $0x38, R10 + MOVL CX, R9 + SUBL 16(SP), R9 + MOVL 1(DX)(R9*1), R11 + MOVQ DI, R9 + SHRQ $0x08, R9 + CMPL R9, R11 + JNE no_repeat_found_encodeSnappyBlockAsm8B + LEAL 1(CX), DI + MOVL 12(SP), SI + MOVL DI, R8 + SUBL 16(SP), R8 + JZ repeat_extend_back_end_encodeSnappyBlockAsm8B + +repeat_extend_back_loop_encodeSnappyBlockAsm8B: + CMPL DI, SI + JLE repeat_extend_back_end_encodeSnappyBlockAsm8B + MOVB -1(DX)(R8*1), BL + MOVB -1(DX)(DI*1), R9 + CMPB BL, R9 + JNE repeat_extend_back_end_encodeSnappyBlockAsm8B + LEAL -1(DI), DI + DECL R8 + JNZ repeat_extend_back_loop_encodeSnappyBlockAsm8B + +repeat_extend_back_end_encodeSnappyBlockAsm8B: + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_repeat_emit_encodeSnappyBlockAsm8B + MOVL DI, R8 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R9 + SUBL SI, R8 + LEAL -1(R8), SI + CMPL SI, $0x3c + JLT one_byte_repeat_emit_encodeSnappyBlockAsm8B + CMPL SI, $0x00000100 + JLT two_bytes_repeat_emit_encodeSnappyBlockAsm8B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm8B + +two_bytes_repeat_emit_encodeSnappyBlockAsm8B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_repeat_emit_encodeSnappyBlockAsm8B + JMP memmove_long_repeat_emit_encodeSnappyBlockAsm8B + +one_byte_repeat_emit_encodeSnappyBlockAsm8B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_repeat_emit_encodeSnappyBlockAsm8B: + LEAQ (AX)(R8*1), SI + + // genMemMoveShort + CMPQ R8, $0x08 + JLE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_8 + CMPQ R8, $0x10 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_8through16 + CMPQ R8, $0x20 + JBE emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_8: + MOVQ (R9), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_8through16: + MOVQ (R9), R10 + MOVQ -8(R9)(R8*1), R9 + MOVQ R10, (AX) + MOVQ R9, -8(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_17through32: + MOVOU (R9), X0 + MOVOU -16(R9)(R8*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R8*1) + JMP memmove_end_copy_repeat_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_repeat_emit_encodeSnappyBlockAsm8B_memmove_move_33through64: + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + +memmove_end_copy_repeat_emit_encodeSnappyBlockAsm8B: + MOVQ SI, AX + JMP emit_literal_done_repeat_emit_encodeSnappyBlockAsm8B + +memmove_long_repeat_emit_encodeSnappyBlockAsm8B: + LEAQ (AX)(R8*1), SI + + // genMemMoveLong + MOVOU (R9), X0 + MOVOU 16(R9), X1 + MOVOU -32(R9)(R8*1), X2 + MOVOU -16(R9)(R8*1), X3 + MOVQ R8, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(R9)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm8Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(R9)(R12*1), X4 + MOVOU -16(R9)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R8, R12 + JAE emit_lit_memmove_long_repeat_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R8*1) + MOVOU X3, -16(AX)(R8*1) + MOVQ SI, AX + +emit_literal_done_repeat_emit_encodeSnappyBlockAsm8B: + ADDL $0x05, CX + MOVL CX, SI + SUBL 16(SP), SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R11, R11 + CMPL R8, $0x08 + JL matchlen_match4_repeat_extend_encodeSnappyBlockAsm8B + +matchlen_loopback_repeat_extend_encodeSnappyBlockAsm8B: + MOVQ (R9)(R11*1), R10 + XORQ (SI)(R11*1), R10 + TESTQ R10, R10 + JZ matchlen_loop_repeat_extend_encodeSnappyBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R10, R10 + +#else + BSFQ R10, R10 + +#endif + SARQ $0x03, R10 + LEAL (R11)(R10*1), R11 + JMP repeat_extend_forward_end_encodeSnappyBlockAsm8B + +matchlen_loop_repeat_extend_encodeSnappyBlockAsm8B: + LEAL -8(R8), R8 + LEAL 8(R11), R11 + CMPL R8, $0x08 + JGE matchlen_loopback_repeat_extend_encodeSnappyBlockAsm8B + JZ repeat_extend_forward_end_encodeSnappyBlockAsm8B + +matchlen_match4_repeat_extend_encodeSnappyBlockAsm8B: + CMPL R8, $0x04 + JL matchlen_match2_repeat_extend_encodeSnappyBlockAsm8B + MOVL (R9)(R11*1), R10 + CMPL (SI)(R11*1), R10 + JNE matchlen_match2_repeat_extend_encodeSnappyBlockAsm8B + SUBL $0x04, R8 + LEAL 4(R11), R11 + +matchlen_match2_repeat_extend_encodeSnappyBlockAsm8B: + CMPL R8, $0x02 + JL matchlen_match1_repeat_extend_encodeSnappyBlockAsm8B + MOVW (R9)(R11*1), R10 + CMPW (SI)(R11*1), R10 + JNE matchlen_match1_repeat_extend_encodeSnappyBlockAsm8B + SUBL $0x02, R8 + LEAL 2(R11), R11 + +matchlen_match1_repeat_extend_encodeSnappyBlockAsm8B: + CMPL R8, $0x01 + JL repeat_extend_forward_end_encodeSnappyBlockAsm8B + MOVB (R9)(R11*1), R10 + CMPB (SI)(R11*1), R10 + JNE repeat_extend_forward_end_encodeSnappyBlockAsm8B + LEAL 1(R11), R11 + +repeat_extend_forward_end_encodeSnappyBlockAsm8B: + ADDL R11, CX + MOVL CX, SI + SUBL DI, SI + MOVL 16(SP), DI + + // emitCopy +two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm8B: + CMPL SI, $0x40 + JLE two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm8B + MOVB $0xee, (AX) + MOVW DI, 1(AX) + LEAL -60(SI), SI + ADDQ $0x03, AX + JMP two_byte_offset_repeat_as_copy_encodeSnappyBlockAsm8B + +two_byte_offset_short_repeat_as_copy_encodeSnappyBlockAsm8B: + CMPL SI, $0x0c + JGE emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(SI*4), SI + MOVB DI, 1(AX) + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, AX + JMP repeat_end_emit_encodeSnappyBlockAsm8B + +emit_copy_three_repeat_as_copy_encodeSnappyBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(SI*4), SI + MOVB SI, (AX) + MOVW DI, 1(AX) + ADDQ $0x03, AX + +repeat_end_emit_encodeSnappyBlockAsm8B: + MOVL CX, 12(SP) + JMP search_loop_encodeSnappyBlockAsm8B + +no_repeat_found_encodeSnappyBlockAsm8B: + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBlockAsm8B + SHRQ $0x08, DI + MOVL 24(SP)(R10*4), SI + LEAL 2(CX), R9 + CMPL (DX)(R8*1), DI + JEQ candidate2_match_encodeSnappyBlockAsm8B + MOVL R9, 24(SP)(R10*4) + SHRQ $0x08, DI + CMPL (DX)(SI*1), DI + JEQ candidate3_match_encodeSnappyBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBlockAsm8B + +candidate3_match_encodeSnappyBlockAsm8B: + ADDL $0x02, CX + JMP candidate_match_encodeSnappyBlockAsm8B + +candidate2_match_encodeSnappyBlockAsm8B: + MOVL R9, 24(SP)(R10*4) + INCL CX + MOVL R8, SI + +candidate_match_encodeSnappyBlockAsm8B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBlockAsm8B + +match_extend_back_loop_encodeSnappyBlockAsm8B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBlockAsm8B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBlockAsm8B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBlockAsm8B + JMP match_extend_back_loop_encodeSnappyBlockAsm8B + +match_extend_back_end_encodeSnappyBlockAsm8B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBlockAsm8B: + MOVL CX, DI + MOVL 12(SP), R8 + CMPL R8, DI + JEQ emit_literal_done_match_emit_encodeSnappyBlockAsm8B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(R8*1), DI + SUBL R8, R9 + LEAL -1(R9), R8 + CMPL R8, $0x3c + JLT one_byte_match_emit_encodeSnappyBlockAsm8B + CMPL R8, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBlockAsm8B + MOVB $0xf4, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBlockAsm8B + +two_bytes_match_emit_encodeSnappyBlockAsm8B: + MOVB $0xf0, (AX) + MOVB R8, 1(AX) + ADDQ $0x02, AX + CMPL R8, $0x40 + JL memmove_match_emit_encodeSnappyBlockAsm8B + JMP memmove_long_match_emit_encodeSnappyBlockAsm8B + +one_byte_match_emit_encodeSnappyBlockAsm8B: + SHLB $0x02, R8 + MOVB R8, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBlockAsm8B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_8: + MOVQ (DI), R10 + MOVQ R10, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_8through16: + MOVQ (DI), R10 + MOVQ -8(DI)(R9*1), DI + MOVQ R10, (AX) + MOVQ DI, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_17through32: + MOVOU (DI), X0 + MOVOU -16(DI)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBlockAsm8B_memmove_move_33through64: + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBlockAsm8B: + MOVQ R8, AX + JMP emit_literal_done_match_emit_encodeSnappyBlockAsm8B + +memmove_long_match_emit_encodeSnappyBlockAsm8B: + LEAQ (AX)(R9*1), R8 + + // genMemMoveLong + MOVOU (DI), X0 + MOVOU 16(DI), X1 + MOVOU -32(DI)(R9*1), X2 + MOVOU -16(DI)(R9*1), X3 + MOVQ R9, R11 + SHRQ $0x05, R11 + MOVQ AX, R10 + ANDL $0x0000001f, R10 + MOVQ $0x00000040, R12 + SUBQ R10, R12 + DECQ R11 + JA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(DI)(R12*1), R10 + LEAQ -32(AX)(R12*1), R13 + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm8Blarge_big_loop_back: + MOVOU (R10), X4 + MOVOU 16(R10), X5 + MOVOA X4, (R13) + MOVOA X5, 16(R13) + ADDQ $0x20, R13 + ADDQ $0x20, R10 + ADDQ $0x20, R12 + DECQ R11 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(DI)(R12*1), X4 + MOVOU -16(DI)(R12*1), X5 + MOVOA X4, -32(AX)(R12*1) + MOVOA X5, -16(AX)(R12*1) + ADDQ $0x20, R12 + CMPQ R9, R12 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ R8, AX + +emit_literal_done_match_emit_encodeSnappyBlockAsm8B: +match_nolit_loop_encodeSnappyBlockAsm8B: + MOVL CX, DI + SUBL SI, DI + MOVL DI, 16(SP) + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), DI + SUBL CX, DI + LEAQ (DX)(CX*1), R8 + LEAQ (DX)(SI*1), SI + + // matchLen + XORL R10, R10 + CMPL DI, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBlockAsm8B + +matchlen_loopback_match_nolit_encodeSnappyBlockAsm8B: + MOVQ (R8)(R10*1), R9 + XORQ (SI)(R10*1), R9 + TESTQ R9, R9 + JZ matchlen_loop_match_nolit_encodeSnappyBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R9, R9 + +#else + BSFQ R9, R9 + +#endif + SARQ $0x03, R9 + LEAL (R10)(R9*1), R10 + JMP match_nolit_end_encodeSnappyBlockAsm8B + +matchlen_loop_match_nolit_encodeSnappyBlockAsm8B: + LEAL -8(DI), DI + LEAL 8(R10), R10 + CMPL DI, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBlockAsm8B + JZ match_nolit_end_encodeSnappyBlockAsm8B + +matchlen_match4_match_nolit_encodeSnappyBlockAsm8B: + CMPL DI, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBlockAsm8B + MOVL (R8)(R10*1), R9 + CMPL (SI)(R10*1), R9 + JNE matchlen_match2_match_nolit_encodeSnappyBlockAsm8B + SUBL $0x04, DI + LEAL 4(R10), R10 + +matchlen_match2_match_nolit_encodeSnappyBlockAsm8B: + CMPL DI, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBlockAsm8B + MOVW (R8)(R10*1), R9 + CMPW (SI)(R10*1), R9 + JNE matchlen_match1_match_nolit_encodeSnappyBlockAsm8B + SUBL $0x02, DI + LEAL 2(R10), R10 + +matchlen_match1_match_nolit_encodeSnappyBlockAsm8B: + CMPL DI, $0x01 + JL match_nolit_end_encodeSnappyBlockAsm8B + MOVB (R8)(R10*1), R9 + CMPB (SI)(R10*1), R9 + JNE match_nolit_end_encodeSnappyBlockAsm8B + LEAL 1(R10), R10 + +match_nolit_end_encodeSnappyBlockAsm8B: + ADDL R10, CX + MOVL 16(SP), SI + ADDL $0x04, R10 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBlockAsm8B: + CMPL R10, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBlockAsm8B + MOVB $0xee, (AX) + MOVW SI, 1(AX) + LEAL -60(R10), R10 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBlockAsm8B + +two_byte_offset_short_match_nolit_encodeSnappyBlockAsm8B: + CMPL R10, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(R10*4), R10 + MOVB SI, 1(AX) + SHRL $0x08, SI + SHLL $0x05, SI + ORL SI, R10 + MOVB R10, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBlockAsm8B + +emit_copy_three_match_nolit_encodeSnappyBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(R10*4), R10 + MOVB R10, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBlockAsm8B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBlockAsm8B + MOVQ -2(DX)(CX*1), DI + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBlockAsm8B: + MOVQ $0x9e3779b1, R9 + MOVQ DI, R8 + SHRQ $0x10, DI + MOVQ DI, SI + SHLQ $0x20, R8 + IMULQ R9, R8 + SHRQ $0x38, R8 + SHLQ $0x20, SI + IMULQ R9, SI + SHRQ $0x38, SI + LEAL -2(CX), R9 + LEAQ 24(SP)(SI*4), R10 + MOVL (R10), SI + MOVL R9, 24(SP)(R8*4) + MOVL CX, (R10) + CMPL (DX)(SI*1), DI + JEQ match_nolit_loop_encodeSnappyBlockAsm8B + INCL CX + JMP search_loop_encodeSnappyBlockAsm8B + +emit_remainder_encodeSnappyBlockAsm8B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBlockAsm8B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBlockAsm8B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBlockAsm8B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBlockAsm8B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm8B + +two_bytes_emit_remainder_encodeSnappyBlockAsm8B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBlockAsm8B + JMP memmove_long_emit_remainder_encodeSnappyBlockAsm8B + +one_byte_emit_remainder_encodeSnappyBlockAsm8B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBlockAsm8B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBlockAsm8B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBlockAsm8B + +memmove_long_emit_remainder_encodeSnappyBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm8Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBlockAsm8B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBetterBlockAsm(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBetterBlockAsm(SB), $327704-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000a00, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBetterBlockAsm: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBetterBlockAsm + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBetterBlockAsm: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x07, SI + CMPL SI, $0x63 + JLE check_maxskip_ok_encodeSnappyBetterBlockAsm + LEAL 100(CX), SI + JMP check_maxskip_cont_encodeSnappyBetterBlockAsm + +check_maxskip_ok_encodeSnappyBetterBlockAsm: + LEAL 1(CX)(SI*1), SI + +check_maxskip_cont_encodeSnappyBetterBlockAsm: + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x00cf1bbcdcbfa563, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 262168(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 262168(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm + +candidateS_match_encodeSnappyBetterBlockAsm: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm + DECL CX + MOVL R8, SI + +candidate_match_encodeSnappyBetterBlockAsm: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm + +match_extend_back_loop_encodeSnappyBetterBlockAsm: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBetterBlockAsm + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBetterBlockAsm + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm + JMP match_extend_back_loop_encodeSnappyBetterBlockAsm + +match_extend_back_end_encodeSnappyBetterBlockAsm: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 5(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBetterBlockAsm: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm + +matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeSnappyBetterBlockAsm + +matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm + JZ match_nolit_end_encodeSnappyBetterBlockAsm + +matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R8, $0x01 + JL match_nolit_end_encodeSnappyBetterBlockAsm + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeSnappyBetterBlockAsm + LEAL 1(R12), R12 + +match_nolit_end_encodeSnappyBetterBlockAsm: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + CMPL R12, $0x01 + JG match_length_ok_encodeSnappyBetterBlockAsm + CMPL R8, $0x0000ffff + JLE match_length_ok_encodeSnappyBetterBlockAsm + MOVL 20(SP), CX + INCL CX + JMP search_loop_encodeSnappyBetterBlockAsm + +match_length_ok_encodeSnappyBetterBlockAsm: + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeSnappyBetterBlockAsm + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeSnappyBetterBlockAsm + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBetterBlockAsm + CMPL SI, $0x00010000 + JLT three_bytes_match_emit_encodeSnappyBetterBlockAsm + CMPL SI, $0x01000000 + JLT four_bytes_match_emit_encodeSnappyBetterBlockAsm + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm + +four_bytes_match_emit_encodeSnappyBetterBlockAsm: + MOVL SI, R11 + SHRL $0x10, R11 + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB R11, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm + +three_bytes_match_emit_encodeSnappyBetterBlockAsm: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm + +two_bytes_match_emit_encodeSnappyBetterBlockAsm: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeSnappyBetterBlockAsm + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm + +one_byte_match_emit_encodeSnappyBetterBlockAsm: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeSnappyBetterBlockAsm + +memmove_long_match_emit_encodeSnappyBetterBlockAsm: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsmlarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeSnappyBetterBlockAsm: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy + CMPL R8, $0x00010000 + JL two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm + +four_bytes_loop_back_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R12, $0x40 + JLE four_bytes_remain_match_nolit_encodeSnappyBetterBlockAsm + MOVB $0xff, (AX) + MOVL R8, 1(AX) + LEAL -64(R12), R12 + ADDQ $0x05, AX + CMPL R12, $0x04 + JL four_bytes_remain_match_nolit_encodeSnappyBetterBlockAsm + JMP four_bytes_loop_back_match_nolit_encodeSnappyBetterBlockAsm + +four_bytes_remain_match_nolit_encodeSnappyBetterBlockAsm: + TESTL R12, R12 + JZ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm + MOVB $0x03, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVL R8, 1(AX) + ADDQ $0x05, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm + +two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm + +two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm + +emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBetterBlockAsm: + MOVQ $0x00cf1bbcdcbfa563, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x32, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 262168(SP)(R11*4) + MOVL R15, 262168(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 262168(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeSnappyBetterBlockAsm + +emit_remainder_encodeSnappyBetterBlockAsm: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 5(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBetterBlockAsm + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBetterBlockAsm: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBetterBlockAsm + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBetterBlockAsm + CMPL DX, $0x00010000 + JLT three_bytes_emit_remainder_encodeSnappyBetterBlockAsm + CMPL DX, $0x01000000 + JLT four_bytes_emit_remainder_encodeSnappyBetterBlockAsm + MOVB $0xfc, (AX) + MOVL DX, 1(AX) + ADDQ $0x05, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm + +four_bytes_emit_remainder_encodeSnappyBetterBlockAsm: + MOVL DX, BX + SHRL $0x10, BX + MOVB $0xf8, (AX) + MOVW DX, 1(AX) + MOVB BL, 3(AX) + ADDQ $0x04, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm + +three_bytes_emit_remainder_encodeSnappyBetterBlockAsm: + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm + +two_bytes_emit_remainder_encodeSnappyBetterBlockAsm: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBetterBlockAsm + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm + +one_byte_emit_remainder_encodeSnappyBetterBlockAsm: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBetterBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm + +memmove_long_emit_remainder_encodeSnappyBetterBlockAsm: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsmlarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsmlarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsmlarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBetterBlockAsm64K(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBetterBlockAsm64K(SB), $327704-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000a00, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBetterBlockAsm64K: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBetterBlockAsm64K + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBetterBlockAsm64K: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x07, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm64K + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x00cf1bbcdcbfa563, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x32, R11 + MOVL 24(SP)(R10*4), SI + MOVL 262168(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 262168(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm64K + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm64K + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm64K + +candidateS_match_encodeSnappyBetterBlockAsm64K: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x08, R10 + IMULQ R9, R10 + SHRQ $0x30, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm64K + DECL CX + MOVL R8, SI + +candidate_match_encodeSnappyBetterBlockAsm64K: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm64K + +match_extend_back_loop_encodeSnappyBetterBlockAsm64K: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBetterBlockAsm64K + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBetterBlockAsm64K + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm64K + JMP match_extend_back_loop_encodeSnappyBetterBlockAsm64K + +match_extend_back_end_encodeSnappyBetterBlockAsm64K: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBetterBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBetterBlockAsm64K: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm64K + +matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm64K: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm64K + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeSnappyBetterBlockAsm64K + +matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm64K: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm64K + JZ match_nolit_end_encodeSnappyBetterBlockAsm64K + +matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm64K: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm64K + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm64K + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm64K: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm64K + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm64K + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm64K: + CMPL R8, $0x01 + JL match_nolit_end_encodeSnappyBetterBlockAsm64K + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeSnappyBetterBlockAsm64K + LEAL 1(R12), R12 + +match_nolit_end_encodeSnappyBetterBlockAsm64K: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeSnappyBetterBlockAsm64K + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeSnappyBetterBlockAsm64K + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBetterBlockAsm64K + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm64K + +two_bytes_match_emit_encodeSnappyBetterBlockAsm64K: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeSnappyBetterBlockAsm64K + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm64K + +one_byte_match_emit_encodeSnappyBetterBlockAsm64K: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBetterBlockAsm64K: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm64K_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm64K: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeSnappyBetterBlockAsm64K + +memmove_long_match_emit_encodeSnappyBetterBlockAsm64K: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm64Klarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm64Klarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeSnappyBetterBlockAsm64K: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm64K: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm64K + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm64K + +two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm64K: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm64K + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm64K + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm64K + +emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm64K: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm64K: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm64K + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBetterBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBetterBlockAsm64K: + MOVQ $0x00cf1bbcdcbfa563, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x32, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 262168(SP)(R11*4) + MOVL R15, 262168(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x08, R10 + IMULQ SI, R10 + SHRQ $0x30, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x32, R11 + SHLQ $0x08, R13 + IMULQ SI, R13 + SHRQ $0x30, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 262168(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeSnappyBetterBlockAsm64K + +emit_remainder_encodeSnappyBetterBlockAsm64K: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBetterBlockAsm64K + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBetterBlockAsm64K: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm64K + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBetterBlockAsm64K + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBetterBlockAsm64K + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64K + +two_bytes_emit_remainder_encodeSnappyBetterBlockAsm64K: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBetterBlockAsm64K + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64K + +one_byte_emit_remainder_encodeSnappyBetterBlockAsm64K: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBetterBlockAsm64K: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm64K_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm64K: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm64K + +memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64K: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64Klarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64Klarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm64Klarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm64K: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBetterBlockAsm12B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBetterBlockAsm12B(SB), $81944-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000280, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBetterBlockAsm12B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBetterBlockAsm12B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBetterBlockAsm12B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x06, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm12B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x34, R11 + MOVL 24(SP)(R10*4), SI + MOVL 65560(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 65560(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm12B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm12B + +candidateS_match_encodeSnappyBetterBlockAsm12B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x32, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm12B + DECL CX + MOVL R8, SI + +candidate_match_encodeSnappyBetterBlockAsm12B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm12B + +match_extend_back_loop_encodeSnappyBetterBlockAsm12B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBetterBlockAsm12B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBetterBlockAsm12B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm12B + JMP match_extend_back_loop_encodeSnappyBetterBlockAsm12B + +match_extend_back_end_encodeSnappyBetterBlockAsm12B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBetterBlockAsm12B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm12B + +matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm12B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm12B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeSnappyBetterBlockAsm12B + +matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm12B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm12B + JZ match_nolit_end_encodeSnappyBetterBlockAsm12B + +matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm12B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm12B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm12B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm12B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm12B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm12B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm12B: + CMPL R8, $0x01 + JL match_nolit_end_encodeSnappyBetterBlockAsm12B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeSnappyBetterBlockAsm12B + LEAL 1(R12), R12 + +match_nolit_end_encodeSnappyBetterBlockAsm12B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeSnappyBetterBlockAsm12B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeSnappyBetterBlockAsm12B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBetterBlockAsm12B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm12B + +two_bytes_match_emit_encodeSnappyBetterBlockAsm12B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeSnappyBetterBlockAsm12B + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm12B + +one_byte_match_emit_encodeSnappyBetterBlockAsm12B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm12B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm12B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeSnappyBetterBlockAsm12B + +memmove_long_match_emit_encodeSnappyBetterBlockAsm12B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm12Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeSnappyBetterBlockAsm12B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm12B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm12B + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm12B + +two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm12B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm12B + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm12B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm12B + +emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm12B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm12B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm12B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBetterBlockAsm12B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x32, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x32, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x34, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x34, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 65560(SP)(R11*4) + MOVL R15, 65560(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x32, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x34, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x32, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 65560(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeSnappyBetterBlockAsm12B + +emit_remainder_encodeSnappyBetterBlockAsm12B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBetterBlockAsm12B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBetterBlockAsm12B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm12B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBetterBlockAsm12B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBetterBlockAsm12B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12B + +two_bytes_emit_remainder_encodeSnappyBetterBlockAsm12B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBetterBlockAsm12B + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12B + +one_byte_emit_remainder_encodeSnappyBetterBlockAsm12B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBetterBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm12B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm12B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm12B + +memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm12Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm12B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBetterBlockAsm10B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBetterBlockAsm10B(SB), $20504-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x000000a0, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBetterBlockAsm10B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBetterBlockAsm10B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBetterBlockAsm10B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x05, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm10B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x36, R11 + MOVL 24(SP)(R10*4), SI + MOVL 16408(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 16408(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm10B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm10B + +candidateS_match_encodeSnappyBetterBlockAsm10B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x34, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm10B + DECL CX + MOVL R8, SI + +candidate_match_encodeSnappyBetterBlockAsm10B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm10B + +match_extend_back_loop_encodeSnappyBetterBlockAsm10B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBetterBlockAsm10B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBetterBlockAsm10B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm10B + JMP match_extend_back_loop_encodeSnappyBetterBlockAsm10B + +match_extend_back_end_encodeSnappyBetterBlockAsm10B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBetterBlockAsm10B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm10B + +matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm10B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm10B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeSnappyBetterBlockAsm10B + +matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm10B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm10B + JZ match_nolit_end_encodeSnappyBetterBlockAsm10B + +matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm10B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm10B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm10B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm10B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm10B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm10B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm10B: + CMPL R8, $0x01 + JL match_nolit_end_encodeSnappyBetterBlockAsm10B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeSnappyBetterBlockAsm10B + LEAL 1(R12), R12 + +match_nolit_end_encodeSnappyBetterBlockAsm10B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeSnappyBetterBlockAsm10B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeSnappyBetterBlockAsm10B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBetterBlockAsm10B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm10B + +two_bytes_match_emit_encodeSnappyBetterBlockAsm10B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeSnappyBetterBlockAsm10B + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm10B + +one_byte_match_emit_encodeSnappyBetterBlockAsm10B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm10B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm10B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeSnappyBetterBlockAsm10B + +memmove_long_match_emit_encodeSnappyBetterBlockAsm10B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm10Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeSnappyBetterBlockAsm10B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm10B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm10B + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm10B + +two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm10B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm10B + CMPL R8, $0x00000800 + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm10B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm10B + +emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm10B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm10B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm10B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBetterBlockAsm10B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x34, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x34, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x36, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x36, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 16408(SP)(R11*4) + MOVL R15, 16408(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x34, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x36, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x34, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 16408(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeSnappyBetterBlockAsm10B + +emit_remainder_encodeSnappyBetterBlockAsm10B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBetterBlockAsm10B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBetterBlockAsm10B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm10B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBetterBlockAsm10B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBetterBlockAsm10B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10B + +two_bytes_emit_remainder_encodeSnappyBetterBlockAsm10B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBetterBlockAsm10B + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10B + +one_byte_emit_remainder_encodeSnappyBetterBlockAsm10B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBetterBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm10B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm10B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm10B + +memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm10Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm10B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func encodeSnappyBetterBlockAsm8B(dst []byte, src []byte) int +// Requires: BMI, SSE2 +TEXT ·encodeSnappyBetterBlockAsm8B(SB), $5144-56 + MOVQ dst_base+0(FP), AX + MOVQ $0x00000028, CX + LEAQ 24(SP), DX + PXOR X0, X0 + +zero_loop_encodeSnappyBetterBlockAsm8B: + MOVOU X0, (DX) + MOVOU X0, 16(DX) + MOVOU X0, 32(DX) + MOVOU X0, 48(DX) + MOVOU X0, 64(DX) + MOVOU X0, 80(DX) + MOVOU X0, 96(DX) + MOVOU X0, 112(DX) + ADDQ $0x80, DX + DECQ CX + JNZ zero_loop_encodeSnappyBetterBlockAsm8B + MOVL $0x00000000, 12(SP) + MOVQ src_len+32(FP), CX + LEAQ -9(CX), DX + LEAQ -8(CX), SI + MOVL SI, 8(SP) + SHRQ $0x05, CX + SUBL CX, DX + LEAQ (AX)(DX*1), DX + MOVQ DX, (SP) + MOVL $0x00000001, CX + MOVL $0x00000000, 16(SP) + MOVQ src_base+24(FP), DX + +search_loop_encodeSnappyBetterBlockAsm8B: + MOVL CX, SI + SUBL 12(SP), SI + SHRL $0x04, SI + LEAL 1(CX)(SI*1), SI + CMPL SI, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm8B + MOVQ (DX)(CX*1), DI + MOVL SI, 20(SP) + MOVQ $0x0000cf1bbcdcbf9b, R9 + MOVQ $0x9e3779b1, SI + MOVQ DI, R10 + MOVQ DI, R11 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ SI, R11 + SHRQ $0x38, R11 + MOVL 24(SP)(R10*4), SI + MOVL 4120(SP)(R11*4), R8 + MOVL CX, 24(SP)(R10*4) + MOVL CX, 4120(SP)(R11*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm8B + CMPL (DX)(R8*1), DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm8B + +candidateS_match_encodeSnappyBetterBlockAsm8B: + SHRQ $0x08, DI + MOVQ DI, R10 + SHLQ $0x10, R10 + IMULQ R9, R10 + SHRQ $0x36, R10 + MOVL 24(SP)(R10*4), SI + INCL CX + MOVL CX, 24(SP)(R10*4) + CMPL (DX)(SI*1), DI + JEQ candidate_match_encodeSnappyBetterBlockAsm8B + DECL CX + MOVL R8, SI + +candidate_match_encodeSnappyBetterBlockAsm8B: + MOVL 12(SP), DI + TESTL SI, SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm8B + +match_extend_back_loop_encodeSnappyBetterBlockAsm8B: + CMPL CX, DI + JLE match_extend_back_end_encodeSnappyBetterBlockAsm8B + MOVB -1(DX)(SI*1), BL + MOVB -1(DX)(CX*1), R8 + CMPB BL, R8 + JNE match_extend_back_end_encodeSnappyBetterBlockAsm8B + LEAL -1(CX), CX + DECL SI + JZ match_extend_back_end_encodeSnappyBetterBlockAsm8B + JMP match_extend_back_loop_encodeSnappyBetterBlockAsm8B + +match_extend_back_end_encodeSnappyBetterBlockAsm8B: + MOVL CX, DI + SUBL 12(SP), DI + LEAQ 3(AX)(DI*1), DI + CMPQ DI, (SP) + JL match_dst_size_check_encodeSnappyBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_dst_size_check_encodeSnappyBetterBlockAsm8B: + MOVL CX, DI + ADDL $0x04, CX + ADDL $0x04, SI + MOVQ src_len+32(FP), R8 + SUBL CX, R8 + LEAQ (DX)(CX*1), R9 + LEAQ (DX)(SI*1), R10 + + // matchLen + XORL R12, R12 + CMPL R8, $0x08 + JL matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm8B + +matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm8B: + MOVQ (R9)(R12*1), R11 + XORQ (R10)(R12*1), R11 + TESTQ R11, R11 + JZ matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm8B + +#ifdef GOAMD64_v3 + TZCNTQ R11, R11 + +#else + BSFQ R11, R11 + +#endif + SARQ $0x03, R11 + LEAL (R12)(R11*1), R12 + JMP match_nolit_end_encodeSnappyBetterBlockAsm8B + +matchlen_loop_match_nolit_encodeSnappyBetterBlockAsm8B: + LEAL -8(R8), R8 + LEAL 8(R12), R12 + CMPL R8, $0x08 + JGE matchlen_loopback_match_nolit_encodeSnappyBetterBlockAsm8B + JZ match_nolit_end_encodeSnappyBetterBlockAsm8B + +matchlen_match4_match_nolit_encodeSnappyBetterBlockAsm8B: + CMPL R8, $0x04 + JL matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm8B + MOVL (R9)(R12*1), R11 + CMPL (R10)(R12*1), R11 + JNE matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm8B + SUBL $0x04, R8 + LEAL 4(R12), R12 + +matchlen_match2_match_nolit_encodeSnappyBetterBlockAsm8B: + CMPL R8, $0x02 + JL matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm8B + MOVW (R9)(R12*1), R11 + CMPW (R10)(R12*1), R11 + JNE matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm8B + SUBL $0x02, R8 + LEAL 2(R12), R12 + +matchlen_match1_match_nolit_encodeSnappyBetterBlockAsm8B: + CMPL R8, $0x01 + JL match_nolit_end_encodeSnappyBetterBlockAsm8B + MOVB (R9)(R12*1), R11 + CMPB (R10)(R12*1), R11 + JNE match_nolit_end_encodeSnappyBetterBlockAsm8B + LEAL 1(R12), R12 + +match_nolit_end_encodeSnappyBetterBlockAsm8B: + MOVL CX, R8 + SUBL SI, R8 + + // Check if repeat + MOVL R8, 16(SP) + MOVL 12(SP), SI + CMPL SI, DI + JEQ emit_literal_done_match_emit_encodeSnappyBetterBlockAsm8B + MOVL DI, R9 + MOVL DI, 12(SP) + LEAQ (DX)(SI*1), R10 + SUBL SI, R9 + LEAL -1(R9), SI + CMPL SI, $0x3c + JLT one_byte_match_emit_encodeSnappyBetterBlockAsm8B + CMPL SI, $0x00000100 + JLT two_bytes_match_emit_encodeSnappyBetterBlockAsm8B + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm8B + +two_bytes_match_emit_encodeSnappyBetterBlockAsm8B: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_match_emit_encodeSnappyBetterBlockAsm8B + JMP memmove_long_match_emit_encodeSnappyBetterBlockAsm8B + +one_byte_match_emit_encodeSnappyBetterBlockAsm8B: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, AX + +memmove_match_emit_encodeSnappyBetterBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveShort + CMPQ R9, $0x08 + JLE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_8 + CMPQ R9, $0x10 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_8through16 + CMPQ R9, $0x20 + JBE emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_8: + MOVQ (R10), R11 + MOVQ R11, (AX) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_8through16: + MOVQ (R10), R11 + MOVQ -8(R10)(R9*1), R10 + MOVQ R11, (AX) + MOVQ R10, -8(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_17through32: + MOVOU (R10), X0 + MOVOU -16(R10)(R9*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(R9*1) + JMP memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_match_emit_encodeSnappyBetterBlockAsm8B_memmove_move_33through64: + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + +memmove_end_copy_match_emit_encodeSnappyBetterBlockAsm8B: + MOVQ SI, AX + JMP emit_literal_done_match_emit_encodeSnappyBetterBlockAsm8B + +memmove_long_match_emit_encodeSnappyBetterBlockAsm8B: + LEAQ (AX)(R9*1), SI + + // genMemMoveLong + MOVOU (R10), X0 + MOVOU 16(R10), X1 + MOVOU -32(R10)(R9*1), X2 + MOVOU -16(R10)(R9*1), X3 + MOVQ R9, R13 + SHRQ $0x05, R13 + MOVQ AX, R11 + ANDL $0x0000001f, R11 + MOVQ $0x00000040, R14 + SUBQ R11, R14 + DECQ R13 + JA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(R10)(R14*1), R11 + LEAQ -32(AX)(R14*1), R15 + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm8Blarge_big_loop_back: + MOVOU (R11), X4 + MOVOU 16(R11), X5 + MOVOA X4, (R15) + MOVOA X5, 16(R15) + ADDQ $0x20, R15 + ADDQ $0x20, R11 + ADDQ $0x20, R14 + DECQ R13 + JNA emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(R10)(R14*1), X4 + MOVOU -16(R10)(R14*1), X5 + MOVOA X4, -32(AX)(R14*1) + MOVOA X5, -16(AX)(R14*1) + ADDQ $0x20, R14 + CMPQ R9, R14 + JAE emit_lit_memmove_long_match_emit_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(R9*1) + MOVOU X3, -16(AX)(R9*1) + MOVQ SI, AX + +emit_literal_done_match_emit_encodeSnappyBetterBlockAsm8B: + ADDL R12, CX + ADDL $0x04, R12 + MOVL CX, 12(SP) + + // emitCopy +two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm8B: + CMPL R12, $0x40 + JLE two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm8B + MOVB $0xee, (AX) + MOVW R8, 1(AX) + LEAL -60(R12), R12 + ADDQ $0x03, AX + JMP two_byte_offset_match_nolit_encodeSnappyBetterBlockAsm8B + +two_byte_offset_short_match_nolit_encodeSnappyBetterBlockAsm8B: + CMPL R12, $0x0c + JGE emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm8B + MOVB $0x01, BL + LEAL -16(BX)(R12*4), R12 + MOVB R8, 1(AX) + SHRL $0x08, R8 + SHLL $0x05, R8 + ORL R8, R12 + MOVB R12, (AX) + ADDQ $0x02, AX + JMP match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm8B + +emit_copy_three_match_nolit_encodeSnappyBetterBlockAsm8B: + MOVB $0x02, BL + LEAL -4(BX)(R12*4), R12 + MOVB R12, (AX) + MOVW R8, 1(AX) + ADDQ $0x03, AX + +match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm8B: + CMPL CX, 8(SP) + JGE emit_remainder_encodeSnappyBetterBlockAsm8B + CMPQ AX, (SP) + JL match_nolit_dst_ok_encodeSnappyBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +match_nolit_dst_ok_encodeSnappyBetterBlockAsm8B: + MOVQ $0x0000cf1bbcdcbf9b, SI + MOVQ $0x9e3779b1, R8 + INCL DI + MOVQ (DX)(DI*1), R9 + MOVQ R9, R10 + MOVQ R9, R11 + MOVQ R9, R12 + SHRQ $0x08, R11 + MOVQ R11, R13 + SHRQ $0x10, R12 + LEAL 1(DI), R14 + LEAL 2(DI), R15 + MOVQ -2(DX)(CX*1), R9 + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x36, R10 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x36, R13 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x38, R11 + SHLQ $0x20, R12 + IMULQ R8, R12 + SHRQ $0x38, R12 + MOVL DI, 24(SP)(R10*4) + MOVL R14, 24(SP)(R13*4) + MOVL R14, 4120(SP)(R11*4) + MOVL R15, 4120(SP)(R12*4) + MOVQ R9, R10 + MOVQ R9, R11 + SHRQ $0x08, R11 + MOVQ R11, R13 + LEAL -2(CX), R9 + LEAL -1(CX), DI + SHLQ $0x10, R10 + IMULQ SI, R10 + SHRQ $0x36, R10 + SHLQ $0x20, R11 + IMULQ R8, R11 + SHRQ $0x38, R11 + SHLQ $0x10, R13 + IMULQ SI, R13 + SHRQ $0x36, R13 + MOVL R9, 24(SP)(R10*4) + MOVL DI, 4120(SP)(R11*4) + MOVL DI, 24(SP)(R13*4) + JMP search_loop_encodeSnappyBetterBlockAsm8B + +emit_remainder_encodeSnappyBetterBlockAsm8B: + MOVQ src_len+32(FP), CX + SUBL 12(SP), CX + LEAQ 3(AX)(CX*1), CX + CMPQ CX, (SP) + JL emit_remainder_ok_encodeSnappyBetterBlockAsm8B + MOVQ $0x00000000, ret+48(FP) + RET + +emit_remainder_ok_encodeSnappyBetterBlockAsm8B: + MOVQ src_len+32(FP), CX + MOVL 12(SP), BX + CMPL BX, CX + JEQ emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm8B + MOVL CX, SI + MOVL CX, 12(SP) + LEAQ (DX)(BX*1), CX + SUBL BX, SI + LEAL -1(SI), DX + CMPL DX, $0x3c + JLT one_byte_emit_remainder_encodeSnappyBetterBlockAsm8B + CMPL DX, $0x00000100 + JLT two_bytes_emit_remainder_encodeSnappyBetterBlockAsm8B + MOVB $0xf4, (AX) + MOVW DX, 1(AX) + ADDQ $0x03, AX + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8B + +two_bytes_emit_remainder_encodeSnappyBetterBlockAsm8B: + MOVB $0xf0, (AX) + MOVB DL, 1(AX) + ADDQ $0x02, AX + CMPL DX, $0x40 + JL memmove_emit_remainder_encodeSnappyBetterBlockAsm8B + JMP memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8B + +one_byte_emit_remainder_encodeSnappyBetterBlockAsm8B: + SHLB $0x02, DL + MOVB DL, (AX) + ADDQ $0x01, AX + +memmove_emit_remainder_encodeSnappyBetterBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveShort + CMPQ BX, $0x03 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_1or2 + JE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_3 + CMPQ BX, $0x08 + JB emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_4through7 + CMPQ BX, $0x10 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_8through16 + CMPQ BX, $0x20 + JBE emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_17through32 + JMP emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_33through64 + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(BX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(BX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(BX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(BX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(BX*1) + JMP memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B + +emit_lit_memmove_emit_remainder_encodeSnappyBetterBlockAsm8B_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + +memmove_end_copy_emit_remainder_encodeSnappyBetterBlockAsm8B: + MOVQ DX, AX + JMP emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm8B + +memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8B: + LEAQ (AX)(SI*1), DX + MOVL SI, BX + + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(BX*1), X2 + MOVOU -16(CX)(BX*1), X3 + MOVQ BX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8Blarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8Blarge_big_loop_back + +emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ BX, R8 + JAE emit_lit_memmove_long_emit_remainder_encodeSnappyBetterBlockAsm8Blarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(BX*1) + MOVOU X3, -16(AX)(BX*1) + MOVQ DX, AX + +emit_literal_done_emit_remainder_encodeSnappyBetterBlockAsm8B: + MOVQ dst_base+0(FP), CX + SUBQ CX, AX + MOVQ AX, ret+48(FP) + RET + +// func emitLiteral(dst []byte, lit []byte) int +// Requires: SSE2 +TEXT ·emitLiteral(SB), NOSPLIT, $0-56 + MOVQ lit_len+32(FP), DX + MOVQ dst_base+0(FP), AX + MOVQ lit_base+24(FP), CX + TESTQ DX, DX + JZ emit_literal_end_standalone_skip + MOVL DX, BX + LEAL -1(DX), SI + CMPL SI, $0x3c + JLT one_byte_standalone + CMPL SI, $0x00000100 + JLT two_bytes_standalone + CMPL SI, $0x00010000 + JLT three_bytes_standalone + CMPL SI, $0x01000000 + JLT four_bytes_standalone + MOVB $0xfc, (AX) + MOVL SI, 1(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP memmove_long_standalone + +four_bytes_standalone: + MOVL SI, DI + SHRL $0x10, DI + MOVB $0xf8, (AX) + MOVW SI, 1(AX) + MOVB DI, 3(AX) + ADDQ $0x04, BX + ADDQ $0x04, AX + JMP memmove_long_standalone + +three_bytes_standalone: + MOVB $0xf4, (AX) + MOVW SI, 1(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + JMP memmove_long_standalone + +two_bytes_standalone: + MOVB $0xf0, (AX) + MOVB SI, 1(AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + CMPL SI, $0x40 + JL memmove_standalone + JMP memmove_long_standalone + +one_byte_standalone: + SHLB $0x02, SI + MOVB SI, (AX) + ADDQ $0x01, BX + ADDQ $0x01, AX + +memmove_standalone: + // genMemMoveShort + CMPQ DX, $0x03 + JB emit_lit_memmove_standalone_memmove_move_1or2 + JE emit_lit_memmove_standalone_memmove_move_3 + CMPQ DX, $0x08 + JB emit_lit_memmove_standalone_memmove_move_4through7 + CMPQ DX, $0x10 + JBE emit_lit_memmove_standalone_memmove_move_8through16 + CMPQ DX, $0x20 + JBE emit_lit_memmove_standalone_memmove_move_17through32 + JMP emit_lit_memmove_standalone_memmove_move_33through64 + +emit_lit_memmove_standalone_memmove_move_1or2: + MOVB (CX), SI + MOVB -1(CX)(DX*1), CL + MOVB SI, (AX) + MOVB CL, -1(AX)(DX*1) + JMP emit_literal_end_standalone + +emit_lit_memmove_standalone_memmove_move_3: + MOVW (CX), SI + MOVB 2(CX), CL + MOVW SI, (AX) + MOVB CL, 2(AX) + JMP emit_literal_end_standalone + +emit_lit_memmove_standalone_memmove_move_4through7: + MOVL (CX), SI + MOVL -4(CX)(DX*1), CX + MOVL SI, (AX) + MOVL CX, -4(AX)(DX*1) + JMP emit_literal_end_standalone + +emit_lit_memmove_standalone_memmove_move_8through16: + MOVQ (CX), SI + MOVQ -8(CX)(DX*1), CX + MOVQ SI, (AX) + MOVQ CX, -8(AX)(DX*1) + JMP emit_literal_end_standalone + +emit_lit_memmove_standalone_memmove_move_17through32: + MOVOU (CX), X0 + MOVOU -16(CX)(DX*1), X1 + MOVOU X0, (AX) + MOVOU X1, -16(AX)(DX*1) + JMP emit_literal_end_standalone + +emit_lit_memmove_standalone_memmove_move_33through64: + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(DX*1), X2 + MOVOU -16(CX)(DX*1), X3 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(DX*1) + MOVOU X3, -16(AX)(DX*1) + JMP emit_literal_end_standalone + JMP emit_literal_end_standalone + +memmove_long_standalone: + // genMemMoveLong + MOVOU (CX), X0 + MOVOU 16(CX), X1 + MOVOU -32(CX)(DX*1), X2 + MOVOU -16(CX)(DX*1), X3 + MOVQ DX, DI + SHRQ $0x05, DI + MOVQ AX, SI + ANDL $0x0000001f, SI + MOVQ $0x00000040, R8 + SUBQ SI, R8 + DECQ DI + JA emit_lit_memmove_long_standalonelarge_forward_sse_loop_32 + LEAQ -32(CX)(R8*1), SI + LEAQ -32(AX)(R8*1), R9 + +emit_lit_memmove_long_standalonelarge_big_loop_back: + MOVOU (SI), X4 + MOVOU 16(SI), X5 + MOVOA X4, (R9) + MOVOA X5, 16(R9) + ADDQ $0x20, R9 + ADDQ $0x20, SI + ADDQ $0x20, R8 + DECQ DI + JNA emit_lit_memmove_long_standalonelarge_big_loop_back + +emit_lit_memmove_long_standalonelarge_forward_sse_loop_32: + MOVOU -32(CX)(R8*1), X4 + MOVOU -16(CX)(R8*1), X5 + MOVOA X4, -32(AX)(R8*1) + MOVOA X5, -16(AX)(R8*1) + ADDQ $0x20, R8 + CMPQ DX, R8 + JAE emit_lit_memmove_long_standalonelarge_forward_sse_loop_32 + MOVOU X0, (AX) + MOVOU X1, 16(AX) + MOVOU X2, -32(AX)(DX*1) + MOVOU X3, -16(AX)(DX*1) + JMP emit_literal_end_standalone + JMP emit_literal_end_standalone + +emit_literal_end_standalone_skip: + XORQ BX, BX + +emit_literal_end_standalone: + MOVQ BX, ret+48(FP) + RET + +// func emitRepeat(dst []byte, offset int, length int) int +TEXT ·emitRepeat(SB), NOSPLIT, $0-48 + XORQ BX, BX + MOVQ dst_base+0(FP), AX + MOVQ offset+24(FP), CX + MOVQ length+32(FP), DX + + // emitRepeat +emit_repeat_again_standalone: + MOVL DX, SI + LEAL -4(DX), DX + CMPL SI, $0x08 + JLE repeat_two_standalone + CMPL SI, $0x0c + JGE cant_repeat_two_offset_standalone + CMPL CX, $0x00000800 + JLT repeat_two_offset_standalone + +cant_repeat_two_offset_standalone: + CMPL DX, $0x00000104 + JLT repeat_three_standalone + CMPL DX, $0x00010100 + JLT repeat_four_standalone + CMPL DX, $0x0100ffff + JLT repeat_five_standalone + LEAL -16842747(DX), DX + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + ADDQ $0x05, BX + JMP emit_repeat_again_standalone + +repeat_five_standalone: + LEAL -65536(DX), DX + MOVL DX, CX + MOVW $0x001d, (AX) + MOVW DX, 2(AX) + SARL $0x10, CX + MOVB CL, 4(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_repeat_end + +repeat_four_standalone: + LEAL -256(DX), DX + MOVW $0x0019, (AX) + MOVW DX, 2(AX) + ADDQ $0x04, BX + ADDQ $0x04, AX + JMP gen_emit_repeat_end + +repeat_three_standalone: + LEAL -4(DX), DX + MOVW $0x0015, (AX) + MOVB DL, 2(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + JMP gen_emit_repeat_end + +repeat_two_standalone: + SHLL $0x02, DX + ORL $0x01, DX + MOVW DX, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_repeat_end + +repeat_two_offset_standalone: + XORQ SI, SI + LEAL 1(SI)(DX*4), DX + MOVB CL, 1(AX) + SARL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + +gen_emit_repeat_end: + MOVQ BX, ret+40(FP) + RET + +// func emitCopy(dst []byte, offset int, length int) int +TEXT ·emitCopy(SB), NOSPLIT, $0-48 + XORQ BX, BX + MOVQ dst_base+0(FP), AX + MOVQ offset+24(FP), CX + MOVQ length+32(FP), DX + + // emitCopy + CMPL CX, $0x00010000 + JL two_byte_offset_standalone + +four_bytes_loop_back_standalone: + CMPL DX, $0x40 + JLE four_bytes_remain_standalone + MOVB $0xff, (AX) + MOVL CX, 1(AX) + LEAL -64(DX), DX + ADDQ $0x05, BX + ADDQ $0x05, AX + CMPL DX, $0x04 + JL four_bytes_remain_standalone + + // emitRepeat +emit_repeat_again_standalone_emit_copy: + MOVL DX, SI + LEAL -4(DX), DX + CMPL SI, $0x08 + JLE repeat_two_standalone_emit_copy + CMPL SI, $0x0c + JGE cant_repeat_two_offset_standalone_emit_copy + CMPL CX, $0x00000800 + JLT repeat_two_offset_standalone_emit_copy + +cant_repeat_two_offset_standalone_emit_copy: + CMPL DX, $0x00000104 + JLT repeat_three_standalone_emit_copy + CMPL DX, $0x00010100 + JLT repeat_four_standalone_emit_copy + CMPL DX, $0x0100ffff + JLT repeat_five_standalone_emit_copy + LEAL -16842747(DX), DX + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + ADDQ $0x05, BX + JMP emit_repeat_again_standalone_emit_copy + +repeat_five_standalone_emit_copy: + LEAL -65536(DX), DX + MOVL DX, CX + MOVW $0x001d, (AX) + MOVW DX, 2(AX) + SARL $0x10, CX + MOVB CL, 4(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_copy_end + +repeat_four_standalone_emit_copy: + LEAL -256(DX), DX + MOVW $0x0019, (AX) + MOVW DX, 2(AX) + ADDQ $0x04, BX + ADDQ $0x04, AX + JMP gen_emit_copy_end + +repeat_three_standalone_emit_copy: + LEAL -4(DX), DX + MOVW $0x0015, (AX) + MOVB DL, 2(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + JMP gen_emit_copy_end + +repeat_two_standalone_emit_copy: + SHLL $0x02, DX + ORL $0x01, DX + MOVW DX, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + +repeat_two_offset_standalone_emit_copy: + XORQ SI, SI + LEAL 1(SI)(DX*4), DX + MOVB CL, 1(AX) + SARL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + JMP four_bytes_loop_back_standalone + +four_bytes_remain_standalone: + TESTL DX, DX + JZ gen_emit_copy_end + MOVB $0x03, SI + LEAL -4(SI)(DX*4), DX + MOVB DL, (AX) + MOVL CX, 1(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_copy_end + +two_byte_offset_standalone: + CMPL DX, $0x40 + JLE two_byte_offset_short_standalone + CMPL CX, $0x00000800 + JAE long_offset_short_standalone + MOVL $0x00000001, SI + LEAL 16(SI), SI + MOVB CL, 1(AX) + MOVL CX, DI + SHRL $0x08, DI + SHLL $0x05, DI + ORL DI, SI + MOVB SI, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + SUBL $0x08, DX + + // emitRepeat + LEAL -4(DX), DX + JMP cant_repeat_two_offset_standalone_emit_copy_short_2b + +emit_repeat_again_standalone_emit_copy_short_2b: + MOVL DX, SI + LEAL -4(DX), DX + CMPL SI, $0x08 + JLE repeat_two_standalone_emit_copy_short_2b + CMPL SI, $0x0c + JGE cant_repeat_two_offset_standalone_emit_copy_short_2b + CMPL CX, $0x00000800 + JLT repeat_two_offset_standalone_emit_copy_short_2b + +cant_repeat_two_offset_standalone_emit_copy_short_2b: + CMPL DX, $0x00000104 + JLT repeat_three_standalone_emit_copy_short_2b + CMPL DX, $0x00010100 + JLT repeat_four_standalone_emit_copy_short_2b + CMPL DX, $0x0100ffff + JLT repeat_five_standalone_emit_copy_short_2b + LEAL -16842747(DX), DX + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + ADDQ $0x05, BX + JMP emit_repeat_again_standalone_emit_copy_short_2b + +repeat_five_standalone_emit_copy_short_2b: + LEAL -65536(DX), DX + MOVL DX, CX + MOVW $0x001d, (AX) + MOVW DX, 2(AX) + SARL $0x10, CX + MOVB CL, 4(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_copy_end + +repeat_four_standalone_emit_copy_short_2b: + LEAL -256(DX), DX + MOVW $0x0019, (AX) + MOVW DX, 2(AX) + ADDQ $0x04, BX + ADDQ $0x04, AX + JMP gen_emit_copy_end + +repeat_three_standalone_emit_copy_short_2b: + LEAL -4(DX), DX + MOVW $0x0015, (AX) + MOVB DL, 2(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + JMP gen_emit_copy_end + +repeat_two_standalone_emit_copy_short_2b: + SHLL $0x02, DX + ORL $0x01, DX + MOVW DX, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + +repeat_two_offset_standalone_emit_copy_short_2b: + XORQ SI, SI + LEAL 1(SI)(DX*4), DX + MOVB CL, 1(AX) + SARL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + +long_offset_short_standalone: + MOVB $0xee, (AX) + MOVW CX, 1(AX) + LEAL -60(DX), DX + ADDQ $0x03, AX + ADDQ $0x03, BX + + // emitRepeat +emit_repeat_again_standalone_emit_copy_short: + MOVL DX, SI + LEAL -4(DX), DX + CMPL SI, $0x08 + JLE repeat_two_standalone_emit_copy_short + CMPL SI, $0x0c + JGE cant_repeat_two_offset_standalone_emit_copy_short + CMPL CX, $0x00000800 + JLT repeat_two_offset_standalone_emit_copy_short + +cant_repeat_two_offset_standalone_emit_copy_short: + CMPL DX, $0x00000104 + JLT repeat_three_standalone_emit_copy_short + CMPL DX, $0x00010100 + JLT repeat_four_standalone_emit_copy_short + CMPL DX, $0x0100ffff + JLT repeat_five_standalone_emit_copy_short + LEAL -16842747(DX), DX + MOVW $0x001d, (AX) + MOVW $0xfffb, 2(AX) + MOVB $0xff, 4(AX) + ADDQ $0x05, AX + ADDQ $0x05, BX + JMP emit_repeat_again_standalone_emit_copy_short + +repeat_five_standalone_emit_copy_short: + LEAL -65536(DX), DX + MOVL DX, CX + MOVW $0x001d, (AX) + MOVW DX, 2(AX) + SARL $0x10, CX + MOVB CL, 4(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_copy_end + +repeat_four_standalone_emit_copy_short: + LEAL -256(DX), DX + MOVW $0x0019, (AX) + MOVW DX, 2(AX) + ADDQ $0x04, BX + ADDQ $0x04, AX + JMP gen_emit_copy_end + +repeat_three_standalone_emit_copy_short: + LEAL -4(DX), DX + MOVW $0x0015, (AX) + MOVB DL, 2(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + JMP gen_emit_copy_end + +repeat_two_standalone_emit_copy_short: + SHLL $0x02, DX + ORL $0x01, DX + MOVW DX, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + +repeat_two_offset_standalone_emit_copy_short: + XORQ SI, SI + LEAL 1(SI)(DX*4), DX + MOVB CL, 1(AX) + SARL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + JMP two_byte_offset_standalone + +two_byte_offset_short_standalone: + CMPL DX, $0x0c + JGE emit_copy_three_standalone + CMPL CX, $0x00000800 + JGE emit_copy_three_standalone + MOVB $0x01, SI + LEAL -16(SI)(DX*4), DX + MOVB CL, 1(AX) + SHRL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end + +emit_copy_three_standalone: + MOVB $0x02, SI + LEAL -4(SI)(DX*4), DX + MOVB DL, (AX) + MOVW CX, 1(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + +gen_emit_copy_end: + MOVQ BX, ret+40(FP) + RET + +// func emitCopyNoRepeat(dst []byte, offset int, length int) int +TEXT ·emitCopyNoRepeat(SB), NOSPLIT, $0-48 + XORQ BX, BX + MOVQ dst_base+0(FP), AX + MOVQ offset+24(FP), CX + MOVQ length+32(FP), DX + + // emitCopy + CMPL CX, $0x00010000 + JL two_byte_offset_standalone_snappy + +four_bytes_loop_back_standalone_snappy: + CMPL DX, $0x40 + JLE four_bytes_remain_standalone_snappy + MOVB $0xff, (AX) + MOVL CX, 1(AX) + LEAL -64(DX), DX + ADDQ $0x05, BX + ADDQ $0x05, AX + CMPL DX, $0x04 + JL four_bytes_remain_standalone_snappy + JMP four_bytes_loop_back_standalone_snappy + +four_bytes_remain_standalone_snappy: + TESTL DX, DX + JZ gen_emit_copy_end_snappy + MOVB $0x03, SI + LEAL -4(SI)(DX*4), DX + MOVB DL, (AX) + MOVL CX, 1(AX) + ADDQ $0x05, BX + ADDQ $0x05, AX + JMP gen_emit_copy_end_snappy + +two_byte_offset_standalone_snappy: + CMPL DX, $0x40 + JLE two_byte_offset_short_standalone_snappy + MOVB $0xee, (AX) + MOVW CX, 1(AX) + LEAL -60(DX), DX + ADDQ $0x03, AX + ADDQ $0x03, BX + JMP two_byte_offset_standalone_snappy + +two_byte_offset_short_standalone_snappy: + CMPL DX, $0x0c + JGE emit_copy_three_standalone_snappy + CMPL CX, $0x00000800 + JGE emit_copy_three_standalone_snappy + MOVB $0x01, SI + LEAL -16(SI)(DX*4), DX + MOVB CL, 1(AX) + SHRL $0x08, CX + SHLL $0x05, CX + ORL CX, DX + MOVB DL, (AX) + ADDQ $0x02, BX + ADDQ $0x02, AX + JMP gen_emit_copy_end_snappy + +emit_copy_three_standalone_snappy: + MOVB $0x02, SI + LEAL -4(SI)(DX*4), DX + MOVB DL, (AX) + MOVW CX, 1(AX) + ADDQ $0x03, BX + ADDQ $0x03, AX + +gen_emit_copy_end_snappy: + MOVQ BX, ret+40(FP) + RET + +// func matchLen(a []byte, b []byte) int +// Requires: BMI +TEXT ·matchLen(SB), NOSPLIT, $0-56 + MOVQ a_base+0(FP), AX + MOVQ b_base+24(FP), CX + MOVQ a_len+8(FP), DX + + // matchLen + XORL SI, SI + CMPL DX, $0x08 + JL matchlen_match4_standalone + +matchlen_loopback_standalone: + MOVQ (AX)(SI*1), BX + XORQ (CX)(SI*1), BX + TESTQ BX, BX + JZ matchlen_loop_standalone + +#ifdef GOAMD64_v3 + TZCNTQ BX, BX + +#else + BSFQ BX, BX + +#endif + SARQ $0x03, BX + LEAL (SI)(BX*1), SI + JMP gen_match_len_end + +matchlen_loop_standalone: + LEAL -8(DX), DX + LEAL 8(SI), SI + CMPL DX, $0x08 + JGE matchlen_loopback_standalone + JZ gen_match_len_end + +matchlen_match4_standalone: + CMPL DX, $0x04 + JL matchlen_match2_standalone + MOVL (AX)(SI*1), BX + CMPL (CX)(SI*1), BX + JNE matchlen_match2_standalone + SUBL $0x04, DX + LEAL 4(SI), SI + +matchlen_match2_standalone: + CMPL DX, $0x02 + JL matchlen_match1_standalone + MOVW (AX)(SI*1), BX + CMPW (CX)(SI*1), BX + JNE matchlen_match1_standalone + SUBL $0x02, DX + LEAL 2(SI), SI + +matchlen_match1_standalone: + CMPL DX, $0x01 + JL gen_match_len_end + MOVB (AX)(SI*1), BL + CMPB (CX)(SI*1), BL + JNE gen_match_len_end + LEAL 1(SI), SI + +gen_match_len_end: + MOVQ SI, ret+48(FP) + RET diff --git a/vendor/github.com/klauspost/compress/s2/index.go b/vendor/github.com/klauspost/compress/s2/index.go new file mode 100644 index 000000000..dd9ecfe71 --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/index.go @@ -0,0 +1,598 @@ +// Copyright (c) 2022+ Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package s2 + +import ( + "bytes" + "encoding/binary" + "encoding/json" + "fmt" + "io" + "sort" +) + +const ( + S2IndexHeader = "s2idx\x00" + S2IndexTrailer = "\x00xdi2s" + maxIndexEntries = 1 << 16 +) + +// Index represents an S2/Snappy index. +type Index struct { + TotalUncompressed int64 // Total Uncompressed size if known. Will be -1 if unknown. + TotalCompressed int64 // Total Compressed size if known. Will be -1 if unknown. + info []struct { + compressedOffset int64 + uncompressedOffset int64 + } + estBlockUncomp int64 +} + +func (i *Index) reset(maxBlock int) { + i.estBlockUncomp = int64(maxBlock) + i.TotalCompressed = -1 + i.TotalUncompressed = -1 + if len(i.info) > 0 { + i.info = i.info[:0] + } +} + +// allocInfos will allocate an empty slice of infos. +func (i *Index) allocInfos(n int) { + if n > maxIndexEntries { + panic("n > maxIndexEntries") + } + i.info = make([]struct { + compressedOffset int64 + uncompressedOffset int64 + }, 0, n) +} + +// add an uncompressed and compressed pair. +// Entries must be sent in order. +func (i *Index) add(compressedOffset, uncompressedOffset int64) error { + if i == nil { + return nil + } + lastIdx := len(i.info) - 1 + if lastIdx >= 0 { + latest := i.info[lastIdx] + if latest.uncompressedOffset == uncompressedOffset { + // Uncompressed didn't change, don't add entry, + // but update start index. + latest.compressedOffset = compressedOffset + i.info[lastIdx] = latest + return nil + } + if latest.uncompressedOffset > uncompressedOffset { + return fmt.Errorf("internal error: Earlier uncompressed received (%d > %d)", latest.uncompressedOffset, uncompressedOffset) + } + if latest.compressedOffset > compressedOffset { + return fmt.Errorf("internal error: Earlier compressed received (%d > %d)", latest.uncompressedOffset, uncompressedOffset) + } + } + i.info = append(i.info, struct { + compressedOffset int64 + uncompressedOffset int64 + }{compressedOffset: compressedOffset, uncompressedOffset: uncompressedOffset}) + return nil +} + +// Find the offset at or before the wanted (uncompressed) offset. +// If offset is 0 or positive it is the offset from the beginning of the file. +// If the uncompressed size is known, the offset must be within the file. +// If an offset outside the file is requested io.ErrUnexpectedEOF is returned. +// If the offset is negative, it is interpreted as the distance from the end of the file, +// where -1 represents the last byte. +// If offset from the end of the file is requested, but size is unknown, +// ErrUnsupported will be returned. +func (i *Index) Find(offset int64) (compressedOff, uncompressedOff int64, err error) { + if i.TotalUncompressed < 0 { + return 0, 0, ErrCorrupt + } + if offset < 0 { + offset = i.TotalUncompressed + offset + if offset < 0 { + return 0, 0, io.ErrUnexpectedEOF + } + } + if offset > i.TotalUncompressed { + return 0, 0, io.ErrUnexpectedEOF + } + if len(i.info) > 200 { + n := sort.Search(len(i.info), func(n int) bool { + return i.info[n].uncompressedOffset > offset + }) + if n == 0 { + n = 1 + } + return i.info[n-1].compressedOffset, i.info[n-1].uncompressedOffset, nil + } + for _, info := range i.info { + if info.uncompressedOffset > offset { + break + } + compressedOff = info.compressedOffset + uncompressedOff = info.uncompressedOffset + } + return compressedOff, uncompressedOff, nil +} + +// reduce to stay below maxIndexEntries +func (i *Index) reduce() { + if len(i.info) < maxIndexEntries && i.estBlockUncomp >= 1<<20 { + return + } + + // Algorithm, keep 1, remove removeN entries... + removeN := (len(i.info) + 1) / maxIndexEntries + src := i.info + j := 0 + + // Each block should be at least 1MB, but don't reduce below 1000 entries. + for i.estBlockUncomp*(int64(removeN)+1) < 1<<20 && len(i.info)/(removeN+1) > 1000 { + removeN++ + } + for idx := 0; idx < len(src); idx++ { + i.info[j] = src[idx] + j++ + idx += removeN + } + i.info = i.info[:j] + // Update maxblock estimate. + i.estBlockUncomp += i.estBlockUncomp * int64(removeN) +} + +func (i *Index) appendTo(b []byte, uncompTotal, compTotal int64) []byte { + i.reduce() + var tmp [binary.MaxVarintLen64]byte + + initSize := len(b) + // We make the start a skippable header+size. + b = append(b, ChunkTypeIndex, 0, 0, 0) + b = append(b, []byte(S2IndexHeader)...) + // Total Uncompressed size + n := binary.PutVarint(tmp[:], uncompTotal) + b = append(b, tmp[:n]...) + // Total Compressed size + n = binary.PutVarint(tmp[:], compTotal) + b = append(b, tmp[:n]...) + // Put EstBlockUncomp size + n = binary.PutVarint(tmp[:], i.estBlockUncomp) + b = append(b, tmp[:n]...) + // Put length + n = binary.PutVarint(tmp[:], int64(len(i.info))) + b = append(b, tmp[:n]...) + + // Check if we should add uncompressed offsets + var hasUncompressed byte + for idx, info := range i.info { + if idx == 0 { + if info.uncompressedOffset != 0 { + hasUncompressed = 1 + break + } + continue + } + if info.uncompressedOffset != i.info[idx-1].uncompressedOffset+i.estBlockUncomp { + hasUncompressed = 1 + break + } + } + b = append(b, hasUncompressed) + + // Add each entry + if hasUncompressed == 1 { + for idx, info := range i.info { + uOff := info.uncompressedOffset + if idx > 0 { + prev := i.info[idx-1] + uOff -= prev.uncompressedOffset + (i.estBlockUncomp) + } + n = binary.PutVarint(tmp[:], uOff) + b = append(b, tmp[:n]...) + } + } + + // Initial compressed size estimate. + cPredict := i.estBlockUncomp / 2 + + for idx, info := range i.info { + cOff := info.compressedOffset + if idx > 0 { + prev := i.info[idx-1] + cOff -= prev.compressedOffset + cPredict + // Update compressed size prediction, with half the error. + cPredict += cOff / 2 + } + n = binary.PutVarint(tmp[:], cOff) + b = append(b, tmp[:n]...) + } + + // Add Total Size. + // Stored as fixed size for easier reading. + binary.LittleEndian.PutUint32(tmp[:], uint32(len(b)-initSize+4+len(S2IndexTrailer))) + b = append(b, tmp[:4]...) + // Trailer + b = append(b, []byte(S2IndexTrailer)...) + + // Update size + chunkLen := len(b) - initSize - skippableFrameHeader + b[initSize+1] = uint8(chunkLen >> 0) + b[initSize+2] = uint8(chunkLen >> 8) + b[initSize+3] = uint8(chunkLen >> 16) + //fmt.Printf("chunklen: 0x%x Uncomp:%d, Comp:%d\n", chunkLen, uncompTotal, compTotal) + return b +} + +// Load a binary index. +// A zero value Index can be used or a previous one can be reused. +func (i *Index) Load(b []byte) ([]byte, error) { + if len(b) <= 4+len(S2IndexHeader)+len(S2IndexTrailer) { + return b, io.ErrUnexpectedEOF + } + if b[0] != ChunkTypeIndex { + return b, ErrCorrupt + } + chunkLen := int(b[1]) | int(b[2])<<8 | int(b[3])<<16 + b = b[4:] + + // Validate we have enough... + if len(b) < chunkLen { + return b, io.ErrUnexpectedEOF + } + if !bytes.Equal(b[:len(S2IndexHeader)], []byte(S2IndexHeader)) { + return b, ErrUnsupported + } + b = b[len(S2IndexHeader):] + + // Total Uncompressed + if v, n := binary.Varint(b); n <= 0 || v < 0 { + return b, ErrCorrupt + } else { + i.TotalUncompressed = v + b = b[n:] + } + + // Total Compressed + if v, n := binary.Varint(b); n <= 0 { + return b, ErrCorrupt + } else { + i.TotalCompressed = v + b = b[n:] + } + + // Read EstBlockUncomp + if v, n := binary.Varint(b); n <= 0 { + return b, ErrCorrupt + } else { + if v < 0 { + return b, ErrCorrupt + } + i.estBlockUncomp = v + b = b[n:] + } + + var entries int + if v, n := binary.Varint(b); n <= 0 { + return b, ErrCorrupt + } else { + if v < 0 || v > maxIndexEntries { + return b, ErrCorrupt + } + entries = int(v) + b = b[n:] + } + if cap(i.info) < entries { + i.allocInfos(entries) + } + i.info = i.info[:entries] + + if len(b) < 1 { + return b, io.ErrUnexpectedEOF + } + hasUncompressed := b[0] + b = b[1:] + if hasUncompressed&1 != hasUncompressed { + return b, ErrCorrupt + } + + // Add each uncompressed entry + for idx := range i.info { + var uOff int64 + if hasUncompressed != 0 { + // Load delta + if v, n := binary.Varint(b); n <= 0 { + return b, ErrCorrupt + } else { + uOff = v + b = b[n:] + } + } + + if idx > 0 { + prev := i.info[idx-1].uncompressedOffset + uOff += prev + (i.estBlockUncomp) + if uOff <= prev { + return b, ErrCorrupt + } + } + if uOff < 0 { + return b, ErrCorrupt + } + i.info[idx].uncompressedOffset = uOff + } + + // Initial compressed size estimate. + cPredict := i.estBlockUncomp / 2 + + // Add each compressed entry + for idx := range i.info { + var cOff int64 + if v, n := binary.Varint(b); n <= 0 { + return b, ErrCorrupt + } else { + cOff = v + b = b[n:] + } + + if idx > 0 { + // Update compressed size prediction, with half the error. + cPredictNew := cPredict + cOff/2 + + prev := i.info[idx-1].compressedOffset + cOff += prev + cPredict + if cOff <= prev { + return b, ErrCorrupt + } + cPredict = cPredictNew + } + if cOff < 0 { + return b, ErrCorrupt + } + i.info[idx].compressedOffset = cOff + } + if len(b) < 4+len(S2IndexTrailer) { + return b, io.ErrUnexpectedEOF + } + // Skip size... + b = b[4:] + + // Check trailer... + if !bytes.Equal(b[:len(S2IndexTrailer)], []byte(S2IndexTrailer)) { + return b, ErrCorrupt + } + return b[len(S2IndexTrailer):], nil +} + +// LoadStream will load an index from the end of the supplied stream. +// ErrUnsupported will be returned if the signature cannot be found. +// ErrCorrupt will be returned if unexpected values are found. +// io.ErrUnexpectedEOF is returned if there are too few bytes. +// IO errors are returned as-is. +func (i *Index) LoadStream(rs io.ReadSeeker) error { + // Go to end. + _, err := rs.Seek(-10, io.SeekEnd) + if err != nil { + return err + } + var tmp [10]byte + _, err = io.ReadFull(rs, tmp[:]) + if err != nil { + return err + } + // Check trailer... + if !bytes.Equal(tmp[4:4+len(S2IndexTrailer)], []byte(S2IndexTrailer)) { + return ErrUnsupported + } + sz := binary.LittleEndian.Uint32(tmp[:4]) + if sz > maxChunkSize+skippableFrameHeader { + return ErrCorrupt + } + _, err = rs.Seek(-int64(sz), io.SeekEnd) + if err != nil { + return err + } + + // Read index. + buf := make([]byte, sz) + _, err = io.ReadFull(rs, buf) + if err != nil { + return err + } + _, err = i.Load(buf) + return err +} + +// IndexStream will return an index for a stream. +// The stream structure will be checked, but +// data within blocks is not verified. +// The returned index can either be appended to the end of the stream +// or stored separately. +func IndexStream(r io.Reader) ([]byte, error) { + var i Index + var buf [maxChunkSize]byte + var readHeader bool + for { + _, err := io.ReadFull(r, buf[:4]) + if err != nil { + if err == io.EOF { + return i.appendTo(nil, i.TotalUncompressed, i.TotalCompressed), nil + } + return nil, err + } + // Start of this chunk. + startChunk := i.TotalCompressed + i.TotalCompressed += 4 + + chunkType := buf[0] + if !readHeader { + if chunkType != chunkTypeStreamIdentifier { + return nil, ErrCorrupt + } + readHeader = true + } + chunkLen := int(buf[1]) | int(buf[2])<<8 | int(buf[3])<<16 + if chunkLen < checksumSize { + return nil, ErrCorrupt + } + + i.TotalCompressed += int64(chunkLen) + _, err = io.ReadFull(r, buf[:chunkLen]) + if err != nil { + return nil, io.ErrUnexpectedEOF + } + // The chunk types are specified at + // https://github.com/google/snappy/blob/master/framing_format.txt + switch chunkType { + case chunkTypeCompressedData: + // Section 4.2. Compressed data (chunk type 0x00). + // Skip checksum. + dLen, err := DecodedLen(buf[checksumSize:]) + if err != nil { + return nil, err + } + if dLen > maxBlockSize { + return nil, ErrCorrupt + } + if i.estBlockUncomp == 0 { + // Use first block for estimate... + i.estBlockUncomp = int64(dLen) + } + err = i.add(startChunk, i.TotalUncompressed) + if err != nil { + return nil, err + } + i.TotalUncompressed += int64(dLen) + continue + case chunkTypeUncompressedData: + n2 := chunkLen - checksumSize + if n2 > maxBlockSize { + return nil, ErrCorrupt + } + if i.estBlockUncomp == 0 { + // Use first block for estimate... + i.estBlockUncomp = int64(n2) + } + err = i.add(startChunk, i.TotalUncompressed) + if err != nil { + return nil, err + } + i.TotalUncompressed += int64(n2) + continue + case chunkTypeStreamIdentifier: + // Section 4.1. Stream identifier (chunk type 0xff). + if chunkLen != len(magicBody) { + return nil, ErrCorrupt + } + + if string(buf[:len(magicBody)]) != magicBody { + if string(buf[:len(magicBody)]) != magicBodySnappy { + return nil, ErrCorrupt + } + } + + continue + } + + if chunkType <= 0x7f { + // Section 4.5. Reserved unskippable chunks (chunk types 0x02-0x7f). + return nil, ErrUnsupported + } + if chunkLen > maxChunkSize { + return nil, ErrUnsupported + } + // Section 4.4 Padding (chunk type 0xfe). + // Section 4.6. Reserved skippable chunks (chunk types 0x80-0xfd). + } +} + +// JSON returns the index as JSON text. +func (i *Index) JSON() []byte { + x := struct { + TotalUncompressed int64 `json:"total_uncompressed"` // Total Uncompressed size if known. Will be -1 if unknown. + TotalCompressed int64 `json:"total_compressed"` // Total Compressed size if known. Will be -1 if unknown. + Offsets []struct { + CompressedOffset int64 `json:"compressed"` + UncompressedOffset int64 `json:"uncompressed"` + } `json:"offsets"` + EstBlockUncomp int64 `json:"est_block_uncompressed"` + }{ + TotalUncompressed: i.TotalUncompressed, + TotalCompressed: i.TotalCompressed, + EstBlockUncomp: i.estBlockUncomp, + } + for _, v := range i.info { + x.Offsets = append(x.Offsets, struct { + CompressedOffset int64 `json:"compressed"` + UncompressedOffset int64 `json:"uncompressed"` + }{CompressedOffset: v.compressedOffset, UncompressedOffset: v.uncompressedOffset}) + } + b, _ := json.MarshalIndent(x, "", " ") + return b +} + +// RemoveIndexHeaders will trim all headers and trailers from a given index. +// This is expected to save 20 bytes. +// These can be restored using RestoreIndexHeaders. +// This removes a layer of security, but is the most compact representation. +// Returns nil if headers contains errors. +// The returned slice references the provided slice. +func RemoveIndexHeaders(b []byte) []byte { + const save = 4 + len(S2IndexHeader) + len(S2IndexTrailer) + 4 + if len(b) <= save { + return nil + } + if b[0] != ChunkTypeIndex { + return nil + } + chunkLen := int(b[1]) | int(b[2])<<8 | int(b[3])<<16 + b = b[4:] + + // Validate we have enough... + if len(b) < chunkLen { + return nil + } + b = b[:chunkLen] + + if !bytes.Equal(b[:len(S2IndexHeader)], []byte(S2IndexHeader)) { + return nil + } + b = b[len(S2IndexHeader):] + if !bytes.HasSuffix(b, []byte(S2IndexTrailer)) { + return nil + } + b = bytes.TrimSuffix(b, []byte(S2IndexTrailer)) + + if len(b) < 4 { + return nil + } + return b[:len(b)-4] +} + +// RestoreIndexHeaders will index restore headers removed by RemoveIndexHeaders. +// No error checking is performed on the input. +// If a 0 length slice is sent, it is returned without modification. +func RestoreIndexHeaders(in []byte) []byte { + if len(in) == 0 { + return in + } + b := make([]byte, 0, 4+len(S2IndexHeader)+len(in)+len(S2IndexTrailer)+4) + b = append(b, ChunkTypeIndex, 0, 0, 0) + b = append(b, []byte(S2IndexHeader)...) + b = append(b, in...) + + var tmp [4]byte + binary.LittleEndian.PutUint32(tmp[:], uint32(len(b)+4+len(S2IndexTrailer))) + b = append(b, tmp[:4]...) + // Trailer + b = append(b, []byte(S2IndexTrailer)...) + + chunkLen := len(b) - skippableFrameHeader + b[1] = uint8(chunkLen >> 0) + b[2] = uint8(chunkLen >> 8) + b[3] = uint8(chunkLen >> 16) + return b +} diff --git a/vendor/github.com/klauspost/compress/s2/s2.go b/vendor/github.com/klauspost/compress/s2/s2.go new file mode 100644 index 000000000..dae3f731f --- /dev/null +++ b/vendor/github.com/klauspost/compress/s2/s2.go @@ -0,0 +1,143 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Copyright (c) 2019 Klaus Post. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package s2 implements the S2 compression format. +// +// S2 is an extension of Snappy. Similar to Snappy S2 is aimed for high throughput, +// which is why it features concurrent compression for bigger payloads. +// +// Decoding is compatible with Snappy compressed content, +// but content compressed with S2 cannot be decompressed by Snappy. +// +// For more information on Snappy/S2 differences see README in: https://github.com/klauspost/compress/tree/master/s2 +// +// There are actually two S2 formats: block and stream. They are related, +// but different: trying to decompress block-compressed data as a S2 stream +// will fail, and vice versa. The block format is the Decode and Encode +// functions and the stream format is the Reader and Writer types. +// +// A "better" compression option is available. This will trade some compression +// speed +// +// The block format, the more common case, is used when the complete size (the +// number of bytes) of the original data is known upfront, at the time +// compression starts. The stream format, also known as the framing format, is +// for when that isn't always true. +// +// Blocks to not offer much data protection, so it is up to you to +// add data validation of decompressed blocks. +// +// Streams perform CRC validation of the decompressed data. +// Stream compression will also be performed on multiple CPU cores concurrently +// significantly improving throughput. +package s2 + +import ( + "bytes" + "hash/crc32" +) + +/* +Each encoded block begins with the varint-encoded length of the decoded data, +followed by a sequence of chunks. Chunks begin and end on byte boundaries. The +first byte of each chunk is broken into its 2 least and 6 most significant bits +called l and m: l ranges in [0, 4) and m ranges in [0, 64). l is the chunk tag. +Zero means a literal tag. All other values mean a copy tag. + +For literal tags: + - If m < 60, the next 1 + m bytes are literal bytes. + - Otherwise, let n be the little-endian unsigned integer denoted by the next + m - 59 bytes. The next 1 + n bytes after that are literal bytes. + +For copy tags, length bytes are copied from offset bytes ago, in the style of +Lempel-Ziv compression algorithms. In particular: + - For l == 1, the offset ranges in [0, 1<<11) and the length in [4, 12). + The length is 4 + the low 3 bits of m. The high 3 bits of m form bits 8-10 + of the offset. The next byte is bits 0-7 of the offset. + - For l == 2, the offset ranges in [0, 1<<16) and the length in [1, 65). + The length is 1 + m. The offset is the little-endian unsigned integer + denoted by the next 2 bytes. + - For l == 3, the offset ranges in [0, 1<<32) and the length in + [1, 65). The length is 1 + m. The offset is the little-endian unsigned + integer denoted by the next 4 bytes. +*/ +const ( + tagLiteral = 0x00 + tagCopy1 = 0x01 + tagCopy2 = 0x02 + tagCopy4 = 0x03 +) + +const ( + checksumSize = 4 + chunkHeaderSize = 4 + magicChunk = "\xff\x06\x00\x00" + magicBody + magicChunkSnappy = "\xff\x06\x00\x00" + magicBodySnappy + magicBodySnappy = "sNaPpY" + magicBody = "S2sTwO" + + // maxBlockSize is the maximum size of the input to encodeBlock. + // + // For the framing format (Writer type instead of Encode function), + // this is the maximum uncompressed size of a block. + maxBlockSize = 4 << 20 + + // minBlockSize is the minimum size of block setting when creating a writer. + minBlockSize = 4 << 10 + + skippableFrameHeader = 4 + maxChunkSize = 1<<24 - 1 // 16777215 + + // Default block size + defaultBlockSize = 1 << 20 + + // maxSnappyBlockSize is the maximum snappy block size. + maxSnappyBlockSize = 1 << 16 + + obufHeaderLen = checksumSize + chunkHeaderSize +) + +const ( + chunkTypeCompressedData = 0x00 + chunkTypeUncompressedData = 0x01 + ChunkTypeIndex = 0x99 + chunkTypePadding = 0xfe + chunkTypeStreamIdentifier = 0xff +) + +var crcTable = crc32.MakeTable(crc32.Castagnoli) + +// crc implements the checksum specified in section 3 of +// https://github.com/google/snappy/blob/master/framing_format.txt +func crc(b []byte) uint32 { + c := crc32.Update(0, crcTable, b) + return c>>15 | c<<17 + 0xa282ead8 +} + +// literalExtraSize returns the extra size of encoding n literals. +// n should be >= 0 and <= math.MaxUint32. +func literalExtraSize(n int64) int64 { + if n == 0 { + return 0 + } + switch { + case n < 60: + return 1 + case n < 1<<8: + return 2 + case n < 1<<16: + return 3 + case n < 1<<24: + return 4 + default: + return 5 + } +} + +type byter interface { + Bytes() []byte +} + +var _ byter = &bytes.Buffer{} diff --git a/vendor/modules.txt b/vendor/modules.txt index 17b196718..b02361808 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -16,7 +16,7 @@ cloud.google.com/go/storage cloud.google.com/go/storage/internal cloud.google.com/go/storage/internal/apiv2 cloud.google.com/go/storage/internal/apiv2/stubs -# github.com/VictoriaMetrics/fastcache v1.10.0 +# github.com/VictoriaMetrics/fastcache v1.12.0 ## explicit; go 1.13 github.com/VictoriaMetrics/fastcache # github.com/VictoriaMetrics/fasthttp v1.1.0 @@ -166,6 +166,7 @@ github.com/klauspost/compress/gzip github.com/klauspost/compress/huff0 github.com/klauspost/compress/internal/cpuinfo github.com/klauspost/compress/internal/snapref +github.com/klauspost/compress/s2 github.com/klauspost/compress/zlib github.com/klauspost/compress/zstd github.com/klauspost/compress/zstd/internal/xxhash From 584a5d6b1f3f78f5a0e21dfe81b2a3fb87a8da80 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 14:52:46 +0300 Subject: [PATCH 14/18] docs/managed_victoriametrics/quickstart.md: small fixes after 606166ef6831dd8334e81552cb1b6b1fa4b91745 --- docs/managed_victoriametrics/quickstart.md | 34 ++++++++++++---------- 1 file changed, 18 insertions(+), 16 deletions(-) diff --git a/docs/managed_victoriametrics/quickstart.md b/docs/managed_victoriametrics/quickstart.md index 6dbc02980..ec5826e9c 100644 --- a/docs/managed_victoriametrics/quickstart.md +++ b/docs/managed_victoriametrics/quickstart.md @@ -13,40 +13,42 @@ See more details [here](https://dbaas.victoriametrics.com/howToRegister). ## How to restore password -If you forgot your password you can restore it the following steps: -1. Click on `Forgot your password?` link at https://dbaas.victoriametrics.com/signIn +If you forgot password, it can be restored in the following way: + +1. Click `Forgot your password?` link at [this page](https://dbaas.victoriametrics.com/signIn):

- +

-2. Enter your email in the field and press button `Send Email` +2. Enter your email in the field and click `Send Email` button:

- +

-Follow the instruction sent to the specified email address: +3. Follow the instruction sent to your email in order to gain access to your VictoriaMetrics cloud account: + ``` Victoria Metrics Cloud password restore Follow https://dbaas.victoriametrics.com/login_by_link/{id} the link in order to restore access to Victoria Metrics Cloud. Access link expires once you login successfully or after 30min. You can change your password after login https://dbaas.victoriametrics.com/profile profile -Please, ignore this email if you didn't init this action on Victoria Metrics Cloud.
+Please, ignore this email if you didn't init this action on Victoria Metrics Cloud. In case of questions contact our support support@victoriametrics.com ``` -3. To change you password visit Profile page in the top right corner of the web page +4. Navigate to the Profile page by clicking the corresponding link at the top right corner:

- +

-4. At the Profile page, enter new password and repeat it in the next field and press `Save` button. +5. Enter new password at the Profile page and press `Save` button:

- +

@@ -56,7 +58,7 @@ Instances is a page where user can list and manage VictoriaMetrics single-node i To create an instance click on the button `Create`:

- +

In the opened form, choose parameters of the new instance such as: @@ -67,7 +69,7 @@ In the opened form, choose parameters of the new instance such as: * `Retention` period for stored metrics.

- +

Once created, instance will remain for a short period of time in `PROVISIONING` status @@ -75,7 +77,7 @@ while the hardware spins-up, just wait for a couple of minutes and reload the pa You'll also be notified via email once provisioning is finished:

- +

## Access @@ -84,7 +86,7 @@ After transition from `PROVISIONING` to `RUNNING` state, VictoriaMetrics is full and ready to accept write or read requests. But first, click on instance name to get the access token:

- +

Access tokens are used in token-based authentication to allow an application to access the VictoriaMetrics API. @@ -92,7 +94,7 @@ Supported token types are `Read-Only`, `Write-Only` and `Read-Write`. Click on t to see usage examples:

- +

Follow usage example in order to configure access to VictoriaMetrics for your Prometheus, From 1aef635de429e49dbe5179932b0ea70b55220ae0 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 15:01:32 +0300 Subject: [PATCH 15/18] docs/CHANGELOG.md: clarify the change at 622bbedbe11149ad0a7dbfef075a2c4e2b57e527 Updates https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3119 --- docs/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/CHANGELOG.md b/docs/CHANGELOG.md index 8cb11f34a..76a03d0cb 100644 --- a/docs/CHANGELOG.md +++ b/docs/CHANGELOG.md @@ -31,7 +31,7 @@ The following tip changes can be tested by building VictoriaMetrics components f * BUGFIX: [VictoriaMetrics cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html): properly calculate query results at `vmselect`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3067). The issue has been introduced in [v1.81.0](https://docs.victoriametrics.com/CHANGELOG.html#v1810). * BUGFIX: [VictoriaMetrics cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html): log clear error when multiple identical `-storageNode` command-line flags are passed to `vmselect` or to `vminsert`. Previously these components were crashed with cryptic panic `metric ... is already registered` in this case. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3076). * BUGFIX: [vmui](https://docs.victoriametrics.com/#vmui): fix `RangeError: Maximum call stack size exceeded` error when the query returns too many data points at `Table` view. See [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3092/files). -* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): re-evaluate alert's annotations on each execution. Previously, annotations were evaluated only on alert's value change. See [this PR](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3119) for details. +* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): re-evaluate annotations per each each alert evaluation. Previously, annotations were evaluated only on alert's value change. This could result in stale annotations in some cases described in [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3119). ## [v1.81.2](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.81.2) From fd98ec8ba3432310256e4ec0b88af97afa1f2695 Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 15:12:03 +0300 Subject: [PATCH 16/18] Makefile: fix -compat value passed to `go mod tidy` --- Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Makefile b/Makefile index f2d78468d..b6ba009a6 100644 --- a/Makefile +++ b/Makefile @@ -364,7 +364,7 @@ benchmark-pure: vendor-update: go get -u -d ./lib/... go get -u -d ./app/... - go mod tidy -compat=1.19.1 + go mod tidy -compat=1.19 go mod vendor app-local: From 310d0caec299497d0a7ab11b6a6830140c999b6c Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 15:12:22 +0300 Subject: [PATCH 17/18] vendor: `make vendor-update` --- go.mod | 12 +- go.sum | 26 +- .../aws/aws-sdk-go/aws/endpoints/defaults.go | 91 ++ .../github.com/aws/aws-sdk-go/aws/version.go | 2 +- .../github.com/klauspost/compress/README.md | 6 + .../klauspost/compress/flate/deflate.go | 3 +- .../klauspost/compress/flate/dict_decoder.go | 24 +- .../klauspost/compress/flate/fast_encoder.go | 3 +- .../compress/flate/huffman_bit_writer.go | 20 +- .../klauspost/compress/flate/huffman_code.go | 15 +- .../klauspost/compress/flate/level5.go | 15 +- .../klauspost/compress/flate/level6.go | 21 +- .../klauspost/compress/huff0/decompress.go | 36 +- .../compress/huff0/decompress_amd64.go | 4 + .../compress/huff0/decompress_amd64.s | 1 - .../compress/huff0/decompress_generic.go | 18 +- .../compress/internal/snapref/encode_other.go | 6 +- .../klauspost/compress/s2/README.md | 11 +- .../klauspost/compress/s2/decode_other.go | 34 +- .../klauspost/compress/s2/encode_all.go | 3 +- .../klauspost/compress/s2/encode_amd64.go | 12 +- .../klauspost/compress/s2/encode_best.go | 35 +- .../klauspost/compress/s2/encode_better.go | 105 +- .../klauspost/compress/s2/encode_go.go | 9 +- .../compress/s2/encodeblock_amd64.go | 23 +- .../klauspost/compress/s2/encodeblock_amd64.s | 919 ++++++++++-------- .../klauspost/compress/zstd/README.md | 2 + .../klauspost/compress/zstd/blockdec.go | 3 +- .../klauspost/compress/zstd/bytebuf.go | 3 +- .../klauspost/compress/zstd/decoder.go | 21 +- .../compress/zstd/decoder_options.go | 12 + .../klauspost/compress/zstd/enc_better.go | 23 +- .../klauspost/compress/zstd/enc_dfast.go | 7 +- .../klauspost/compress/zstd/enc_fast.go | 6 +- .../klauspost/compress/zstd/framedec.go | 23 +- .../compress/zstd/fse_decoder_amd64.s | 1 - .../klauspost/compress/zstd/seqdec_amd64.go | 10 + .../klauspost/compress/zstd/seqdec_amd64.s | 1 - vendor/golang.org/x/sys/unix/mkall.sh | 5 - vendor/golang.org/x/sys/unix/syscall.go | 3 +- .../x/sys/unix/syscall_darwin.1_12.go | 32 - .../x/sys/unix/syscall_darwin.1_13.go | 100 -- .../golang.org/x/sys/unix/syscall_darwin.go | 90 ++ vendor/golang.org/x/sys/unix/xattr_bsd.go | 95 +- .../x/sys/unix/zsyscall_darwin_amd64.1_13.go | 40 - .../x/sys/unix/zsyscall_darwin_amd64.1_13.s | 25 - .../x/sys/unix/zsyscall_darwin_amd64.go | 32 +- .../x/sys/unix/zsyscall_darwin_amd64.s | 21 +- .../x/sys/unix/zsyscall_darwin_arm64.1_13.go | 40 - .../x/sys/unix/zsyscall_darwin_arm64.1_13.s | 25 - .../x/sys/unix/zsyscall_darwin_arm64.go | 32 +- .../x/sys/unix/zsyscall_darwin_arm64.s | 21 +- vendor/golang.org/x/sys/windows/syscall.go | 10 +- .../x/sys/windows/syscall_windows.go | 12 +- .../golang.org/x/sys/windows/types_windows.go | 26 + .../x/sys/windows/zsyscall_windows.go | 19 + .../google.golang.org/api/internal/version.go | 2 +- vendor/modules.txt | 14 +- 58 files changed, 1283 insertions(+), 927 deletions(-) delete mode 100644 vendor/golang.org/x/sys/unix/syscall_darwin.1_12.go delete mode 100644 vendor/golang.org/x/sys/unix/syscall_darwin.1_13.go delete mode 100644 vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.go delete mode 100644 vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.s delete mode 100644 vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.go delete mode 100644 vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.s diff --git a/go.mod b/go.mod index b379cfb95..d1f3e3a6a 100644 --- a/go.mod +++ b/go.mod @@ -11,7 +11,7 @@ require ( github.com/VictoriaMetrics/fasthttp v1.1.0 github.com/VictoriaMetrics/metrics v1.22.2 github.com/VictoriaMetrics/metricsql v0.45.0 - github.com/aws/aws-sdk-go v1.44.96 + github.com/aws/aws-sdk-go v1.44.100 github.com/cespare/xxhash/v2 v2.1.2 // TODO: switch back to https://github.com/cheggaaa/pb/v3 when v3-pooling branch @@ -20,7 +20,7 @@ require ( github.com/dmitryk-dk/pb/v3 v3.0.9 github.com/golang/snappy v0.0.4 github.com/influxdata/influxdb v1.10.0 - github.com/klauspost/compress v1.15.9 + github.com/klauspost/compress v1.15.10 github.com/prometheus/prometheus v1.8.2-0.20201119142752-3ad25a6dc3d9 github.com/urfave/cli/v2 v2.16.3 github.com/valyala/fastjson v1.6.3 @@ -30,14 +30,14 @@ require ( github.com/valyala/quicktemplate v1.7.0 golang.org/x/net v0.0.0-20220909164309-bea034e7d591 golang.org/x/oauth2 v0.0.0-20220909003341-f21342109be1 - golang.org/x/sys v0.0.0-20220913120320-3275c407cedc - google.golang.org/api v0.95.0 + golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8 + google.golang.org/api v0.96.0 gopkg.in/yaml.v2 v2.4.0 ) require ( cloud.google.com/go v0.104.0 // indirect - cloud.google.com/go/compute v1.9.0 // indirect + cloud.google.com/go/compute v1.10.0 // indirect cloud.google.com/go/iam v0.4.0 // indirect github.com/VividCortex/ewma v1.2.0 // indirect github.com/beorn7/perks v1.0.1 // indirect @@ -75,7 +75,7 @@ require ( golang.org/x/text v0.3.7 // indirect golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect google.golang.org/appengine v1.6.7 // indirect - google.golang.org/genproto v0.0.0-20220909194730-69f6226f97e5 // indirect + google.golang.org/genproto v0.0.0-20220916172020-2692e8806bfa // indirect google.golang.org/grpc v1.49.0 // indirect google.golang.org/protobuf v1.28.1 // indirect ) diff --git a/go.sum b/go.sum index 211548b25..6f07da281 100644 --- a/go.sum +++ b/go.sum @@ -45,8 +45,8 @@ cloud.google.com/go/compute v1.5.0/go.mod h1:9SMHyhJlzhlkJqrPAc839t2BZFTSk6Jdj6m cloud.google.com/go/compute v1.6.0/go.mod h1:T29tfhtVbq1wvAPo0E3+7vhgmkOYeXjhFvz/FMzPu0s= cloud.google.com/go/compute v1.6.1/go.mod h1:g85FgpzFvNULZ+S8AYq87axRKuf2Kh7deLqV/jJ3thU= cloud.google.com/go/compute v1.7.0/go.mod h1:435lt8av5oL9P3fv1OEzSbSUe+ybHXGMPQHHZWZxy9U= -cloud.google.com/go/compute v1.9.0 h1:ED/FP4xv8GJw63v556/ASNc1CeeLUO2Bs8nzaHchkHg= -cloud.google.com/go/compute v1.9.0/go.mod h1:lWv1h/zUWTm/LozzfTJhBSkd6ShQq8la8VeeuOEGxfY= +cloud.google.com/go/compute v1.10.0 h1:aoLIYaA1fX3ywihqpBk2APQKOo20nXsp1GEZQbx5Jk4= +cloud.google.com/go/compute v1.10.0/go.mod h1:ER5CLbMxl90o2jtNbGSbtfOpQKR0t15FOtRsugnLrlU= cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= cloud.google.com/go/iam v0.3.0/go.mod h1:XzJPvDayI+9zsASAFO68Hk07u3z+f+JrT2xXNdp4bnY= @@ -148,8 +148,8 @@ github.com/aws/aws-lambda-go v1.13.3/go.mod h1:4UKl9IzQMoD+QF79YdCuzCwp8VbmG4VAQ github.com/aws/aws-sdk-go v1.27.0/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= github.com/aws/aws-sdk-go v1.34.28/go.mod h1:H7NKnBqNVzoTJpGfLrQkkD+ytBA93eiDYi/+8rV9s48= github.com/aws/aws-sdk-go v1.35.31/go.mod h1:hcU610XS61/+aQV88ixoOzUoG7v3b31pl2zKMmprdro= -github.com/aws/aws-sdk-go v1.44.96 h1:S9paaqnJ0AJ95t5AB+iK8RM6YNZN0W0Lek1gOVJsEr8= -github.com/aws/aws-sdk-go v1.44.96/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX4oIKwKHZo= +github.com/aws/aws-sdk-go v1.44.100 h1:7I86bWNQB+HGDT5z/dJy61J7qgbgLoZ7O51C9eL6hrA= +github.com/aws/aws-sdk-go v1.44.100/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX4oIKwKHZo= github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= @@ -576,8 +576,8 @@ github.com/klauspost/compress v1.4.0/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0 github.com/klauspost/compress v1.9.5/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= github.com/klauspost/compress v1.13.4/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= github.com/klauspost/compress v1.13.5/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= -github.com/klauspost/compress v1.15.9 h1:wKRjX6JRtDdrE9qwa4b/Cip7ACOshUI4smpCQanqjSY= -github.com/klauspost/compress v1.15.9/go.mod h1:PhcZ0MbTNciWF3rruxRgKxI5NkcHHrHUDtV4Yw2GlzU= +github.com/klauspost/compress v1.15.10 h1:Ai8UzuomSCDw90e1qNMtb15msBXsNpH6gzkkENQNcJo= +github.com/klauspost/compress v1.15.10/go.mod h1:QPwzmACJjUTFsnSHH934V6woptycfrDDJnH7hvFVbGM= github.com/klauspost/cpuid v0.0.0-20170728055534-ae7887de9fa5/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek= github.com/klauspost/crc32 v0.0.0-20161016154125-cb6bfca970f6/go.mod h1:+ZoRqAPRLkC4NPOvfYeR5KNOrY6TD+/sAC3HXPZgDYg= github.com/klauspost/pgzip v1.0.2-0.20170402124221-0bf5dcad4ada/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs= @@ -1140,10 +1140,10 @@ golang.org/x/sys v0.0.0-20220502124256-b6088ccd6cba/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220610221304-9f5ed59c137d/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220624220833-87e55d714810/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220913120320-3275c407cedc h1:dpclq5m2YrqPGStKmtw7IcNbKLfbIqKXvNxDJKdIKYc= -golang.org/x/sys v0.0.0-20220913120320-3275c407cedc/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8 h1:h+EGohizhe9XlX18rfpa8k8RAc5XyaeamM+0VHRd4lc= +golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -1290,8 +1290,8 @@ google.golang.org/api v0.75.0/go.mod h1:pU9QmyHLnzlpar1Mjt4IbapUCy8J+6HD6GeELN69 google.golang.org/api v0.78.0/go.mod h1:1Sg78yoMLOhlQTeF+ARBoytAcH1NNyyl390YMy6rKmw= google.golang.org/api v0.80.0/go.mod h1:xY3nI94gbvBrE0J6NHXhxOmW97HG7Khjkku6AFB3Hyg= google.golang.org/api v0.84.0/go.mod h1:NTsGnUFJMYROtiquksZHBWtHfeMC7iYthki7Eq3pa8o= -google.golang.org/api v0.95.0 h1:d1c24AAS01DYqXreBeuVV7ewY/U8Mnhh47pwtsgVtYg= -google.golang.org/api v0.95.0/go.mod h1:eADj+UBuxkh5zlrSntJghuNeg8HwQ1w5lTKkuqaETEI= +google.golang.org/api v0.96.0 h1:F60cuQPJq7K7FzsxMYHAUJSiXh2oKctHxBMbDygxhfM= +google.golang.org/api v0.96.0/go.mod h1:w7wJQLTM+wvQpNf5JyEcBoxK0RH7EDrh/L4qfsuJ13s= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= google.golang.org/appengine v1.2.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= @@ -1383,8 +1383,8 @@ google.golang.org/genproto v0.0.0-20220523171625-347a074981d8/go.mod h1:RAyBrSAP google.golang.org/genproto v0.0.0-20220608133413-ed9918b62aac/go.mod h1:KEWEmljWE5zPzLBa/oHl6DaEt9LmfH6WtH1OHIvleBA= google.golang.org/genproto v0.0.0-20220616135557-88e70c0c3a90/go.mod h1:KEWEmljWE5zPzLBa/oHl6DaEt9LmfH6WtH1OHIvleBA= google.golang.org/genproto v0.0.0-20220624142145-8cd45d7dbd1f/go.mod h1:KEWEmljWE5zPzLBa/oHl6DaEt9LmfH6WtH1OHIvleBA= -google.golang.org/genproto v0.0.0-20220909194730-69f6226f97e5 h1:ngtP8S8JkBWfJACT9cmj5eTkS9tIWPQI5leBz/7Bq/c= -google.golang.org/genproto v0.0.0-20220909194730-69f6226f97e5/go.mod h1:0Nb8Qy+Sk5eDzHnzlStwW3itdNaWoZA5XeSG+R3JHSo= +google.golang.org/genproto v0.0.0-20220916172020-2692e8806bfa h1:VWkrxnAx2C2hirAP+W5ADU7e/+93Yhk//ioKd2XFyDI= +google.golang.org/genproto v0.0.0-20220916172020-2692e8806bfa/go.mod h1:0Nb8Qy+Sk5eDzHnzlStwW3itdNaWoZA5XeSG+R3JHSo= google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.0/go.mod h1:chYK+tFQF0nDUGJgXMSgLCQk3phJEuONr2DCgLDdAQM= diff --git a/vendor/github.com/aws/aws-sdk-go/aws/endpoints/defaults.go b/vendor/github.com/aws/aws-sdk-go/aws/endpoints/defaults.go index 7ec5aa0aa..4df415be5 100644 --- a/vendor/github.com/aws/aws-sdk-go/aws/endpoints/defaults.go +++ b/vendor/github.com/aws/aws-sdk-go/aws/endpoints/defaults.go @@ -16671,6 +16671,70 @@ var awsPartition = partition{ }, }, }, + "participant.connect": service{ + Endpoints: serviceEndpoints{ + endpointKey{ + Region: "af-south-1", + }: endpoint{}, + endpointKey{ + Region: "ap-northeast-1", + }: endpoint{}, + endpointKey{ + Region: "ap-northeast-2", + }: endpoint{}, + endpointKey{ + Region: "ap-southeast-1", + }: endpoint{}, + endpointKey{ + Region: "ap-southeast-2", + }: endpoint{}, + endpointKey{ + Region: "ca-central-1", + }: endpoint{}, + endpointKey{ + Region: "eu-central-1", + }: endpoint{}, + endpointKey{ + Region: "eu-west-2", + }: endpoint{}, + endpointKey{ + Region: "fips-us-east-1", + }: endpoint{ + Hostname: "participant.connect-fips.us-east-1.amazonaws.com", + CredentialScope: credentialScope{ + Region: "us-east-1", + }, + Deprecated: boxedTrue, + }, + endpointKey{ + Region: "fips-us-west-2", + }: endpoint{ + Hostname: "participant.connect-fips.us-west-2.amazonaws.com", + CredentialScope: credentialScope{ + Region: "us-west-2", + }, + Deprecated: boxedTrue, + }, + endpointKey{ + Region: "us-east-1", + }: endpoint{}, + endpointKey{ + Region: "us-east-1", + Variant: fipsVariant, + }: endpoint{ + Hostname: "participant.connect-fips.us-east-1.amazonaws.com", + }, + endpointKey{ + Region: "us-west-2", + }: endpoint{}, + endpointKey{ + Region: "us-west-2", + Variant: fipsVariant, + }: endpoint{ + Hostname: "participant.connect-fips.us-west-2.amazonaws.com", + }, + }, + }, "personalize": service{ Endpoints: serviceEndpoints{ endpointKey{ @@ -18710,6 +18774,9 @@ var awsPartition = partition{ }, "rolesanywhere": service{ Endpoints: serviceEndpoints{ + endpointKey{ + Region: "af-south-1", + }: endpoint{}, endpointKey{ Region: "ap-east-1", }: endpoint{}, @@ -18731,6 +18798,9 @@ var awsPartition = partition{ endpointKey{ Region: "ap-southeast-2", }: endpoint{}, + endpointKey{ + Region: "ap-southeast-3", + }: endpoint{}, endpointKey{ Region: "ca-central-1", }: endpoint{}, @@ -18740,6 +18810,9 @@ var awsPartition = partition{ endpointKey{ Region: "eu-north-1", }: endpoint{}, + endpointKey{ + Region: "eu-south-1", + }: endpoint{}, endpointKey{ Region: "eu-west-1", }: endpoint{}, @@ -21533,6 +21606,9 @@ var awsPartition = partition{ }: endpoint{ Hostname: "snowball-fips.ap-southeast-2.amazonaws.com", }, + endpointKey{ + Region: "ap-southeast-3", + }: endpoint{}, endpointKey{ Region: "ca-central-1", }: endpoint{}, @@ -25262,6 +25338,9 @@ var awsPartition = partition{ }, "workspaces": service{ Endpoints: serviceEndpoints{ + endpointKey{ + Region: "af-south-1", + }: endpoint{}, endpointKey{ Region: "ap-northeast-1", }: endpoint{}, @@ -30495,6 +30574,18 @@ var awsusgovPartition = partition{ }, }, }, + "participant.connect": service{ + Endpoints: serviceEndpoints{ + endpointKey{ + Region: "us-gov-west-1", + }: endpoint{ + Hostname: "participant.connect.us-gov-west-1.amazonaws.com", + CredentialScope: credentialScope{ + Region: "us-gov-west-1", + }, + }, + }, + }, "pinpoint": service{ Defaults: endpointDefaults{ defaultKey{}: endpoint{ diff --git a/vendor/github.com/aws/aws-sdk-go/aws/version.go b/vendor/github.com/aws/aws-sdk-go/aws/version.go index 43c5f9307..7922995d0 100644 --- a/vendor/github.com/aws/aws-sdk-go/aws/version.go +++ b/vendor/github.com/aws/aws-sdk-go/aws/version.go @@ -5,4 +5,4 @@ package aws const SDKName = "aws-sdk-go" // SDKVersion is the version of this SDK -const SDKVersion = "1.44.96" +const SDKVersion = "1.44.100" diff --git a/vendor/github.com/klauspost/compress/README.md b/vendor/github.com/klauspost/compress/README.md index ad5c63a82..2d6b01077 100644 --- a/vendor/github.com/klauspost/compress/README.md +++ b/vendor/github.com/klauspost/compress/README.md @@ -17,6 +17,12 @@ This package provides various compression algorithms. # changelog +* July 21, 2022 (v1.15.9) + + * zstd: Fix decoder crash on amd64 (no BMI) on invalid input https://github.com/klauspost/compress/pull/645 + * zstd: Disable decoder extended memory copies (amd64) due to possible crashes https://github.com/klauspost/compress/pull/644 + * zstd: Allow single segments up to "max decoded size" by @klauspost in https://github.com/klauspost/compress/pull/643 + * July 13, 2022 (v1.15.8) * gzip: fix stack exhaustion bug in Reader.Read https://github.com/klauspost/compress/pull/641 diff --git a/vendor/github.com/klauspost/compress/flate/deflate.go b/vendor/github.com/klauspost/compress/flate/deflate.go index f8435998e..f00da5b21 100644 --- a/vendor/github.com/klauspost/compress/flate/deflate.go +++ b/vendor/github.com/klauspost/compress/flate/deflate.go @@ -131,7 +131,8 @@ func (d *compressor) fillDeflate(b []byte) int { s := d.state if s.index >= 2*windowSize-(minMatchLength+maxMatchLength) { // shift the window by windowSize - copy(d.window[:], d.window[windowSize:2*windowSize]) + //copy(d.window[:], d.window[windowSize:2*windowSize]) + *(*[windowSize]byte)(d.window) = *(*[windowSize]byte)(d.window[windowSize:]) s.index -= windowSize d.windowEnd -= windowSize if d.blockStart >= windowSize { diff --git a/vendor/github.com/klauspost/compress/flate/dict_decoder.go b/vendor/github.com/klauspost/compress/flate/dict_decoder.go index 71c75a065..bb36351a5 100644 --- a/vendor/github.com/klauspost/compress/flate/dict_decoder.go +++ b/vendor/github.com/klauspost/compress/flate/dict_decoder.go @@ -7,19 +7,19 @@ package flate // dictDecoder implements the LZ77 sliding dictionary as used in decompression. // LZ77 decompresses data through sequences of two forms of commands: // -// * Literal insertions: Runs of one or more symbols are inserted into the data -// stream as is. This is accomplished through the writeByte method for a -// single symbol, or combinations of writeSlice/writeMark for multiple symbols. -// Any valid stream must start with a literal insertion if no preset dictionary -// is used. +// - Literal insertions: Runs of one or more symbols are inserted into the data +// stream as is. This is accomplished through the writeByte method for a +// single symbol, or combinations of writeSlice/writeMark for multiple symbols. +// Any valid stream must start with a literal insertion if no preset dictionary +// is used. // -// * Backward copies: Runs of one or more symbols are copied from previously -// emitted data. Backward copies come as the tuple (dist, length) where dist -// determines how far back in the stream to copy from and length determines how -// many bytes to copy. Note that it is valid for the length to be greater than -// the distance. Since LZ77 uses forward copies, that situation is used to -// perform a form of run-length encoding on repeated runs of symbols. -// The writeCopy and tryWriteCopy are used to implement this command. +// - Backward copies: Runs of one or more symbols are copied from previously +// emitted data. Backward copies come as the tuple (dist, length) where dist +// determines how far back in the stream to copy from and length determines how +// many bytes to copy. Note that it is valid for the length to be greater than +// the distance. Since LZ77 uses forward copies, that situation is used to +// perform a form of run-length encoding on repeated runs of symbols. +// The writeCopy and tryWriteCopy are used to implement this command. // // For performance reasons, this implementation performs little to no sanity // checks about the arguments. As such, the invariants documented for each diff --git a/vendor/github.com/klauspost/compress/flate/fast_encoder.go b/vendor/github.com/klauspost/compress/flate/fast_encoder.go index f781aaa62..cd77a2cc4 100644 --- a/vendor/github.com/klauspost/compress/flate/fast_encoder.go +++ b/vendor/github.com/klauspost/compress/flate/fast_encoder.go @@ -104,7 +104,8 @@ func (e *fastGen) addBlock(src []byte) int32 { } // Move down offset := int32(len(e.hist)) - maxMatchOffset - copy(e.hist[0:maxMatchOffset], e.hist[offset:]) + // copy(e.hist[0:maxMatchOffset], e.hist[offset:]) + *(*[maxMatchOffset]byte)(e.hist) = *(*[maxMatchOffset]byte)(e.hist[offset:]) e.cur += offset e.hist = e.hist[:maxMatchOffset] } diff --git a/vendor/github.com/klauspost/compress/flate/huffman_bit_writer.go b/vendor/github.com/klauspost/compress/flate/huffman_bit_writer.go index 40ef45c2f..89a5dd89f 100644 --- a/vendor/github.com/klauspost/compress/flate/huffman_bit_writer.go +++ b/vendor/github.com/klauspost/compress/flate/huffman_bit_writer.go @@ -265,9 +265,9 @@ func (w *huffmanBitWriter) writeBytes(bytes []byte) { // Codes 0-15 are single byte codes. Codes 16-18 are followed by additional // information. Code badCode is an end marker // -// numLiterals The number of literals in literalEncoding -// numOffsets The number of offsets in offsetEncoding -// litenc, offenc The literal and offset encoder to use +// numLiterals The number of literals in literalEncoding +// numOffsets The number of offsets in offsetEncoding +// litenc, offenc The literal and offset encoder to use func (w *huffmanBitWriter) generateCodegen(numLiterals int, numOffsets int, litEnc, offEnc *huffmanEncoder) { for i := range w.codegenFreq { w.codegenFreq[i] = 0 @@ -460,9 +460,9 @@ func (w *huffmanBitWriter) writeOutBits() { // Write the header of a dynamic Huffman block to the output stream. // -// numLiterals The number of literals specified in codegen -// numOffsets The number of offsets specified in codegen -// numCodegens The number of codegens used in codegen +// numLiterals The number of literals specified in codegen +// numOffsets The number of offsets specified in codegen +// numCodegens The number of codegens used in codegen func (w *huffmanBitWriter) writeDynamicHeader(numLiterals int, numOffsets int, numCodegens int, isEof bool) { if w.err != nil { return @@ -790,9 +790,11 @@ func (w *huffmanBitWriter) fillTokens() { // and offsetEncoding. // The number of literal and offset tokens is returned. func (w *huffmanBitWriter) indexTokens(t *tokens, filled bool) (numLiterals, numOffsets int) { - copy(w.literalFreq[:], t.litHist[:]) - copy(w.literalFreq[256:], t.extraHist[:]) - copy(w.offsetFreq[:], t.offHist[:offsetCodeCount]) + //copy(w.literalFreq[:], t.litHist[:]) + *(*[256]uint16)(w.literalFreq[:]) = t.litHist + //copy(w.literalFreq[256:], t.extraHist[:]) + *(*[32]uint16)(w.literalFreq[256:]) = t.extraHist + w.offsetFreq = t.offHist if t.n == 0 { return diff --git a/vendor/github.com/klauspost/compress/flate/huffman_code.go b/vendor/github.com/klauspost/compress/flate/huffman_code.go index 5ac144f28..be7b58b47 100644 --- a/vendor/github.com/klauspost/compress/flate/huffman_code.go +++ b/vendor/github.com/klauspost/compress/flate/huffman_code.go @@ -168,13 +168,18 @@ func (h *huffmanEncoder) canReuseBits(freq []uint16) int { // The cases of 0, 1, and 2 literals are handled by special case code. // // list An array of the literals with non-zero frequencies -// and their associated frequencies. The array is in order of increasing -// frequency, and has as its last element a special element with frequency -// MaxInt32 +// +// and their associated frequencies. The array is in order of increasing +// frequency, and has as its last element a special element with frequency +// MaxInt32 +// // maxBits The maximum number of bits that should be used to encode any literal. -// Must be less than 16. +// +// Must be less than 16. +// // return An integer array in which array[i] indicates the number of literals -// that should be encoded in i bits. +// +// that should be encoded in i bits. func (h *huffmanEncoder) bitCounts(list []literalNode, maxBits int32) []int32 { if maxBits >= maxBitsLimit { panic("flate: maxBits too large") diff --git a/vendor/github.com/klauspost/compress/flate/level5.go b/vendor/github.com/klauspost/compress/flate/level5.go index 4b97576bd..ef6339d95 100644 --- a/vendor/github.com/klauspost/compress/flate/level5.go +++ b/vendor/github.com/klauspost/compress/flate/level5.go @@ -191,14 +191,21 @@ func (e *fastEncL5) Encode(dst *tokens, src []byte) { // Try to locate a better match by checking the end of best match... if sAt := s + l; l < 30 && sAt < sLimit { + // Allow some bytes at the beginning to mismatch. + // Sweet spot is 2/3 bytes depending on input. + // 3 is only a little better when it is but sometimes a lot worse. + // The skipped bytes are tested in Extend backwards, + // and still picked up as part of the match if they do. + const skipBeginning = 2 eLong := e.bTable[hash7(load6432(src, sAt), tableBits)].Cur.offset - // Test current - t2 := eLong - e.cur - l - off := s - t2 + t2 := eLong - e.cur - l + skipBeginning + s2 := s + skipBeginning + off := s2 - t2 if t2 >= 0 && off < maxMatchOffset && off > 0 { - if l2 := e.matchlenLong(s, t2, src); l2 > l { + if l2 := e.matchlenLong(s2, t2, src); l2 > l { t = t2 l = l2 + s = s2 } } } diff --git a/vendor/github.com/klauspost/compress/flate/level6.go b/vendor/github.com/klauspost/compress/flate/level6.go index 62888edf3..85e4b2095 100644 --- a/vendor/github.com/klauspost/compress/flate/level6.go +++ b/vendor/github.com/klauspost/compress/flate/level6.go @@ -213,24 +213,33 @@ func (e *fastEncL6) Encode(dst *tokens, src []byte) { // Try to locate a better match by checking the end-of-match... if sAt := s + l; sAt < sLimit { + // Allow some bytes at the beginning to mismatch. + // Sweet spot is 2/3 bytes depending on input. + // 3 is only a little better when it is but sometimes a lot worse. + // The skipped bytes are tested in Extend backwards, + // and still picked up as part of the match if they do. + const skipBeginning = 2 eLong := &e.bTable[hash7(load6432(src, sAt), tableBits)] // Test current - t2 := eLong.Cur.offset - e.cur - l - off := s - t2 + t2 := eLong.Cur.offset - e.cur - l + skipBeginning + s2 := s + skipBeginning + off := s2 - t2 if off < maxMatchOffset { if off > 0 && t2 >= 0 { - if l2 := e.matchlenLong(s, t2, src); l2 > l { + if l2 := e.matchlenLong(s2, t2, src); l2 > l { t = t2 l = l2 + s = s2 } } // Test next: - t2 = eLong.Prev.offset - e.cur - l - off := s - t2 + t2 = eLong.Prev.offset - e.cur - l + skipBeginning + off := s2 - t2 if off > 0 && off < maxMatchOffset && t2 >= 0 { - if l2 := e.matchlenLong(s, t2, src); l2 > l { + if l2 := e.matchlenLong(s2, t2, src); l2 > l { t = t2 l = l2 + s = s2 } } } diff --git a/vendor/github.com/klauspost/compress/huff0/decompress.go b/vendor/github.com/klauspost/compress/huff0/decompress.go index c0c48bd70..42a237eac 100644 --- a/vendor/github.com/klauspost/compress/huff0/decompress.go +++ b/vendor/github.com/klauspost/compress/huff0/decompress.go @@ -763,17 +763,20 @@ func (d *Decoder) decompress4X8bit(dst, src []byte) ([]byte, error) { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 1") } - copy(out, buf[0][:]) - copy(out[dstEvery:], buf[1][:]) - copy(out[dstEvery*2:], buf[2][:]) - copy(out[dstEvery*3:], buf[3][:]) - out = out[bufoff:] - decoded += bufoff * 4 // There must at least be 3 buffers left. - if len(out) < dstEvery*3 { + if len(out)-bufoff < dstEvery*3 { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 2") } + //copy(out, buf[0][:]) + //copy(out[dstEvery:], buf[1][:]) + //copy(out[dstEvery*2:], buf[2][:]) + *(*[bufoff]byte)(out) = buf[0] + *(*[bufoff]byte)(out[dstEvery:]) = buf[1] + *(*[bufoff]byte)(out[dstEvery*2:]) = buf[2] + *(*[bufoff]byte)(out[dstEvery*3:]) = buf[3] + out = out[bufoff:] + decoded += bufoff * 4 } } if off > 0 { @@ -997,17 +1000,22 @@ func (d *Decoder) decompress4X8bitExactly(dst, src []byte) ([]byte, error) { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 1") } - copy(out, buf[0][:]) - copy(out[dstEvery:], buf[1][:]) - copy(out[dstEvery*2:], buf[2][:]) - copy(out[dstEvery*3:], buf[3][:]) - out = out[bufoff:] - decoded += bufoff * 4 // There must at least be 3 buffers left. - if len(out) < dstEvery*3 { + if len(out)-bufoff < dstEvery*3 { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 2") } + + //copy(out, buf[0][:]) + //copy(out[dstEvery:], buf[1][:]) + //copy(out[dstEvery*2:], buf[2][:]) + // copy(out[dstEvery*3:], buf[3][:]) + *(*[bufoff]byte)(out) = buf[0] + *(*[bufoff]byte)(out[dstEvery:]) = buf[1] + *(*[bufoff]byte)(out[dstEvery*2:]) = buf[2] + *(*[bufoff]byte)(out[dstEvery*3:]) = buf[3] + out = out[bufoff:] + decoded += bufoff * 4 } } if off > 0 { diff --git a/vendor/github.com/klauspost/compress/huff0/decompress_amd64.go b/vendor/github.com/klauspost/compress/huff0/decompress_amd64.go index 9f3e9f79e..ba7e8e6b0 100644 --- a/vendor/github.com/klauspost/compress/huff0/decompress_amd64.go +++ b/vendor/github.com/klauspost/compress/huff0/decompress_amd64.go @@ -14,12 +14,14 @@ import ( // decompress4x_main_loop_x86 is an x86 assembler implementation // of Decompress4X when tablelog > 8. +// //go:noescape func decompress4x_main_loop_amd64(ctx *decompress4xContext) // decompress4x_8b_loop_x86 is an x86 assembler implementation // of Decompress4X when tablelog <= 8 which decodes 4 entries // per loop. +// //go:noescape func decompress4x_8b_main_loop_amd64(ctx *decompress4xContext) @@ -145,11 +147,13 @@ func (d *Decoder) Decompress4X(dst, src []byte) ([]byte, error) { // decompress4x_main_loop_x86 is an x86 assembler implementation // of Decompress1X when tablelog > 8. +// //go:noescape func decompress1x_main_loop_amd64(ctx *decompress1xContext) // decompress4x_main_loop_x86 is an x86 with BMI2 assembler implementation // of Decompress1X when tablelog > 8. +// //go:noescape func decompress1x_main_loop_bmi2(ctx *decompress1xContext) diff --git a/vendor/github.com/klauspost/compress/huff0/decompress_amd64.s b/vendor/github.com/klauspost/compress/huff0/decompress_amd64.s index dd1a5aecd..8d2187a2c 100644 --- a/vendor/github.com/klauspost/compress/huff0/decompress_amd64.s +++ b/vendor/github.com/klauspost/compress/huff0/decompress_amd64.s @@ -1,7 +1,6 @@ // Code generated by command: go run gen.go -out ../decompress_amd64.s -pkg=huff0. DO NOT EDIT. //go:build amd64 && !appengine && !noasm && gc -// +build amd64,!appengine,!noasm,gc // func decompress4x_main_loop_amd64(ctx *decompress4xContext) TEXT ·decompress4x_main_loop_amd64(SB), $0-8 diff --git a/vendor/github.com/klauspost/compress/huff0/decompress_generic.go b/vendor/github.com/klauspost/compress/huff0/decompress_generic.go index 4f6f37cb2..908c17de6 100644 --- a/vendor/github.com/klauspost/compress/huff0/decompress_generic.go +++ b/vendor/github.com/klauspost/compress/huff0/decompress_generic.go @@ -122,17 +122,21 @@ func (d *Decoder) Decompress4X(dst, src []byte) ([]byte, error) { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 1") } - copy(out, buf[0][:]) - copy(out[dstEvery:], buf[1][:]) - copy(out[dstEvery*2:], buf[2][:]) - copy(out[dstEvery*3:], buf[3][:]) - out = out[bufoff:] - decoded += bufoff * 4 // There must at least be 3 buffers left. - if len(out) < dstEvery*3 { + if len(out)-bufoff < dstEvery*3 { d.bufs.Put(buf) return nil, errors.New("corruption detected: stream overrun 2") } + //copy(out, buf[0][:]) + //copy(out[dstEvery:], buf[1][:]) + //copy(out[dstEvery*2:], buf[2][:]) + //copy(out[dstEvery*3:], buf[3][:]) + *(*[bufoff]byte)(out) = buf[0] + *(*[bufoff]byte)(out[dstEvery:]) = buf[1] + *(*[bufoff]byte)(out[dstEvery*2:]) = buf[2] + *(*[bufoff]byte)(out[dstEvery*3:]) = buf[3] + out = out[bufoff:] + decoded += bufoff * 4 } } if off > 0 { diff --git a/vendor/github.com/klauspost/compress/internal/snapref/encode_other.go b/vendor/github.com/klauspost/compress/internal/snapref/encode_other.go index 511bba65d..298c4f8e9 100644 --- a/vendor/github.com/klauspost/compress/internal/snapref/encode_other.go +++ b/vendor/github.com/klauspost/compress/internal/snapref/encode_other.go @@ -18,6 +18,7 @@ func load64(b []byte, i int) uint64 { // emitLiteral writes a literal chunk and returns the number of bytes written. // // It assumes that: +// // dst is long enough to hold the encoded bytes // 1 <= len(lit) && len(lit) <= 65536 func emitLiteral(dst, lit []byte) int { @@ -42,6 +43,7 @@ func emitLiteral(dst, lit []byte) int { // emitCopy writes a copy chunk and returns the number of bytes written. // // It assumes that: +// // dst is long enough to hold the encoded bytes // 1 <= offset && offset <= 65535 // 4 <= length && length <= 65535 @@ -89,6 +91,7 @@ func emitCopy(dst []byte, offset, length int) int { // src[i:i+k-j] and src[j:k] have the same contents. // // It assumes that: +// // 0 <= i && i < j && j <= len(src) func extendMatch(src []byte, i, j int) int { for ; j < len(src) && src[i] == src[j]; i, j = i+1, j+1 { @@ -105,8 +108,9 @@ func hash(u, shift uint32) uint32 { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlock(dst, src []byte) (d int) { // Initialize the hash table. Its size ranges from 1<<8 to 1<<14 inclusive. // The table element type is uint16, as s < sLimit and sLimit < len(src) diff --git a/vendor/github.com/klauspost/compress/s2/README.md b/vendor/github.com/klauspost/compress/s2/README.md index 73c0c462d..700f997c2 100644 --- a/vendor/github.com/klauspost/compress/s2/README.md +++ b/vendor/github.com/klauspost/compress/s2/README.md @@ -835,6 +835,13 @@ This is done using the regular "Skip" function: This will ensure that we are at exactly the offset we want, and reading from `dec` will start at the requested offset. +# Compact storage + +For compact storage [RemoveIndexHeaders](https://pkg.go.dev/github.com/klauspost/compress/s2#RemoveIndexHeaders) can be used to remove any redundant info from +a serialized index. If you remove the header it must be restored before [Loading](https://pkg.go.dev/github.com/klauspost/compress/s2#Index.Load). + +This is expected to save 20 bytes. These can be restored using [RestoreIndexHeaders](https://pkg.go.dev/github.com/klauspost/compress/s2#RestoreIndexHeaders). This removes a layer of security, but is the most compact representation. Returns nil if headers contains errors. + ## Index Format: Each block is structured as a snappy skippable block, with the chunk ID 0x99. @@ -929,6 +936,7 @@ To decode from any given uncompressed offset `(wantOffset)`: See [using indexes](https://github.com/klauspost/compress/tree/master/s2#using-indexes) for functions that perform the operations with a simpler interface. + # Format Extensions * Frame [Stream identifier](https://github.com/google/snappy/blob/master/framing_format.txt#L68) changed from `sNaPpY` to `S2sTwO`. @@ -951,10 +959,11 @@ The length is specified by reading the 3-bit length specified in the tag and dec | 7 | 65540 + read 3 bytes | This allows any repeat offset + length to be represented by 2 to 5 bytes. +It also allows to emit matches longer than 64 bytes with one copy + one repeat instead of several 64 byte copies. Lengths are stored as little endian values. -The first copy of a block cannot be a repeat offset and the offset is not carried across blocks in streams. +The first copy of a block cannot be a repeat offset and the offset is reset on every block in streams. Default streaming block size is 1MB. diff --git a/vendor/github.com/klauspost/compress/s2/decode_other.go b/vendor/github.com/klauspost/compress/s2/decode_other.go index 1074ebd21..11300c3a8 100644 --- a/vendor/github.com/klauspost/compress/s2/decode_other.go +++ b/vendor/github.com/klauspost/compress/s2/decode_other.go @@ -28,6 +28,9 @@ func s2Decode(dst, src []byte) int { // As long as we can read at least 5 bytes... for s < len(src)-5 { + // Removing bounds checks is SLOWER, when if doing + // in := src[s:s+5] + // Checked on Go 1.18 switch src[s] & 0x03 { case tagLiteral: x := uint32(src[s] >> 2) @@ -38,14 +41,19 @@ func s2Decode(dst, src []byte) int { s += 2 x = uint32(src[s-1]) case x == 61: + in := src[s : s+3] + x = uint32(in[1]) | uint32(in[2])<<8 s += 3 - x = uint32(src[s-2]) | uint32(src[s-1])<<8 case x == 62: + in := src[s : s+4] + // Load as 32 bit and shift down. + x = uint32(in[0]) | uint32(in[1])<<8 | uint32(in[2])<<16 | uint32(in[3])<<24 + x >>= 8 s += 4 - x = uint32(src[s-3]) | uint32(src[s-2])<<8 | uint32(src[s-1])<<16 case x == 63: + in := src[s : s+5] + x = uint32(in[1]) | uint32(in[2])<<8 | uint32(in[3])<<16 | uint32(in[4])<<24 s += 5 - x = uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24 } length = int(x) + 1 if length > len(dst)-d || length > len(src)-s || (strconv.IntSize == 32 && length <= 0) { @@ -62,8 +70,8 @@ func s2Decode(dst, src []byte) int { case tagCopy1: s += 2 - length = int(src[s-2]) >> 2 & 0x7 toffset := int(uint32(src[s-2])&0xe0<<3 | uint32(src[s-1])) + length = int(src[s-2]) >> 2 & 0x7 if toffset == 0 { if debug { fmt.Print("(repeat) ") @@ -71,14 +79,16 @@ func s2Decode(dst, src []byte) int { // keep last offset switch length { case 5: + length = int(src[s]) + 4 s += 1 - length = int(uint32(src[s-1])) + 4 case 6: + in := src[s : s+2] + length = int(uint32(in[0])|(uint32(in[1])<<8)) + (1 << 8) s += 2 - length = int(uint32(src[s-2])|(uint32(src[s-1])<<8)) + (1 << 8) case 7: + in := src[s : s+3] + length = int((uint32(in[2])<<16)|(uint32(in[1])<<8)|uint32(in[0])) + (1 << 16) s += 3 - length = int(uint32(src[s-3])|(uint32(src[s-2])<<8)|(uint32(src[s-1])<<16)) + (1 << 16) default: // 0-> 4 } } else { @@ -86,14 +96,16 @@ func s2Decode(dst, src []byte) int { } length += 4 case tagCopy2: + in := src[s : s+3] + offset = int(uint32(in[1]) | uint32(in[2])<<8) + length = 1 + int(in[0])>>2 s += 3 - length = 1 + int(src[s-3])>>2 - offset = int(uint32(src[s-2]) | uint32(src[s-1])<<8) case tagCopy4: + in := src[s : s+5] + offset = int(uint32(in[1]) | uint32(in[2])<<8 | uint32(in[3])<<16 | uint32(in[4])<<24) + length = 1 + int(in[0])>>2 s += 5 - length = 1 + int(src[s-5])>>2 - offset = int(uint32(src[s-4]) | uint32(src[s-3])<<8 | uint32(src[s-2])<<16 | uint32(src[s-1])<<24) } if offset <= 0 || d < offset || length > len(dst)-d { diff --git a/vendor/github.com/klauspost/compress/s2/encode_all.go b/vendor/github.com/klauspost/compress/s2/encode_all.go index 8b16c38a6..54c71d3b5 100644 --- a/vendor/github.com/klauspost/compress/s2/encode_all.go +++ b/vendor/github.com/klauspost/compress/s2/encode_all.go @@ -58,8 +58,9 @@ func encodeGo(dst, src []byte) []byte { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockGo(dst, src []byte) (d int) { // Initialize the hash table. const ( diff --git a/vendor/github.com/klauspost/compress/s2/encode_amd64.go b/vendor/github.com/klauspost/compress/s2/encode_amd64.go index e612225f4..6b93daa5a 100644 --- a/vendor/github.com/klauspost/compress/s2/encode_amd64.go +++ b/vendor/github.com/klauspost/compress/s2/encode_amd64.go @@ -8,8 +8,9 @@ package s2 // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlock(dst, src []byte) (d int) { const ( // Use 12 bit table when less than... @@ -43,8 +44,9 @@ func encodeBlock(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBetter(dst, src []byte) (d int) { const ( // Use 12 bit table when less than... @@ -78,8 +80,9 @@ func encodeBlockBetter(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockSnappy(dst, src []byte) (d int) { const ( // Use 12 bit table when less than... @@ -112,8 +115,9 @@ func encodeBlockSnappy(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBetterSnappy(dst, src []byte) (d int) { const ( // Use 12 bit table when less than... diff --git a/vendor/github.com/klauspost/compress/s2/encode_best.go b/vendor/github.com/klauspost/compress/s2/encode_best.go index 4bc80bc6a..1b7ea394f 100644 --- a/vendor/github.com/klauspost/compress/s2/encode_best.go +++ b/vendor/github.com/klauspost/compress/s2/encode_best.go @@ -15,8 +15,9 @@ import ( // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBest(dst, src []byte) (d int) { // Initialize the hash tables. const ( @@ -176,14 +177,21 @@ func encodeBlockBest(dst, src []byte) (d int) { best = bestOf(best, matchAt(getPrev(nextLong), s, uint32(cv), false)) } // Search for a match at best match end, see if that is better. - if sAt := best.s + best.length; sAt < sLimit { - sBack := best.s - backL := best.length + // Allow some bytes at the beginning to mismatch. + // Sweet spot is around 1-2 bytes, but depends on input. + // The skipped bytes are tested in Extend backwards, + // and still picked up as part of the match if they do. + const skipBeginning = 2 + const skipEnd = 1 + if sAt := best.s + best.length - skipEnd; sAt < sLimit { + + sBack := best.s + skipBeginning - skipEnd + backL := best.length - skipBeginning // Load initial values cv = load64(src, sBack) - // Search for mismatch + + // Grab candidates... next := lTable[hash8(load64(src, sAt), lTableBits)] - //next := sTable[hash4(load64(src, sAt), sTableBits)] if checkAt := getCur(next) - backL; checkAt > 0 { best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) @@ -191,6 +199,16 @@ func encodeBlockBest(dst, src []byte) (d int) { if checkAt := getPrev(next) - backL; checkAt > 0 { best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) } + // Disabled: Extremely small gain + if false { + next = sTable[hash4(load64(src, sAt), sTableBits)] + if checkAt := getCur(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) + } + if checkAt := getPrev(next) - backL; checkAt > 0 { + best = bestOf(best, matchAt(checkAt, sBack, uint32(cv), false)) + } + } } } } @@ -288,8 +306,9 @@ emitRemainder: // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBestSnappy(dst, src []byte) (d int) { // Initialize the hash tables. const ( @@ -546,6 +565,7 @@ emitRemainder: // emitCopySize returns the size to encode the offset+length // // It assumes that: +// // 1 <= offset && offset <= math.MaxUint32 // 4 <= length && length <= 1 << 24 func emitCopySize(offset, length int) int { @@ -584,6 +604,7 @@ func emitCopySize(offset, length int) int { // emitCopyNoRepeatSize returns the size to encode the offset+length // // It assumes that: +// // 1 <= offset && offset <= math.MaxUint32 // 4 <= length && length <= 1 << 24 func emitCopyNoRepeatSize(offset, length int) int { diff --git a/vendor/github.com/klauspost/compress/s2/encode_better.go b/vendor/github.com/klauspost/compress/s2/encode_better.go index 943215b8a..3b66ba42b 100644 --- a/vendor/github.com/klauspost/compress/s2/encode_better.go +++ b/vendor/github.com/klauspost/compress/s2/encode_better.go @@ -42,8 +42,9 @@ func hash8(u uint64, h uint8) uint32 { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBetterGo(dst, src []byte) (d int) { // sLimit is when to stop looking for offset/length copies. The inputMargin // lets us use a fast path for emitLiteral in the main loop, while we are @@ -56,7 +57,7 @@ func encodeBlockBetterGo(dst, src []byte) (d int) { // Initialize the hash tables. const ( // Long hash matches. - lTableBits = 16 + lTableBits = 17 maxLTableSize = 1 << lTableBits // Short hash matches. @@ -97,9 +98,26 @@ func encodeBlockBetterGo(dst, src []byte) (d int) { lTable[hashL] = uint32(s) sTable[hashS] = uint32(s) + valLong := load64(src, candidateL) + valShort := load64(src, candidateS) + + // If long matches at least 8 bytes, use that. + if cv == valLong { + break + } + if cv == valShort { + candidateL = candidateS + break + } + // Check repeat at offset checkRep. const checkRep = 1 - if false && uint32(cv>>(checkRep*8)) == load32(src, s-repeat+checkRep) { + // Minimum length of a repeat. Tested with various values. + // While 4-5 offers improvements in some, 6 reduces + // regressions significantly. + const wantRepeatBytes = 6 + const repeatMask = ((1 << (wantRepeatBytes * 8)) - 1) << (8 * checkRep) + if false && repeat > 0 && cv&repeatMask == load64(src, s-repeat)&repeatMask { base := s + checkRep // Extend back for i := base - repeat; base > nextEmit && i > 0 && src[i-1] == src[base-1]; { @@ -109,8 +127,8 @@ func encodeBlockBetterGo(dst, src []byte) (d int) { d += emitLiteral(dst[d:], src[nextEmit:base]) // Extend forward - candidate := s - repeat + 4 + checkRep - s += 4 + checkRep + candidate := s - repeat + wantRepeatBytes + checkRep + s += wantRepeatBytes + checkRep for s < len(src) { if len(src)-s < 8 { if src[s] == src[candidate] { @@ -127,28 +145,40 @@ func encodeBlockBetterGo(dst, src []byte) (d int) { s += 8 candidate += 8 } - if nextEmit > 0 { - // same as `add := emitCopy(dst[d:], repeat, s-base)` but skips storing offset. - d += emitRepeat(dst[d:], repeat, s-base) - } else { - // First match, cannot be repeat. - d += emitCopy(dst[d:], repeat, s-base) - } + // same as `add := emitCopy(dst[d:], repeat, s-base)` but skips storing offset. + d += emitRepeat(dst[d:], repeat, s-base) nextEmit = s if s >= sLimit { goto emitRemainder } + // Index in-between + index0 := base + 1 + index1 := s - 2 + + cv = load64(src, s) + for index0 < index1 { + cv0 := load64(src, index0) + cv1 := load64(src, index1) + lTable[hash7(cv0, lTableBits)] = uint32(index0) + sTable[hash4(cv0>>8, sTableBits)] = uint32(index0 + 1) + + lTable[hash7(cv1, lTableBits)] = uint32(index1) + sTable[hash4(cv1>>8, sTableBits)] = uint32(index1 + 1) + index0 += 2 + index1 -= 2 + } cv = load64(src, s) continue } - if uint32(cv) == load32(src, candidateL) { + // Long likely matches 7, so take that. + if uint32(cv) == uint32(valLong) { break } // Check our short candidate - if uint32(cv) == load32(src, candidateS) { + if uint32(cv) == uint32(valShort) { // Try a long candidate at s+1 hashL = hash7(cv>>8, lTableBits) candidateL = int(lTable[hashL]) @@ -227,21 +257,29 @@ func encodeBlockBetterGo(dst, src []byte) (d int) { // Do we have space for more, if not bail. return 0 } - // Index match start+1 (long) and start+2 (short) + + // Index short & long index0 := base + 1 - // Index match end-2 (long) and end-1 (short) index1 := s - 2 cv0 := load64(src, index0) cv1 := load64(src, index1) - cv = load64(src, s) lTable[hash7(cv0, lTableBits)] = uint32(index0) - lTable[hash7(cv0>>8, lTableBits)] = uint32(index0 + 1) - lTable[hash7(cv1, lTableBits)] = uint32(index1) - lTable[hash7(cv1>>8, lTableBits)] = uint32(index1 + 1) sTable[hash4(cv0>>8, sTableBits)] = uint32(index0 + 1) - sTable[hash4(cv0>>16, sTableBits)] = uint32(index0 + 2) + + lTable[hash7(cv1, lTableBits)] = uint32(index1) sTable[hash4(cv1>>8, sTableBits)] = uint32(index1 + 1) + index0 += 1 + index1 -= 1 + cv = load64(src, s) + + // index every second long in between. + for index0 < index1 { + lTable[hash7(load64(src, index0), lTableBits)] = uint32(index0) + lTable[hash7(load64(src, index1), lTableBits)] = uint32(index1) + index0 += 2 + index1 -= 2 + } } emitRemainder: @@ -260,8 +298,9 @@ emitRemainder: // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) && -// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize +// minNonLiteralBlockSize <= len(src) && len(src) <= maxBlockSize func encodeBlockBetterSnappyGo(dst, src []byte) (d int) { // sLimit is when to stop looking for offset/length copies. The inputMargin // lets us use a fast path for emitLiteral in the main loop, while we are @@ -402,21 +441,29 @@ func encodeBlockBetterSnappyGo(dst, src []byte) (d int) { // Do we have space for more, if not bail. return 0 } - // Index match start+1 (long) and start+2 (short) + + // Index short & long index0 := base + 1 - // Index match end-2 (long) and end-1 (short) index1 := s - 2 cv0 := load64(src, index0) cv1 := load64(src, index1) - cv = load64(src, s) lTable[hash7(cv0, lTableBits)] = uint32(index0) - lTable[hash7(cv0>>8, lTableBits)] = uint32(index0 + 1) - lTable[hash7(cv1, lTableBits)] = uint32(index1) - lTable[hash7(cv1>>8, lTableBits)] = uint32(index1 + 1) sTable[hash4(cv0>>8, sTableBits)] = uint32(index0 + 1) - sTable[hash4(cv0>>16, sTableBits)] = uint32(index0 + 2) + + lTable[hash7(cv1, lTableBits)] = uint32(index1) sTable[hash4(cv1>>8, sTableBits)] = uint32(index1 + 1) + index0 += 1 + index1 -= 1 + cv = load64(src, s) + + // index every second long in between. + for index0 < index1 { + lTable[hash7(load64(src, index0), lTableBits)] = uint32(index0) + lTable[hash7(load64(src, index1), lTableBits)] = uint32(index1) + index0 += 2 + index1 -= 2 + } } emitRemainder: diff --git a/vendor/github.com/klauspost/compress/s2/encode_go.go b/vendor/github.com/klauspost/compress/s2/encode_go.go index 94784b82a..db08fc355 100644 --- a/vendor/github.com/klauspost/compress/s2/encode_go.go +++ b/vendor/github.com/klauspost/compress/s2/encode_go.go @@ -12,6 +12,7 @@ import ( // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) func encodeBlock(dst, src []byte) (d int) { if len(src) < minNonLiteralBlockSize { @@ -25,6 +26,7 @@ func encodeBlock(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) func encodeBlockBetter(dst, src []byte) (d int) { return encodeBlockBetterGo(dst, src) @@ -35,6 +37,7 @@ func encodeBlockBetter(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) func encodeBlockBetterSnappy(dst, src []byte) (d int) { return encodeBlockBetterSnappyGo(dst, src) @@ -45,6 +48,7 @@ func encodeBlockBetterSnappy(dst, src []byte) (d int) { // been written. // // It also assumes that: +// // len(dst) >= MaxEncodedLen(len(src)) func encodeBlockSnappy(dst, src []byte) (d int) { if len(src) < minNonLiteralBlockSize { @@ -56,6 +60,7 @@ func encodeBlockSnappy(dst, src []byte) (d int) { // emitLiteral writes a literal chunk and returns the number of bytes written. // // It assumes that: +// // dst is long enough to hold the encoded bytes // 0 <= len(lit) && len(lit) <= math.MaxUint32 func emitLiteral(dst, lit []byte) int { @@ -146,6 +151,7 @@ func emitRepeat(dst []byte, offset, length int) int { // emitCopy writes a copy chunk and returns the number of bytes written. // // It assumes that: +// // dst is long enough to hold the encoded bytes // 1 <= offset && offset <= math.MaxUint32 // 4 <= length && length <= 1 << 24 @@ -214,6 +220,7 @@ func emitCopy(dst []byte, offset, length int) int { // emitCopyNoRepeat writes a copy chunk and returns the number of bytes written. // // It assumes that: +// // dst is long enough to hold the encoded bytes // 1 <= offset && offset <= math.MaxUint32 // 4 <= length && length <= 1 << 24 @@ -273,8 +280,8 @@ func emitCopyNoRepeat(dst []byte, offset, length int) int { // matchLen returns how many bytes match in a and b // // It assumes that: -// len(a) <= len(b) // +// len(a) <= len(b) func matchLen(a []byte, b []byte) int { b = b[:len(a)] var checked int diff --git a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go index 88f27c099..7e00bac3e 100644 --- a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go +++ b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.go @@ -1,7 +1,6 @@ // Code generated by command: go run gen.go -out ../encodeblock_amd64.s -stubs ../encodeblock_amd64.go -pkg=s2. DO NOT EDIT. //go:build !appengine && !noasm && gc && !noasm -// +build !appengine,!noasm,gc,!noasm package s2 @@ -150,8 +149,9 @@ func encodeSnappyBetterBlockAsm8B(dst []byte, src []byte) int // emitLiteral writes a literal chunk and returns the number of bytes written. // // It assumes that: -// dst is long enough to hold the encoded bytes with margin of 0 bytes -// 0 <= len(lit) && len(lit) <= math.MaxUint32 +// +// dst is long enough to hold the encoded bytes with margin of 0 bytes +// 0 <= len(lit) && len(lit) <= math.MaxUint32 // //go:noescape func emitLiteral(dst []byte, lit []byte) int @@ -165,9 +165,10 @@ func emitRepeat(dst []byte, offset int, length int) int // emitCopy writes a copy chunk and returns the number of bytes written. // // It assumes that: -// dst is long enough to hold the encoded bytes -// 1 <= offset && offset <= math.MaxUint32 -// 4 <= length && length <= 1 << 24 +// +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 // //go:noescape func emitCopy(dst []byte, offset int, length int) int @@ -175,9 +176,10 @@ func emitCopy(dst []byte, offset int, length int) int // emitCopyNoRepeat writes a copy chunk and returns the number of bytes written. // // It assumes that: -// dst is long enough to hold the encoded bytes -// 1 <= offset && offset <= math.MaxUint32 -// 4 <= length && length <= 1 << 24 +// +// dst is long enough to hold the encoded bytes +// 1 <= offset && offset <= math.MaxUint32 +// 4 <= length && length <= 1 << 24 // //go:noescape func emitCopyNoRepeat(dst []byte, offset int, length int) int @@ -185,7 +187,8 @@ func emitCopyNoRepeat(dst []byte, offset int, length int) int // matchLen returns how many bytes match in a and b // // It assumes that: -// len(a) <= len(b) +// +// len(a) <= len(b) // //go:noescape func matchLen(a []byte, b []byte) int diff --git a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s index 36915d949..81a487d6d 100644 --- a/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s +++ b/vendor/github.com/klauspost/compress/s2/encodeblock_amd64.s @@ -1,7 +1,6 @@ // Code generated by command: go run gen.go -out ../encodeblock_amd64.s -stubs ../encodeblock_amd64.go -pkg=s2. DO NOT EDIT. //go:build !appengine && !noasm && gc && !noasm -// +build !appengine,!noasm,gc,!noasm #include "textflag.h" @@ -5743,9 +5742,9 @@ emit_literal_done_emit_remainder_encodeBlockAsm8B: // func encodeBetterBlockAsm(dst []byte, src []byte) int // Requires: BMI, SSE2 -TEXT ·encodeBetterBlockAsm(SB), $327704-56 +TEXT ·encodeBetterBlockAsm(SB), $589848-56 MOVQ dst_base+0(FP), AX - MOVQ $0x00000a00, CX + MOVQ $0x00001200, CX LEAQ 24(SP), DX PXOR X0, X0 @@ -5797,27 +5796,37 @@ check_maxskip_cont_encodeBetterBlockAsm: MOVQ DI, R11 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ SI, R11 SHRQ $0x32, R11 MOVL 24(SP)(R10*4), SI - MOVL 262168(SP)(R11*4), R8 + MOVL 524312(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) - MOVL CX, 262168(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVL CX, 524312(SP)(R11*4) + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeBetterBlockAsm - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeBetterBlockAsm - MOVL 20(SP), CX - JMP search_loop_encodeBetterBlockAsm + CMPQ R11, DI + JNE no_short_found_encodeBetterBlockAsm + MOVL R8, SI + JMP candidate_match_encodeBetterBlockAsm + +no_short_found_encodeBetterBlockAsm: + CMPL R10, DI + JEQ candidate_match_encodeBetterBlockAsm + CMPL R11, DI + JEQ candidateS_match_encodeBetterBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm candidateS_match_encodeBetterBlockAsm: SHRQ $0x08, DI MOVQ DI, R10 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 MOVL 24(SP)(R10*4), SI INCL CX MOVL CX, 24(SP)(R10*4) @@ -6590,52 +6599,49 @@ match_nolit_emitcopy_end_encodeBetterBlockAsm: match_nolit_dst_ok_encodeBetterBlockAsm: MOVQ $0x00cf1bbcdcbfa563, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x32, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x32, R12 + SHLQ $0x08, R12 + IMULQ SI, R12 + SHRQ $0x2f, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x32, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 262168(SP)(R11*4) - MOVL R15, 262168(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 524312(SP)(R11*4) + MOVL R14, 524312(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeBetterBlockAsm: + CMPQ DI, R9 + JAE search_loop_encodeBetterBlockAsm + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x08, R8 + IMULQ SI, R8 + SHRQ $0x2f, R8 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x32, R11 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 262168(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeBetterBlockAsm + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeBetterBlockAsm emit_remainder_encodeBetterBlockAsm: MOVQ src_len+32(FP), CX @@ -6815,9 +6821,9 @@ emit_literal_done_emit_remainder_encodeBetterBlockAsm: // func encodeBetterBlockAsm4MB(dst []byte, src []byte) int // Requires: BMI, SSE2 -TEXT ·encodeBetterBlockAsm4MB(SB), $327704-56 +TEXT ·encodeBetterBlockAsm4MB(SB), $589848-56 MOVQ dst_base+0(FP), AX - MOVQ $0x00000a00, CX + MOVQ $0x00001200, CX LEAQ 24(SP), DX PXOR X0, X0 @@ -6869,27 +6875,37 @@ check_maxskip_cont_encodeBetterBlockAsm4MB: MOVQ DI, R11 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ SI, R11 SHRQ $0x32, R11 MOVL 24(SP)(R10*4), SI - MOVL 262168(SP)(R11*4), R8 + MOVL 524312(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) - MOVL CX, 262168(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVL CX, 524312(SP)(R11*4) + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeBetterBlockAsm4MB - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeBetterBlockAsm4MB - MOVL 20(SP), CX - JMP search_loop_encodeBetterBlockAsm4MB + CMPQ R11, DI + JNE no_short_found_encodeBetterBlockAsm4MB + MOVL R8, SI + JMP candidate_match_encodeBetterBlockAsm4MB + +no_short_found_encodeBetterBlockAsm4MB: + CMPL R10, DI + JEQ candidate_match_encodeBetterBlockAsm4MB + CMPL R11, DI + JEQ candidateS_match_encodeBetterBlockAsm4MB + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm4MB candidateS_match_encodeBetterBlockAsm4MB: SHRQ $0x08, DI MOVQ DI, R10 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 MOVL 24(SP)(R10*4), SI INCL CX MOVL CX, 24(SP)(R10*4) @@ -7600,52 +7616,49 @@ match_nolit_emitcopy_end_encodeBetterBlockAsm4MB: match_nolit_dst_ok_encodeBetterBlockAsm4MB: MOVQ $0x00cf1bbcdcbfa563, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x32, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x32, R12 + SHLQ $0x08, R12 + IMULQ SI, R12 + SHRQ $0x2f, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x32, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 262168(SP)(R11*4) - MOVL R15, 262168(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 524312(SP)(R11*4) + MOVL R14, 524312(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeBetterBlockAsm4MB: + CMPQ DI, R9 + JAE search_loop_encodeBetterBlockAsm4MB + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x08, R8 + IMULQ SI, R8 + SHRQ $0x2f, R8 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x32, R11 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 262168(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeBetterBlockAsm4MB + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeBetterBlockAsm4MB emit_remainder_encodeBetterBlockAsm4MB: MOVQ src_len+32(FP), CX @@ -7871,12 +7884,22 @@ search_loop_encodeBetterBlockAsm12B: MOVL 65560(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 65560(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeBetterBlockAsm12B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeBetterBlockAsm12B - MOVL 20(SP), CX - JMP search_loop_encodeBetterBlockAsm12B + CMPQ R11, DI + JNE no_short_found_encodeBetterBlockAsm12B + MOVL R8, SI + JMP candidate_match_encodeBetterBlockAsm12B + +no_short_found_encodeBetterBlockAsm12B: + CMPL R10, DI + JEQ candidate_match_encodeBetterBlockAsm12B + CMPL R11, DI + JEQ candidateS_match_encodeBetterBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm12B candidateS_match_encodeBetterBlockAsm12B: SHRQ $0x08, DI @@ -8447,52 +8470,49 @@ match_nolit_emitcopy_end_encodeBetterBlockAsm12B: match_nolit_dst_ok_encodeBetterBlockAsm12B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x32, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x32, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x34, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x34, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x32, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x34, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 65560(SP)(R11*4) - MOVL R15, 65560(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 65560(SP)(R11*4) + MOVL R14, 65560(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeBetterBlockAsm12B: + CMPQ DI, R9 + JAE search_loop_encodeBetterBlockAsm12B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x32, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x32, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x34, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x32, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 65560(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeBetterBlockAsm12B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeBetterBlockAsm12B emit_remainder_encodeBetterBlockAsm12B: MOVQ src_len+32(FP), CX @@ -8707,12 +8727,22 @@ search_loop_encodeBetterBlockAsm10B: MOVL 16408(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 16408(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeBetterBlockAsm10B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeBetterBlockAsm10B - MOVL 20(SP), CX - JMP search_loop_encodeBetterBlockAsm10B + CMPQ R11, DI + JNE no_short_found_encodeBetterBlockAsm10B + MOVL R8, SI + JMP candidate_match_encodeBetterBlockAsm10B + +no_short_found_encodeBetterBlockAsm10B: + CMPL R10, DI + JEQ candidate_match_encodeBetterBlockAsm10B + CMPL R11, DI + JEQ candidateS_match_encodeBetterBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm10B candidateS_match_encodeBetterBlockAsm10B: SHRQ $0x08, DI @@ -9283,52 +9313,49 @@ match_nolit_emitcopy_end_encodeBetterBlockAsm10B: match_nolit_dst_ok_encodeBetterBlockAsm10B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x34, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x34, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x36, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x36, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x34, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x36, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 16408(SP)(R11*4) - MOVL R15, 16408(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 16408(SP)(R11*4) + MOVL R14, 16408(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeBetterBlockAsm10B: + CMPQ DI, R9 + JAE search_loop_encodeBetterBlockAsm10B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x34, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x34, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x36, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x34, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 16408(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeBetterBlockAsm10B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeBetterBlockAsm10B emit_remainder_encodeBetterBlockAsm10B: MOVQ src_len+32(FP), CX @@ -9543,12 +9570,22 @@ search_loop_encodeBetterBlockAsm8B: MOVL 4120(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 4120(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeBetterBlockAsm8B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeBetterBlockAsm8B - MOVL 20(SP), CX - JMP search_loop_encodeBetterBlockAsm8B + CMPQ R11, DI + JNE no_short_found_encodeBetterBlockAsm8B + MOVL R8, SI + JMP candidate_match_encodeBetterBlockAsm8B + +no_short_found_encodeBetterBlockAsm8B: + CMPL R10, DI + JEQ candidate_match_encodeBetterBlockAsm8B + CMPL R11, DI + JEQ candidateS_match_encodeBetterBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeBetterBlockAsm8B candidateS_match_encodeBetterBlockAsm8B: SHRQ $0x08, DI @@ -10105,52 +10142,49 @@ match_nolit_emitcopy_end_encodeBetterBlockAsm8B: match_nolit_dst_ok_encodeBetterBlockAsm8B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x36, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x36, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x38, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x38, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x36, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x38, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 4120(SP)(R11*4) - MOVL R15, 4120(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 4120(SP)(R11*4) + MOVL R14, 4120(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeBetterBlockAsm8B: + CMPQ DI, R9 + JAE search_loop_encodeBetterBlockAsm8B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x36, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x36, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x38, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x36, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 4120(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeBetterBlockAsm8B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeBetterBlockAsm8B emit_remainder_encodeBetterBlockAsm8B: MOVQ src_len+32(FP), CX @@ -14287,9 +14321,9 @@ emit_literal_done_emit_remainder_encodeSnappyBlockAsm8B: // func encodeSnappyBetterBlockAsm(dst []byte, src []byte) int // Requires: BMI, SSE2 -TEXT ·encodeSnappyBetterBlockAsm(SB), $327704-56 +TEXT ·encodeSnappyBetterBlockAsm(SB), $589848-56 MOVQ dst_base+0(FP), AX - MOVQ $0x00000a00, CX + MOVQ $0x00001200, CX LEAQ 24(SP), DX PXOR X0, X0 @@ -14341,27 +14375,37 @@ check_maxskip_cont_encodeSnappyBetterBlockAsm: MOVQ DI, R11 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ SI, R11 SHRQ $0x32, R11 MOVL 24(SP)(R10*4), SI - MOVL 262168(SP)(R11*4), R8 + MOVL 524312(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) - MOVL CX, 262168(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVL CX, 524312(SP)(R11*4) + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeSnappyBetterBlockAsm - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeSnappyBetterBlockAsm - MOVL 20(SP), CX - JMP search_loop_encodeSnappyBetterBlockAsm + CMPQ R11, DI + JNE no_short_found_encodeSnappyBetterBlockAsm + MOVL R8, SI + JMP candidate_match_encodeSnappyBetterBlockAsm + +no_short_found_encodeSnappyBetterBlockAsm: + CMPL R10, DI + JEQ candidate_match_encodeSnappyBetterBlockAsm + CMPL R11, DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm candidateS_match_encodeSnappyBetterBlockAsm: SHRQ $0x08, DI MOVQ DI, R10 SHLQ $0x08, R10 IMULQ R9, R10 - SHRQ $0x30, R10 + SHRQ $0x2f, R10 MOVL 24(SP)(R10*4), SI INCL CX MOVL CX, 24(SP)(R10*4) @@ -14685,52 +14729,49 @@ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm: match_nolit_dst_ok_encodeSnappyBetterBlockAsm: MOVQ $0x00cf1bbcdcbfa563, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x32, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x32, R12 + SHLQ $0x08, R12 + IMULQ SI, R12 + SHRQ $0x2f, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x32, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 262168(SP)(R11*4) - MOVL R15, 262168(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 524312(SP)(R11*4) + MOVL R14, 524312(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeSnappyBetterBlockAsm: + CMPQ DI, R9 + JAE search_loop_encodeSnappyBetterBlockAsm + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x08, R8 + IMULQ SI, R8 + SHRQ $0x2f, R8 SHLQ $0x08, R10 IMULQ SI, R10 - SHRQ $0x30, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x32, R11 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + SHRQ $0x2f, R10 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 262168(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeSnappyBetterBlockAsm + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeSnappyBetterBlockAsm emit_remainder_encodeSnappyBetterBlockAsm: MOVQ src_len+32(FP), CX @@ -14964,12 +15005,22 @@ search_loop_encodeSnappyBetterBlockAsm64K: MOVL 262168(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 262168(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeSnappyBetterBlockAsm64K - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeSnappyBetterBlockAsm64K - MOVL 20(SP), CX - JMP search_loop_encodeSnappyBetterBlockAsm64K + CMPQ R11, DI + JNE no_short_found_encodeSnappyBetterBlockAsm64K + MOVL R8, SI + JMP candidate_match_encodeSnappyBetterBlockAsm64K + +no_short_found_encodeSnappyBetterBlockAsm64K: + CMPL R10, DI + JEQ candidate_match_encodeSnappyBetterBlockAsm64K + CMPL R11, DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm64K + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm64K candidateS_match_encodeSnappyBetterBlockAsm64K: SHRQ $0x08, DI @@ -15248,52 +15299,49 @@ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm64K: match_nolit_dst_ok_encodeSnappyBetterBlockAsm64K: MOVQ $0x00cf1bbcdcbfa563, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x08, R10 IMULQ SI, R10 SHRQ $0x30, R10 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x32, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x32, R12 + SHLQ $0x08, R12 + IMULQ SI, R12 + SHRQ $0x30, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x32, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 262168(SP)(R11*4) - MOVL R15, 262168(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 262168(SP)(R11*4) + MOVL R14, 262168(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeSnappyBetterBlockAsm64K: + CMPQ DI, R9 + JAE search_loop_encodeSnappyBetterBlockAsm64K + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x08, R8 + IMULQ SI, R8 + SHRQ $0x30, R8 SHLQ $0x08, R10 IMULQ SI, R10 SHRQ $0x30, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x32, R11 - SHLQ $0x08, R13 - IMULQ SI, R13 - SHRQ $0x30, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 262168(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeSnappyBetterBlockAsm64K + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeSnappyBetterBlockAsm64K emit_remainder_encodeSnappyBetterBlockAsm64K: MOVQ src_len+32(FP), CX @@ -15508,12 +15556,22 @@ search_loop_encodeSnappyBetterBlockAsm12B: MOVL 65560(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 65560(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeSnappyBetterBlockAsm12B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeSnappyBetterBlockAsm12B - MOVL 20(SP), CX - JMP search_loop_encodeSnappyBetterBlockAsm12B + CMPQ R11, DI + JNE no_short_found_encodeSnappyBetterBlockAsm12B + MOVL R8, SI + JMP candidate_match_encodeSnappyBetterBlockAsm12B + +no_short_found_encodeSnappyBetterBlockAsm12B: + CMPL R10, DI + JEQ candidate_match_encodeSnappyBetterBlockAsm12B + CMPL R11, DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm12B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm12B candidateS_match_encodeSnappyBetterBlockAsm12B: SHRQ $0x08, DI @@ -15792,52 +15850,49 @@ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm12B: match_nolit_dst_ok_encodeSnappyBetterBlockAsm12B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x32, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x32, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x34, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x34, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x32, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x34, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 65560(SP)(R11*4) - MOVL R15, 65560(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 65560(SP)(R11*4) + MOVL R14, 65560(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeSnappyBetterBlockAsm12B: + CMPQ DI, R9 + JAE search_loop_encodeSnappyBetterBlockAsm12B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x32, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x32, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x34, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x32, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 65560(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeSnappyBetterBlockAsm12B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeSnappyBetterBlockAsm12B emit_remainder_encodeSnappyBetterBlockAsm12B: MOVQ src_len+32(FP), CX @@ -16052,12 +16107,22 @@ search_loop_encodeSnappyBetterBlockAsm10B: MOVL 16408(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 16408(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeSnappyBetterBlockAsm10B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeSnappyBetterBlockAsm10B - MOVL 20(SP), CX - JMP search_loop_encodeSnappyBetterBlockAsm10B + CMPQ R11, DI + JNE no_short_found_encodeSnappyBetterBlockAsm10B + MOVL R8, SI + JMP candidate_match_encodeSnappyBetterBlockAsm10B + +no_short_found_encodeSnappyBetterBlockAsm10B: + CMPL R10, DI + JEQ candidate_match_encodeSnappyBetterBlockAsm10B + CMPL R11, DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm10B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm10B candidateS_match_encodeSnappyBetterBlockAsm10B: SHRQ $0x08, DI @@ -16336,52 +16401,49 @@ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm10B: match_nolit_dst_ok_encodeSnappyBetterBlockAsm10B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x34, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x34, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x36, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x36, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x34, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x36, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 16408(SP)(R11*4) - MOVL R15, 16408(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 16408(SP)(R11*4) + MOVL R14, 16408(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeSnappyBetterBlockAsm10B: + CMPQ DI, R9 + JAE search_loop_encodeSnappyBetterBlockAsm10B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x34, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x34, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x36, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x34, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 16408(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeSnappyBetterBlockAsm10B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeSnappyBetterBlockAsm10B emit_remainder_encodeSnappyBetterBlockAsm10B: MOVQ src_len+32(FP), CX @@ -16596,12 +16658,22 @@ search_loop_encodeSnappyBetterBlockAsm8B: MOVL 4120(SP)(R11*4), R8 MOVL CX, 24(SP)(R10*4) MOVL CX, 4120(SP)(R11*4) - CMPL (DX)(SI*1), DI + MOVQ (DX)(SI*1), R10 + MOVQ (DX)(R8*1), R11 + CMPQ R10, DI JEQ candidate_match_encodeSnappyBetterBlockAsm8B - CMPL (DX)(R8*1), DI - JEQ candidateS_match_encodeSnappyBetterBlockAsm8B - MOVL 20(SP), CX - JMP search_loop_encodeSnappyBetterBlockAsm8B + CMPQ R11, DI + JNE no_short_found_encodeSnappyBetterBlockAsm8B + MOVL R8, SI + JMP candidate_match_encodeSnappyBetterBlockAsm8B + +no_short_found_encodeSnappyBetterBlockAsm8B: + CMPL R10, DI + JEQ candidate_match_encodeSnappyBetterBlockAsm8B + CMPL R11, DI + JEQ candidateS_match_encodeSnappyBetterBlockAsm8B + MOVL 20(SP), CX + JMP search_loop_encodeSnappyBetterBlockAsm8B candidateS_match_encodeSnappyBetterBlockAsm8B: SHRQ $0x08, DI @@ -16878,52 +16950,49 @@ match_nolit_emitcopy_end_encodeSnappyBetterBlockAsm8B: match_nolit_dst_ok_encodeSnappyBetterBlockAsm8B: MOVQ $0x0000cf1bbcdcbf9b, SI MOVQ $0x9e3779b1, R8 - INCL DI - MOVQ (DX)(DI*1), R9 - MOVQ R9, R10 - MOVQ R9, R11 - MOVQ R9, R12 - SHRQ $0x08, R11 - MOVQ R11, R13 - SHRQ $0x10, R12 - LEAL 1(DI), R14 - LEAL 2(DI), R15 - MOVQ -2(DX)(CX*1), R9 + LEAQ 1(DI), DI + LEAQ -2(CX), R9 + MOVQ (DX)(DI*1), R10 + MOVQ 1(DX)(DI*1), R11 + MOVQ (DX)(R9*1), R12 + MOVQ 1(DX)(R9*1), R13 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x36, R10 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x36, R13 SHLQ $0x20, R11 IMULQ R8, R11 SHRQ $0x38, R11 - SHLQ $0x20, R12 - IMULQ R8, R12 - SHRQ $0x38, R12 + SHLQ $0x10, R12 + IMULQ SI, R12 + SHRQ $0x36, R12 + SHLQ $0x20, R13 + IMULQ R8, R13 + SHRQ $0x38, R13 + LEAQ 1(DI), R8 + LEAQ 1(R9), R14 MOVL DI, 24(SP)(R10*4) - MOVL R14, 24(SP)(R13*4) - MOVL R14, 4120(SP)(R11*4) - MOVL R15, 4120(SP)(R12*4) - MOVQ R9, R10 - MOVQ R9, R11 - SHRQ $0x08, R11 - MOVQ R11, R13 - LEAL -2(CX), R9 - LEAL -1(CX), DI + MOVL R9, 24(SP)(R12*4) + MOVL R8, 4120(SP)(R11*4) + MOVL R14, 4120(SP)(R13*4) + ADDQ $0x01, DI + SUBQ $0x01, R9 + +index_loop_encodeSnappyBetterBlockAsm8B: + CMPQ DI, R9 + JAE search_loop_encodeSnappyBetterBlockAsm8B + MOVQ (DX)(DI*1), R8 + MOVQ (DX)(R9*1), R10 + SHLQ $0x10, R8 + IMULQ SI, R8 + SHRQ $0x36, R8 SHLQ $0x10, R10 IMULQ SI, R10 SHRQ $0x36, R10 - SHLQ $0x20, R11 - IMULQ R8, R11 - SHRQ $0x38, R11 - SHLQ $0x10, R13 - IMULQ SI, R13 - SHRQ $0x36, R13 + MOVL DI, 24(SP)(R8*4) MOVL R9, 24(SP)(R10*4) - MOVL DI, 4120(SP)(R11*4) - MOVL DI, 24(SP)(R13*4) - JMP search_loop_encodeSnappyBetterBlockAsm8B + ADDQ $0x02, DI + SUBQ $0x02, R9 + JMP index_loop_encodeSnappyBetterBlockAsm8B emit_remainder_encodeSnappyBetterBlockAsm8B: MOVQ src_len+32(FP), CX diff --git a/vendor/github.com/klauspost/compress/zstd/README.md b/vendor/github.com/klauspost/compress/zstd/README.md index beb7fa872..65b38abed 100644 --- a/vendor/github.com/klauspost/compress/zstd/README.md +++ b/vendor/github.com/klauspost/compress/zstd/README.md @@ -12,6 +12,8 @@ The `zstd` package is provided as open source software using a Go standard licen Currently the package is heavily optimized for 64 bit processors and will be significantly slower on 32 bit processors. +For seekable zstd streams, see [this excellent package](https://github.com/SaveTheRbtz/zstd-seekable-format-go). + ## Installation Install using `go get -u github.com/klauspost/compress`. The package is located in `github.com/klauspost/compress/zstd`. diff --git a/vendor/github.com/klauspost/compress/zstd/blockdec.go b/vendor/github.com/klauspost/compress/zstd/blockdec.go index 7eed729be..f52d1aed6 100644 --- a/vendor/github.com/klauspost/compress/zstd/blockdec.go +++ b/vendor/github.com/klauspost/compress/zstd/blockdec.go @@ -10,7 +10,6 @@ import ( "errors" "fmt" "io" - "io/ioutil" "os" "path/filepath" "sync" @@ -651,7 +650,7 @@ func (b *blockDec) prepareSequences(in []byte, hist *history) (err error) { fatalErr(binary.Write(&buf, binary.LittleEndian, hist.decoders.matchLengths.fse)) fatalErr(binary.Write(&buf, binary.LittleEndian, hist.decoders.offsets.fse)) buf.Write(in) - ioutil.WriteFile(filepath.Join("testdata", "seqs", fn), buf.Bytes(), os.ModePerm) + os.WriteFile(filepath.Join("testdata", "seqs", fn), buf.Bytes(), os.ModePerm) } return nil diff --git a/vendor/github.com/klauspost/compress/zstd/bytebuf.go b/vendor/github.com/klauspost/compress/zstd/bytebuf.go index 2ad02070d..176788f25 100644 --- a/vendor/github.com/klauspost/compress/zstd/bytebuf.go +++ b/vendor/github.com/klauspost/compress/zstd/bytebuf.go @@ -7,7 +7,6 @@ package zstd import ( "fmt" "io" - "io/ioutil" ) type byteBuffer interface { @@ -124,7 +123,7 @@ func (r *readerWrapper) readByte() (byte, error) { } func (r *readerWrapper) skipN(n int64) error { - n2, err := io.CopyN(ioutil.Discard, r.r, n) + n2, err := io.CopyN(io.Discard, r.r, n) if n2 != n { err = io.ErrUnexpectedEOF } diff --git a/vendor/github.com/klauspost/compress/zstd/decoder.go b/vendor/github.com/klauspost/compress/zstd/decoder.go index d212f4737..6104eb793 100644 --- a/vendor/github.com/klauspost/compress/zstd/decoder.go +++ b/vendor/github.com/klauspost/compress/zstd/decoder.go @@ -312,6 +312,7 @@ func (d *Decoder) DecodeAll(input, dst []byte) ([]byte, error) { // Grab a block decoder and frame decoder. block := <-d.decoders frame := block.localFrame + initialSize := len(dst) defer func() { if debugDecoder { printf("re-adding decoder: %p", block) @@ -354,7 +355,16 @@ func (d *Decoder) DecodeAll(input, dst []byte) ([]byte, error) { return dst, ErrWindowSizeExceeded } if frame.FrameContentSize != fcsUnknown { - if frame.FrameContentSize > d.o.maxDecodedSize-uint64(len(dst)) { + if frame.FrameContentSize > d.o.maxDecodedSize-uint64(len(dst)-initialSize) { + if debugDecoder { + println("decoder size exceeded; fcs:", frame.FrameContentSize, "> mcs:", d.o.maxDecodedSize-uint64(len(dst)-initialSize), "len:", len(dst)) + } + return dst, ErrDecoderSizeExceeded + } + if d.o.limitToCap && frame.FrameContentSize > uint64(cap(dst)-len(dst)) { + if debugDecoder { + println("decoder size exceeded; fcs:", frame.FrameContentSize, "> (cap-len)", cap(dst)-len(dst)) + } return dst, ErrDecoderSizeExceeded } if cap(dst)-len(dst) < int(frame.FrameContentSize) { @@ -364,7 +374,7 @@ func (d *Decoder) DecodeAll(input, dst []byte) ([]byte, error) { } } - if cap(dst) == 0 { + if cap(dst) == 0 && !d.o.limitToCap { // Allocate len(input) * 2 by default if nothing is provided // and we didn't get frame content size. size := len(input) * 2 @@ -382,6 +392,9 @@ func (d *Decoder) DecodeAll(input, dst []byte) ([]byte, error) { if err != nil { return dst, err } + if uint64(len(dst)-initialSize) > d.o.maxDecodedSize { + return dst, ErrDecoderSizeExceeded + } if len(frame.bBuf) == 0 { if debugDecoder { println("frame dbuf empty") @@ -852,6 +865,10 @@ decodeStream: } } if err == nil && d.frame.WindowSize > d.o.maxWindowSize { + if debugDecoder { + println("decoder size exceeded, fws:", d.frame.WindowSize, "> mws:", d.o.maxWindowSize) + } + err = ErrDecoderSizeExceeded } if err != nil { diff --git a/vendor/github.com/klauspost/compress/zstd/decoder_options.go b/vendor/github.com/klauspost/compress/zstd/decoder_options.go index c70e6fa0f..666c2715f 100644 --- a/vendor/github.com/klauspost/compress/zstd/decoder_options.go +++ b/vendor/github.com/klauspost/compress/zstd/decoder_options.go @@ -20,6 +20,7 @@ type decoderOptions struct { maxWindowSize uint64 dicts []dict ignoreChecksum bool + limitToCap bool } func (o *decoderOptions) setDefault() { @@ -114,6 +115,17 @@ func WithDecoderMaxWindow(size uint64) DOption { } } +// WithDecodeAllCapLimit will limit DecodeAll to decoding cap(dst)-len(dst) bytes, +// or any size set in WithDecoderMaxMemory. +// This can be used to limit decoding to a specific maximum output size. +// Disabled by default. +func WithDecodeAllCapLimit(b bool) DOption { + return func(o *decoderOptions) error { + o.limitToCap = b + return nil + } +} + // IgnoreChecksum allows to forcibly ignore checksum checking. func IgnoreChecksum(b bool) DOption { return func(o *decoderOptions) error { diff --git a/vendor/github.com/klauspost/compress/zstd/enc_better.go b/vendor/github.com/klauspost/compress/zstd/enc_better.go index c769f6941..d70e3fd3d 100644 --- a/vendor/github.com/klauspost/compress/zstd/enc_better.go +++ b/vendor/github.com/klauspost/compress/zstd/enc_better.go @@ -416,15 +416,23 @@ encodeLoop: // Try to find a better match by searching for a long match at the end of the current best match if s+matched < sLimit { + // Allow some bytes at the beginning to mismatch. + // Sweet spot is around 3 bytes, but depends on input. + // The skipped bytes are tested in Extend backwards, + // and still picked up as part of the match if they do. + const skipBeginning = 3 + nextHashL := hashLen(load6432(src, s+matched), betterLongTableBits, betterLongLen) - cv := load3232(src, s) + s2 := s + skipBeginning + cv := load3232(src, s2) candidateL := e.longTable[nextHashL] - coffsetL := candidateL.offset - e.cur - matched - if coffsetL >= 0 && coffsetL < s && s-coffsetL < e.maxMatchOff && cv == load3232(src, coffsetL) { + coffsetL := candidateL.offset - e.cur - matched + skipBeginning + if coffsetL >= 0 && coffsetL < s2 && s2-coffsetL < e.maxMatchOff && cv == load3232(src, coffsetL) { // Found a long match, at least 4 bytes. - matchedNext := e.matchlen(s+4, coffsetL+4, src) + 4 + matchedNext := e.matchlen(s2+4, coffsetL+4, src) + 4 if matchedNext > matched { t = coffsetL + s = s2 matched = matchedNext if debugMatches { println("long match at end-of-match") @@ -434,12 +442,13 @@ encodeLoop: // Check prev long... if true { - coffsetL = candidateL.prev - e.cur - matched - if coffsetL >= 0 && coffsetL < s && s-coffsetL < e.maxMatchOff && cv == load3232(src, coffsetL) { + coffsetL = candidateL.prev - e.cur - matched + skipBeginning + if coffsetL >= 0 && coffsetL < s2 && s2-coffsetL < e.maxMatchOff && cv == load3232(src, coffsetL) { // Found a long match, at least 4 bytes. - matchedNext := e.matchlen(s+4, coffsetL+4, src) + 4 + matchedNext := e.matchlen(s2+4, coffsetL+4, src) + 4 if matchedNext > matched { t = coffsetL + s = s2 matched = matchedNext if debugMatches { println("prev long match at end-of-match") diff --git a/vendor/github.com/klauspost/compress/zstd/enc_dfast.go b/vendor/github.com/klauspost/compress/zstd/enc_dfast.go index 7ff0c64fa..1f4a9a245 100644 --- a/vendor/github.com/klauspost/compress/zstd/enc_dfast.go +++ b/vendor/github.com/klauspost/compress/zstd/enc_dfast.go @@ -1103,7 +1103,8 @@ func (e *doubleFastEncoderDict) Reset(d *dict, singleBlock bool) { } if allDirty || dirtyShardCnt > dLongTableShardCnt/2 { - copy(e.longTable[:], e.dictLongTable) + //copy(e.longTable[:], e.dictLongTable) + e.longTable = *(*[dFastLongTableSize]tableEntry)(e.dictLongTable) for i := range e.longTableShardDirty { e.longTableShardDirty[i] = false } @@ -1114,7 +1115,9 @@ func (e *doubleFastEncoderDict) Reset(d *dict, singleBlock bool) { continue } - copy(e.longTable[i*dLongTableShardSize:(i+1)*dLongTableShardSize], e.dictLongTable[i*dLongTableShardSize:(i+1)*dLongTableShardSize]) + // copy(e.longTable[i*dLongTableShardSize:(i+1)*dLongTableShardSize], e.dictLongTable[i*dLongTableShardSize:(i+1)*dLongTableShardSize]) + *(*[dLongTableShardSize]tableEntry)(e.longTable[i*dLongTableShardSize:]) = *(*[dLongTableShardSize]tableEntry)(e.dictLongTable[i*dLongTableShardSize:]) + e.longTableShardDirty[i] = false } } diff --git a/vendor/github.com/klauspost/compress/zstd/enc_fast.go b/vendor/github.com/klauspost/compress/zstd/enc_fast.go index f51ab529a..181edc02b 100644 --- a/vendor/github.com/klauspost/compress/zstd/enc_fast.go +++ b/vendor/github.com/klauspost/compress/zstd/enc_fast.go @@ -871,7 +871,8 @@ func (e *fastEncoderDict) Reset(d *dict, singleBlock bool) { const shardCnt = tableShardCnt const shardSize = tableShardSize if e.allDirty || dirtyShardCnt > shardCnt*4/6 { - copy(e.table[:], e.dictTable) + //copy(e.table[:], e.dictTable) + e.table = *(*[tableSize]tableEntry)(e.dictTable) for i := range e.tableShardDirty { e.tableShardDirty[i] = false } @@ -883,7 +884,8 @@ func (e *fastEncoderDict) Reset(d *dict, singleBlock bool) { continue } - copy(e.table[i*shardSize:(i+1)*shardSize], e.dictTable[i*shardSize:(i+1)*shardSize]) + //copy(e.table[i*shardSize:(i+1)*shardSize], e.dictTable[i*shardSize:(i+1)*shardSize]) + *(*[shardSize]tableEntry)(e.table[i*shardSize:]) = *(*[shardSize]tableEntry)(e.dictTable[i*shardSize:]) e.tableShardDirty[i] = false } e.allDirty = false diff --git a/vendor/github.com/klauspost/compress/zstd/framedec.go b/vendor/github.com/klauspost/compress/zstd/framedec.go index 9568a4ba3..1559a2038 100644 --- a/vendor/github.com/klauspost/compress/zstd/framedec.go +++ b/vendor/github.com/klauspost/compress/zstd/framedec.go @@ -353,12 +353,23 @@ func (d *frameDec) runDecoder(dst []byte, dec *blockDec) ([]byte, error) { // Store input length, so we only check new data. crcStart := len(dst) d.history.decoders.maxSyncLen = 0 + if d.o.limitToCap { + d.history.decoders.maxSyncLen = uint64(cap(dst) - len(dst)) + } if d.FrameContentSize != fcsUnknown { - d.history.decoders.maxSyncLen = d.FrameContentSize + uint64(len(dst)) + if !d.o.limitToCap || d.FrameContentSize+uint64(len(dst)) < d.history.decoders.maxSyncLen { + d.history.decoders.maxSyncLen = d.FrameContentSize + uint64(len(dst)) + } if d.history.decoders.maxSyncLen > d.o.maxDecodedSize { + if debugDecoder { + println("maxSyncLen:", d.history.decoders.maxSyncLen, "> maxDecodedSize:", d.o.maxDecodedSize) + } return dst, ErrDecoderSizeExceeded } - if uint64(cap(dst)) < d.history.decoders.maxSyncLen { + if debugDecoder { + println("maxSyncLen:", d.history.decoders.maxSyncLen) + } + if !d.o.limitToCap && uint64(cap(dst)-len(dst)) < d.history.decoders.maxSyncLen { // Alloc for output dst2 := make([]byte, len(dst), d.history.decoders.maxSyncLen+compressedBlockOverAlloc) copy(dst2, dst) @@ -378,7 +389,13 @@ func (d *frameDec) runDecoder(dst []byte, dec *blockDec) ([]byte, error) { if err != nil { break } - if uint64(len(d.history.b)) > d.o.maxDecodedSize { + if uint64(len(d.history.b)-crcStart) > d.o.maxDecodedSize { + println("runDecoder: maxDecodedSize exceeded", uint64(len(d.history.b)-crcStart), ">", d.o.maxDecodedSize) + err = ErrDecoderSizeExceeded + break + } + if d.o.limitToCap && len(d.history.b) > cap(dst) { + println("runDecoder: cap exceeded", uint64(len(d.history.b)), ">", cap(dst)) err = ErrDecoderSizeExceeded break } diff --git a/vendor/github.com/klauspost/compress/zstd/fse_decoder_amd64.s b/vendor/github.com/klauspost/compress/zstd/fse_decoder_amd64.s index da32b4420..bcde39869 100644 --- a/vendor/github.com/klauspost/compress/zstd/fse_decoder_amd64.s +++ b/vendor/github.com/klauspost/compress/zstd/fse_decoder_amd64.s @@ -1,7 +1,6 @@ // Code generated by command: go run gen_fse.go -out ../fse_decoder_amd64.s -pkg=zstd. DO NOT EDIT. //go:build !appengine && !noasm && gc && !noasm -// +build !appengine,!noasm,gc,!noasm // func buildDtable_asm(s *fseDecoder, ctx *buildDtableAsmContext) int TEXT ·buildDtable_asm(SB), $0-24 diff --git a/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.go b/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.go index 7598c1018..1c704d30c 100644 --- a/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.go +++ b/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.go @@ -32,18 +32,22 @@ type decodeSyncAsmContext struct { // sequenceDecs_decodeSync_amd64 implements the main loop of sequenceDecs.decodeSync in x86 asm. // // Please refer to seqdec_generic.go for the reference implementation. +// //go:noescape func sequenceDecs_decodeSync_amd64(s *sequenceDecs, br *bitReader, ctx *decodeSyncAsmContext) int // sequenceDecs_decodeSync_bmi2 implements the main loop of sequenceDecs.decodeSync in x86 asm with BMI2 extensions. +// //go:noescape func sequenceDecs_decodeSync_bmi2(s *sequenceDecs, br *bitReader, ctx *decodeSyncAsmContext) int // sequenceDecs_decodeSync_safe_amd64 does the same as above, but does not write more than output buffer. +// //go:noescape func sequenceDecs_decodeSync_safe_amd64(s *sequenceDecs, br *bitReader, ctx *decodeSyncAsmContext) int // sequenceDecs_decodeSync_safe_bmi2 does the same as above, but does not write more than output buffer. +// //go:noescape func sequenceDecs_decodeSync_safe_bmi2(s *sequenceDecs, br *bitReader, ctx *decodeSyncAsmContext) int @@ -201,20 +205,24 @@ const errorNotEnoughSpace = 5 // sequenceDecs_decode implements the main loop of sequenceDecs in x86 asm. // // Please refer to seqdec_generic.go for the reference implementation. +// //go:noescape func sequenceDecs_decode_amd64(s *sequenceDecs, br *bitReader, ctx *decodeAsmContext) int // sequenceDecs_decode implements the main loop of sequenceDecs in x86 asm. // // Please refer to seqdec_generic.go for the reference implementation. +// //go:noescape func sequenceDecs_decode_56_amd64(s *sequenceDecs, br *bitReader, ctx *decodeAsmContext) int // sequenceDecs_decode implements the main loop of sequenceDecs in x86 asm with BMI2 extensions. +// //go:noescape func sequenceDecs_decode_bmi2(s *sequenceDecs, br *bitReader, ctx *decodeAsmContext) int // sequenceDecs_decode implements the main loop of sequenceDecs in x86 asm with BMI2 extensions. +// //go:noescape func sequenceDecs_decode_56_bmi2(s *sequenceDecs, br *bitReader, ctx *decodeAsmContext) int @@ -308,10 +316,12 @@ type executeAsmContext struct { // Returns false if a match offset is too big. // // Please refer to seqdec_generic.go for the reference implementation. +// //go:noescape func sequenceDecs_executeSimple_amd64(ctx *executeAsmContext) bool // Same as above, but with safe memcopies +// //go:noescape func sequenceDecs_executeSimple_safe_amd64(ctx *executeAsmContext) bool diff --git a/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.s b/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.s index 27e76774c..52e5703c2 100644 --- a/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.s +++ b/vendor/github.com/klauspost/compress/zstd/seqdec_amd64.s @@ -1,7 +1,6 @@ // Code generated by command: go run gen.go -out ../seqdec_amd64.s -pkg=zstd. DO NOT EDIT. //go:build !appengine && !noasm && gc && !noasm -// +build !appengine,!noasm,gc,!noasm // func sequenceDecs_decode_amd64(s *sequenceDecs, br *bitReader, ctx *decodeAsmContext) int // Requires: CMOV diff --git a/vendor/golang.org/x/sys/unix/mkall.sh b/vendor/golang.org/x/sys/unix/mkall.sh index 3b2335d5f..1b2b424a7 100644 --- a/vendor/golang.org/x/sys/unix/mkall.sh +++ b/vendor/golang.org/x/sys/unix/mkall.sh @@ -214,11 +214,6 @@ esac if [ "$GOOSARCH" == "aix_ppc64" ]; then # aix/ppc64 script generates files instead of writing to stdin. echo "$mksyscall -tags $GOOS,$GOARCH $syscall_goos $GOOSARCH_in && gofmt -w zsyscall_$GOOSARCH.go && gofmt -w zsyscall_"$GOOSARCH"_gccgo.go && gofmt -w zsyscall_"$GOOSARCH"_gc.go " ; - elif [ "$GOOS" == "darwin" ]; then - # 1.12 and later, syscalls via libSystem - echo "$mksyscall -tags $GOOS,$GOARCH,go1.12 $syscall_goos $GOOSARCH_in |gofmt >zsyscall_$GOOSARCH.go"; - # 1.13 and later, syscalls via libSystem (including syscallPtr) - echo "$mksyscall -tags $GOOS,$GOARCH,go1.13 syscall_darwin.1_13.go |gofmt >zsyscall_$GOOSARCH.1_13.go"; elif [ "$GOOS" == "illumos" ]; then # illumos code generation requires a --illumos switch echo "$mksyscall -illumos -tags illumos,$GOARCH syscall_illumos.go |gofmt > zsyscall_illumos_$GOARCH.go"; diff --git a/vendor/golang.org/x/sys/unix/syscall.go b/vendor/golang.org/x/sys/unix/syscall.go index 9916e5e85..63e8c8383 100644 --- a/vendor/golang.org/x/sys/unix/syscall.go +++ b/vendor/golang.org/x/sys/unix/syscall.go @@ -80,8 +80,7 @@ func BytePtrToString(p *byte) string { ptr = unsafe.Pointer(uintptr(ptr) + 1) } - s := unsafe.Slice((*byte)(unsafe.Pointer(p)), n) - return string(s) + return string(unsafe.Slice(p, n)) } // Single-word zero for use when we need a valid pointer to 0 bytes. diff --git a/vendor/golang.org/x/sys/unix/syscall_darwin.1_12.go b/vendor/golang.org/x/sys/unix/syscall_darwin.1_12.go deleted file mode 100644 index b0098607c..000000000 --- a/vendor/golang.org/x/sys/unix/syscall_darwin.1_12.go +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright 2019 The Go Authors. All rights reserved. -// Use of this source code is governed by a BSD-style -// license that can be found in the LICENSE file. - -//go:build darwin && go1.12 && !go1.13 -// +build darwin,go1.12,!go1.13 - -package unix - -import ( - "unsafe" -) - -const _SYS_GETDIRENTRIES64 = 344 - -func Getdirentries(fd int, buf []byte, basep *uintptr) (n int, err error) { - // To implement this using libSystem we'd need syscall_syscallPtr for - // fdopendir. However, syscallPtr was only added in Go 1.13, so we fall - // back to raw syscalls for this func on Go 1.12. - var p unsafe.Pointer - if len(buf) > 0 { - p = unsafe.Pointer(&buf[0]) - } else { - p = unsafe.Pointer(&_zero) - } - r0, _, e1 := Syscall6(_SYS_GETDIRENTRIES64, uintptr(fd), uintptr(p), uintptr(len(buf)), uintptr(unsafe.Pointer(basep)), 0, 0) - n = int(r0) - if e1 != 0 { - return n, errnoErr(e1) - } - return n, nil -} diff --git a/vendor/golang.org/x/sys/unix/syscall_darwin.1_13.go b/vendor/golang.org/x/sys/unix/syscall_darwin.1_13.go deleted file mode 100644 index 1259f6df5..000000000 --- a/vendor/golang.org/x/sys/unix/syscall_darwin.1_13.go +++ /dev/null @@ -1,100 +0,0 @@ -// Copyright 2019 The Go Authors. All rights reserved. -// Use of this source code is governed by a BSD-style -// license that can be found in the LICENSE file. - -//go:build darwin && go1.13 -// +build darwin,go1.13 - -package unix - -import "unsafe" - -//sys closedir(dir uintptr) (err error) -//sys readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) - -func fdopendir(fd int) (dir uintptr, err error) { - r0, _, e1 := syscall_syscallPtr(libc_fdopendir_trampoline_addr, uintptr(fd), 0, 0) - dir = uintptr(r0) - if e1 != 0 { - err = errnoErr(e1) - } - return -} - -var libc_fdopendir_trampoline_addr uintptr - -//go:cgo_import_dynamic libc_fdopendir fdopendir "/usr/lib/libSystem.B.dylib" - -func Getdirentries(fd int, buf []byte, basep *uintptr) (n int, err error) { - // Simulate Getdirentries using fdopendir/readdir_r/closedir. - // We store the number of entries to skip in the seek - // offset of fd. See issue #31368. - // It's not the full required semantics, but should handle the case - // of calling Getdirentries or ReadDirent repeatedly. - // It won't handle assigning the results of lseek to *basep, or handle - // the directory being edited underfoot. - skip, err := Seek(fd, 0, 1 /* SEEK_CUR */) - if err != nil { - return 0, err - } - - // We need to duplicate the incoming file descriptor - // because the caller expects to retain control of it, but - // fdopendir expects to take control of its argument. - // Just Dup'ing the file descriptor is not enough, as the - // result shares underlying state. Use Openat to make a really - // new file descriptor referring to the same directory. - fd2, err := Openat(fd, ".", O_RDONLY, 0) - if err != nil { - return 0, err - } - d, err := fdopendir(fd2) - if err != nil { - Close(fd2) - return 0, err - } - defer closedir(d) - - var cnt int64 - for { - var entry Dirent - var entryp *Dirent - e := readdir_r(d, &entry, &entryp) - if e != 0 { - return n, errnoErr(e) - } - if entryp == nil { - break - } - if skip > 0 { - skip-- - cnt++ - continue - } - - reclen := int(entry.Reclen) - if reclen > len(buf) { - // Not enough room. Return for now. - // The counter will let us know where we should start up again. - // Note: this strategy for suspending in the middle and - // restarting is O(n^2) in the length of the directory. Oh well. - break - } - - // Copy entry into return buffer. - s := unsafe.Slice((*byte)(unsafe.Pointer(&entry)), reclen) - copy(buf, s) - - buf = buf[reclen:] - n += reclen - cnt++ - } - // Set the seek offset of the input fd to record - // how many files we've already returned. - _, err = Seek(fd, cnt, 0 /* SEEK_SET */) - if err != nil { - return n, err - } - - return n, nil -} diff --git a/vendor/golang.org/x/sys/unix/syscall_darwin.go b/vendor/golang.org/x/sys/unix/syscall_darwin.go index 4f87f16ea..1f6338218 100644 --- a/vendor/golang.org/x/sys/unix/syscall_darwin.go +++ b/vendor/golang.org/x/sys/unix/syscall_darwin.go @@ -19,6 +19,96 @@ import ( "unsafe" ) +//sys closedir(dir uintptr) (err error) +//sys readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) + +func fdopendir(fd int) (dir uintptr, err error) { + r0, _, e1 := syscall_syscallPtr(libc_fdopendir_trampoline_addr, uintptr(fd), 0, 0) + dir = uintptr(r0) + if e1 != 0 { + err = errnoErr(e1) + } + return +} + +var libc_fdopendir_trampoline_addr uintptr + +//go:cgo_import_dynamic libc_fdopendir fdopendir "/usr/lib/libSystem.B.dylib" + +func Getdirentries(fd int, buf []byte, basep *uintptr) (n int, err error) { + // Simulate Getdirentries using fdopendir/readdir_r/closedir. + // We store the number of entries to skip in the seek + // offset of fd. See issue #31368. + // It's not the full required semantics, but should handle the case + // of calling Getdirentries or ReadDirent repeatedly. + // It won't handle assigning the results of lseek to *basep, or handle + // the directory being edited underfoot. + skip, err := Seek(fd, 0, 1 /* SEEK_CUR */) + if err != nil { + return 0, err + } + + // We need to duplicate the incoming file descriptor + // because the caller expects to retain control of it, but + // fdopendir expects to take control of its argument. + // Just Dup'ing the file descriptor is not enough, as the + // result shares underlying state. Use Openat to make a really + // new file descriptor referring to the same directory. + fd2, err := Openat(fd, ".", O_RDONLY, 0) + if err != nil { + return 0, err + } + d, err := fdopendir(fd2) + if err != nil { + Close(fd2) + return 0, err + } + defer closedir(d) + + var cnt int64 + for { + var entry Dirent + var entryp *Dirent + e := readdir_r(d, &entry, &entryp) + if e != 0 { + return n, errnoErr(e) + } + if entryp == nil { + break + } + if skip > 0 { + skip-- + cnt++ + continue + } + + reclen := int(entry.Reclen) + if reclen > len(buf) { + // Not enough room. Return for now. + // The counter will let us know where we should start up again. + // Note: this strategy for suspending in the middle and + // restarting is O(n^2) in the length of the directory. Oh well. + break + } + + // Copy entry into return buffer. + s := unsafe.Slice((*byte)(unsafe.Pointer(&entry)), reclen) + copy(buf, s) + + buf = buf[reclen:] + n += reclen + cnt++ + } + // Set the seek offset of the input fd to record + // how many files we've already returned. + _, err = Seek(fd, cnt, 0 /* SEEK_SET */) + if err != nil { + return n, err + } + + return n, nil +} + // SockaddrDatalink implements the Sockaddr interface for AF_LINK type sockets. type SockaddrDatalink struct { Len uint8 diff --git a/vendor/golang.org/x/sys/unix/xattr_bsd.go b/vendor/golang.org/x/sys/unix/xattr_bsd.go index 25df1e378..663b3779d 100644 --- a/vendor/golang.org/x/sys/unix/xattr_bsd.go +++ b/vendor/golang.org/x/sys/unix/xattr_bsd.go @@ -160,13 +160,12 @@ func Lremovexattr(link string, attr string) (err error) { } func Listxattr(file string, dest []byte) (sz int, err error) { - d := initxattrdest(dest, 0) destsiz := len(dest) // FreeBSD won't allow you to list xattrs from multiple namespaces - s := 0 + s, pos := 0, 0 for _, nsid := range [...]int{EXTATTR_NAMESPACE_USER, EXTATTR_NAMESPACE_SYSTEM} { - stmp, e := ExtattrListFile(file, nsid, uintptr(d), destsiz) + stmp, e := ListxattrNS(file, nsid, dest[pos:]) /* Errors accessing system attrs are ignored so that * we can implement the Linux-like behavior of omitting errors that @@ -175,66 +174,102 @@ func Listxattr(file string, dest []byte) (sz int, err error) { * Linux will still error if we ask for user attributes on a file that * we don't have read permissions on, so don't ignore those errors */ - if e != nil && e == EPERM && nsid != EXTATTR_NAMESPACE_USER { - continue - } else if e != nil { + if e != nil { + if e == EPERM && nsid != EXTATTR_NAMESPACE_USER { + continue + } return s, e } s += stmp - destsiz -= s - if destsiz < 0 { - destsiz = 0 + pos = s + if pos > destsiz { + pos = destsiz } - d = initxattrdest(dest, s) + } + + return s, nil +} + +func ListxattrNS(file string, nsid int, dest []byte) (sz int, err error) { + d := initxattrdest(dest, 0) + destsiz := len(dest) + + s, e := ExtattrListFile(file, nsid, uintptr(d), destsiz) + if e != nil { + return 0, err } return s, nil } func Flistxattr(fd int, dest []byte) (sz int, err error) { - d := initxattrdest(dest, 0) destsiz := len(dest) - s := 0 + s, pos := 0, 0 for _, nsid := range [...]int{EXTATTR_NAMESPACE_USER, EXTATTR_NAMESPACE_SYSTEM} { - stmp, e := ExtattrListFd(fd, nsid, uintptr(d), destsiz) - if e != nil && e == EPERM && nsid != EXTATTR_NAMESPACE_USER { - continue - } else if e != nil { + stmp, e := FlistxattrNS(fd, nsid, dest[pos:]) + + if e != nil { + if e == EPERM && nsid != EXTATTR_NAMESPACE_USER { + continue + } return s, e } s += stmp - destsiz -= s - if destsiz < 0 { - destsiz = 0 + pos = s + if pos > destsiz { + pos = destsiz } - d = initxattrdest(dest, s) + } + + return s, nil +} + +func FlistxattrNS(fd int, nsid int, dest []byte) (sz int, err error) { + d := initxattrdest(dest, 0) + destsiz := len(dest) + + s, e := ExtattrListFd(fd, nsid, uintptr(d), destsiz) + if e != nil { + return 0, err } return s, nil } func Llistxattr(link string, dest []byte) (sz int, err error) { - d := initxattrdest(dest, 0) destsiz := len(dest) - s := 0 + s, pos := 0, 0 for _, nsid := range [...]int{EXTATTR_NAMESPACE_USER, EXTATTR_NAMESPACE_SYSTEM} { - stmp, e := ExtattrListLink(link, nsid, uintptr(d), destsiz) - if e != nil && e == EPERM && nsid != EXTATTR_NAMESPACE_USER { - continue - } else if e != nil { + stmp, e := LlistxattrNS(link, nsid, dest[pos:]) + + if e != nil { + if e == EPERM && nsid != EXTATTR_NAMESPACE_USER { + continue + } return s, e } s += stmp - destsiz -= s - if destsiz < 0 { - destsiz = 0 + pos = s + if pos > destsiz { + pos = destsiz } - d = initxattrdest(dest, s) + } + + return s, nil +} + +func LlistxattrNS(link string, nsid int, dest []byte) (sz int, err error) { + d := initxattrdest(dest, 0) + destsiz := len(dest) + + s, e := ExtattrListLink(link, nsid, uintptr(d), destsiz) + if e != nil { + return 0, err } return s, nil diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.go b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.go deleted file mode 100644 index a06eb0932..000000000 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.go +++ /dev/null @@ -1,40 +0,0 @@ -// go run mksyscall.go -tags darwin,amd64,go1.13 syscall_darwin.1_13.go -// Code generated by the command above; see README.md. DO NOT EDIT. - -//go:build darwin && amd64 && go1.13 -// +build darwin,amd64,go1.13 - -package unix - -import ( - "syscall" - "unsafe" -) - -var _ syscall.Errno - -// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT - -func closedir(dir uintptr) (err error) { - _, _, e1 := syscall_syscall(libc_closedir_trampoline_addr, uintptr(dir), 0, 0) - if e1 != 0 { - err = errnoErr(e1) - } - return -} - -var libc_closedir_trampoline_addr uintptr - -//go:cgo_import_dynamic libc_closedir closedir "/usr/lib/libSystem.B.dylib" - -// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT - -func readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) { - r0, _, _ := syscall_syscall(libc_readdir_r_trampoline_addr, uintptr(dir), uintptr(unsafe.Pointer(entry)), uintptr(unsafe.Pointer(result))) - res = Errno(r0) - return -} - -var libc_readdir_r_trampoline_addr uintptr - -//go:cgo_import_dynamic libc_readdir_r readdir_r "/usr/lib/libSystem.B.dylib" diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.s b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.s deleted file mode 100644 index f5bb40eda..000000000 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.1_13.s +++ /dev/null @@ -1,25 +0,0 @@ -// go run mkasm.go darwin amd64 -// Code generated by the command above; DO NOT EDIT. - -//go:build go1.13 -// +build go1.13 - -#include "textflag.h" - -TEXT libc_fdopendir_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_fdopendir(SB) - -GLOBL ·libc_fdopendir_trampoline_addr(SB), RODATA, $8 -DATA ·libc_fdopendir_trampoline_addr(SB)/8, $libc_fdopendir_trampoline<>(SB) - -TEXT libc_closedir_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_closedir(SB) - -GLOBL ·libc_closedir_trampoline_addr(SB), RODATA, $8 -DATA ·libc_closedir_trampoline_addr(SB)/8, $libc_closedir_trampoline<>(SB) - -TEXT libc_readdir_r_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_readdir_r(SB) - -GLOBL ·libc_readdir_r_trampoline_addr(SB), RODATA, $8 -DATA ·libc_readdir_r_trampoline_addr(SB)/8, $libc_readdir_r_trampoline<>(SB) diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.go b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.go index 467deed76..c2461c496 100644 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.go +++ b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.go @@ -1,8 +1,8 @@ -// go run mksyscall.go -tags darwin,amd64,go1.12 syscall_bsd.go syscall_darwin.go syscall_darwin_amd64.go +// go run mksyscall.go -tags darwin,amd64 syscall_bsd.go syscall_darwin.go syscall_darwin_amd64.go // Code generated by the command above; see README.md. DO NOT EDIT. -//go:build darwin && amd64 && go1.12 -// +build darwin,amd64,go1.12 +//go:build darwin && amd64 +// +build darwin,amd64 package unix @@ -463,6 +463,32 @@ var libc_munlockall_trampoline_addr uintptr // THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT +func closedir(dir uintptr) (err error) { + _, _, e1 := syscall_syscall(libc_closedir_trampoline_addr, uintptr(dir), 0, 0) + if e1 != 0 { + err = errnoErr(e1) + } + return +} + +var libc_closedir_trampoline_addr uintptr + +//go:cgo_import_dynamic libc_closedir closedir "/usr/lib/libSystem.B.dylib" + +// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT + +func readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) { + r0, _, _ := syscall_syscall(libc_readdir_r_trampoline_addr, uintptr(dir), uintptr(unsafe.Pointer(entry)), uintptr(unsafe.Pointer(result))) + res = Errno(r0) + return +} + +var libc_readdir_r_trampoline_addr uintptr + +//go:cgo_import_dynamic libc_readdir_r readdir_r "/usr/lib/libSystem.B.dylib" + +// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT + func pipe(p *[2]int32) (err error) { _, _, e1 := syscall_rawSyscall(libc_pipe_trampoline_addr, uintptr(unsafe.Pointer(p)), 0, 0) if e1 != 0 { diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.s b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.s index b41467a0e..95fe4c0eb 100644 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.s +++ b/vendor/golang.org/x/sys/unix/zsyscall_darwin_amd64.s @@ -1,11 +1,14 @@ // go run mkasm.go darwin amd64 // Code generated by the command above; DO NOT EDIT. -//go:build go1.12 -// +build go1.12 - #include "textflag.h" +TEXT libc_fdopendir_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_fdopendir(SB) + +GLOBL ·libc_fdopendir_trampoline_addr(SB), RODATA, $8 +DATA ·libc_fdopendir_trampoline_addr(SB)/8, $libc_fdopendir_trampoline<>(SB) + TEXT libc_getgroups_trampoline<>(SB),NOSPLIT,$0-0 JMP libc_getgroups(SB) @@ -174,6 +177,18 @@ TEXT libc_munlockall_trampoline<>(SB),NOSPLIT,$0-0 GLOBL ·libc_munlockall_trampoline_addr(SB), RODATA, $8 DATA ·libc_munlockall_trampoline_addr(SB)/8, $libc_munlockall_trampoline<>(SB) +TEXT libc_closedir_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_closedir(SB) + +GLOBL ·libc_closedir_trampoline_addr(SB), RODATA, $8 +DATA ·libc_closedir_trampoline_addr(SB)/8, $libc_closedir_trampoline<>(SB) + +TEXT libc_readdir_r_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_readdir_r(SB) + +GLOBL ·libc_readdir_r_trampoline_addr(SB), RODATA, $8 +DATA ·libc_readdir_r_trampoline_addr(SB)/8, $libc_readdir_r_trampoline<>(SB) + TEXT libc_pipe_trampoline<>(SB),NOSPLIT,$0-0 JMP libc_pipe(SB) diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.go b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.go deleted file mode 100644 index cec595d55..000000000 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.go +++ /dev/null @@ -1,40 +0,0 @@ -// go run mksyscall.go -tags darwin,arm64,go1.13 syscall_darwin.1_13.go -// Code generated by the command above; see README.md. DO NOT EDIT. - -//go:build darwin && arm64 && go1.13 -// +build darwin,arm64,go1.13 - -package unix - -import ( - "syscall" - "unsafe" -) - -var _ syscall.Errno - -// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT - -func closedir(dir uintptr) (err error) { - _, _, e1 := syscall_syscall(libc_closedir_trampoline_addr, uintptr(dir), 0, 0) - if e1 != 0 { - err = errnoErr(e1) - } - return -} - -var libc_closedir_trampoline_addr uintptr - -//go:cgo_import_dynamic libc_closedir closedir "/usr/lib/libSystem.B.dylib" - -// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT - -func readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) { - r0, _, _ := syscall_syscall(libc_readdir_r_trampoline_addr, uintptr(dir), uintptr(unsafe.Pointer(entry)), uintptr(unsafe.Pointer(result))) - res = Errno(r0) - return -} - -var libc_readdir_r_trampoline_addr uintptr - -//go:cgo_import_dynamic libc_readdir_r readdir_r "/usr/lib/libSystem.B.dylib" diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.s b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.s deleted file mode 100644 index 0c3f76bc2..000000000 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.1_13.s +++ /dev/null @@ -1,25 +0,0 @@ -// go run mkasm.go darwin arm64 -// Code generated by the command above; DO NOT EDIT. - -//go:build go1.13 -// +build go1.13 - -#include "textflag.h" - -TEXT libc_fdopendir_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_fdopendir(SB) - -GLOBL ·libc_fdopendir_trampoline_addr(SB), RODATA, $8 -DATA ·libc_fdopendir_trampoline_addr(SB)/8, $libc_fdopendir_trampoline<>(SB) - -TEXT libc_closedir_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_closedir(SB) - -GLOBL ·libc_closedir_trampoline_addr(SB), RODATA, $8 -DATA ·libc_closedir_trampoline_addr(SB)/8, $libc_closedir_trampoline<>(SB) - -TEXT libc_readdir_r_trampoline<>(SB),NOSPLIT,$0-0 - JMP libc_readdir_r(SB) - -GLOBL ·libc_readdir_r_trampoline_addr(SB), RODATA, $8 -DATA ·libc_readdir_r_trampoline_addr(SB)/8, $libc_readdir_r_trampoline<>(SB) diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.go b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.go index 35938d34f..26a0fdc50 100644 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.go +++ b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.go @@ -1,8 +1,8 @@ -// go run mksyscall.go -tags darwin,arm64,go1.12 syscall_bsd.go syscall_darwin.go syscall_darwin_arm64.go +// go run mksyscall.go -tags darwin,arm64 syscall_bsd.go syscall_darwin.go syscall_darwin_arm64.go // Code generated by the command above; see README.md. DO NOT EDIT. -//go:build darwin && arm64 && go1.12 -// +build darwin,arm64,go1.12 +//go:build darwin && arm64 +// +build darwin,arm64 package unix @@ -463,6 +463,32 @@ var libc_munlockall_trampoline_addr uintptr // THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT +func closedir(dir uintptr) (err error) { + _, _, e1 := syscall_syscall(libc_closedir_trampoline_addr, uintptr(dir), 0, 0) + if e1 != 0 { + err = errnoErr(e1) + } + return +} + +var libc_closedir_trampoline_addr uintptr + +//go:cgo_import_dynamic libc_closedir closedir "/usr/lib/libSystem.B.dylib" + +// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT + +func readdir_r(dir uintptr, entry *Dirent, result **Dirent) (res Errno) { + r0, _, _ := syscall_syscall(libc_readdir_r_trampoline_addr, uintptr(dir), uintptr(unsafe.Pointer(entry)), uintptr(unsafe.Pointer(result))) + res = Errno(r0) + return +} + +var libc_readdir_r_trampoline_addr uintptr + +//go:cgo_import_dynamic libc_readdir_r readdir_r "/usr/lib/libSystem.B.dylib" + +// THIS FILE IS GENERATED BY THE COMMAND AT THE TOP; DO NOT EDIT + func pipe(p *[2]int32) (err error) { _, _, e1 := syscall_rawSyscall(libc_pipe_trampoline_addr, uintptr(unsafe.Pointer(p)), 0, 0) if e1 != 0 { diff --git a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.s b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.s index e1f9204a2..efa5b4c98 100644 --- a/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.s +++ b/vendor/golang.org/x/sys/unix/zsyscall_darwin_arm64.s @@ -1,11 +1,14 @@ // go run mkasm.go darwin arm64 // Code generated by the command above; DO NOT EDIT. -//go:build go1.12 -// +build go1.12 - #include "textflag.h" +TEXT libc_fdopendir_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_fdopendir(SB) + +GLOBL ·libc_fdopendir_trampoline_addr(SB), RODATA, $8 +DATA ·libc_fdopendir_trampoline_addr(SB)/8, $libc_fdopendir_trampoline<>(SB) + TEXT libc_getgroups_trampoline<>(SB),NOSPLIT,$0-0 JMP libc_getgroups(SB) @@ -174,6 +177,18 @@ TEXT libc_munlockall_trampoline<>(SB),NOSPLIT,$0-0 GLOBL ·libc_munlockall_trampoline_addr(SB), RODATA, $8 DATA ·libc_munlockall_trampoline_addr(SB)/8, $libc_munlockall_trampoline<>(SB) +TEXT libc_closedir_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_closedir(SB) + +GLOBL ·libc_closedir_trampoline_addr(SB), RODATA, $8 +DATA ·libc_closedir_trampoline_addr(SB)/8, $libc_closedir_trampoline<>(SB) + +TEXT libc_readdir_r_trampoline<>(SB),NOSPLIT,$0-0 + JMP libc_readdir_r(SB) + +GLOBL ·libc_readdir_r_trampoline_addr(SB), RODATA, $8 +DATA ·libc_readdir_r_trampoline_addr(SB)/8, $libc_readdir_r_trampoline<>(SB) + TEXT libc_pipe_trampoline<>(SB),NOSPLIT,$0-0 JMP libc_pipe(SB) diff --git a/vendor/golang.org/x/sys/windows/syscall.go b/vendor/golang.org/x/sys/windows/syscall.go index 72074d582..8732cdb95 100644 --- a/vendor/golang.org/x/sys/windows/syscall.go +++ b/vendor/golang.org/x/sys/windows/syscall.go @@ -30,8 +30,6 @@ import ( "strings" "syscall" "unsafe" - - "golang.org/x/sys/internal/unsafeheader" ) // ByteSliceFromString returns a NUL-terminated slice of bytes @@ -83,13 +81,7 @@ func BytePtrToString(p *byte) string { ptr = unsafe.Pointer(uintptr(ptr) + 1) } - var s []byte - h := (*unsafeheader.Slice)(unsafe.Pointer(&s)) - h.Data = unsafe.Pointer(p) - h.Len = n - h.Cap = n - - return string(s) + return string(unsafe.Slice(p, n)) } // Single-word zero for use when we need a valid pointer to 0 bytes. diff --git a/vendor/golang.org/x/sys/windows/syscall_windows.go b/vendor/golang.org/x/sys/windows/syscall_windows.go index 996cc53c0..3f2cbb638 100644 --- a/vendor/golang.org/x/sys/windows/syscall_windows.go +++ b/vendor/golang.org/x/sys/windows/syscall_windows.go @@ -138,13 +138,7 @@ func UTF16PtrToString(p *uint16) string { ptr = unsafe.Pointer(uintptr(ptr) + unsafe.Sizeof(*p)) } - var s []uint16 - h := (*unsafeheader.Slice)(unsafe.Pointer(&s)) - h.Data = unsafe.Pointer(p) - h.Len = n - h.Cap = n - - return string(utf16.Decode(s)) + return string(utf16.Decode(unsafe.Slice(p, n))) } func Getpagesize() int { return 4096 } @@ -448,6 +442,10 @@ func NewCallbackCDecl(fn interface{}) uintptr { //sys RtlAddFunctionTable(functionTable *RUNTIME_FUNCTION, entryCount uint32, baseAddress uintptr) (ret bool) = ntdll.RtlAddFunctionTable //sys RtlDeleteFunctionTable(functionTable *RUNTIME_FUNCTION) (ret bool) = ntdll.RtlDeleteFunctionTable +// Desktop Window Manager API (Dwmapi) +//sys DwmGetWindowAttribute(hwnd HWND, attribute uint32, value unsafe.Pointer, size uint32) (ret error) = dwmapi.DwmGetWindowAttribute +//sys DwmSetWindowAttribute(hwnd HWND, attribute uint32, value unsafe.Pointer, size uint32) (ret error) = dwmapi.DwmSetWindowAttribute + // syscall interface implementation for other packages // GetCurrentProcess returns the handle for the current process. diff --git a/vendor/golang.org/x/sys/windows/types_windows.go b/vendor/golang.org/x/sys/windows/types_windows.go index ef489f546..0c4add974 100644 --- a/vendor/golang.org/x/sys/windows/types_windows.go +++ b/vendor/golang.org/x/sys/windows/types_windows.go @@ -3232,3 +3232,29 @@ type GUIThreadInfo struct { CaretHandle HWND CaretRect Rect } + +const ( + DWMWA_NCRENDERING_ENABLED = 1 + DWMWA_NCRENDERING_POLICY = 2 + DWMWA_TRANSITIONS_FORCEDISABLED = 3 + DWMWA_ALLOW_NCPAINT = 4 + DWMWA_CAPTION_BUTTON_BOUNDS = 5 + DWMWA_NONCLIENT_RTL_LAYOUT = 6 + DWMWA_FORCE_ICONIC_REPRESENTATION = 7 + DWMWA_FLIP3D_POLICY = 8 + DWMWA_EXTENDED_FRAME_BOUNDS = 9 + DWMWA_HAS_ICONIC_BITMAP = 10 + DWMWA_DISALLOW_PEEK = 11 + DWMWA_EXCLUDED_FROM_PEEK = 12 + DWMWA_CLOAK = 13 + DWMWA_CLOAKED = 14 + DWMWA_FREEZE_REPRESENTATION = 15 + DWMWA_PASSIVE_UPDATE_MODE = 16 + DWMWA_USE_HOSTBACKDROPBRUSH = 17 + DWMWA_USE_IMMERSIVE_DARK_MODE = 20 + DWMWA_WINDOW_CORNER_PREFERENCE = 33 + DWMWA_BORDER_COLOR = 34 + DWMWA_CAPTION_COLOR = 35 + DWMWA_TEXT_COLOR = 36 + DWMWA_VISIBLE_FRAME_BORDER_THICKNESS = 37 +) diff --git a/vendor/golang.org/x/sys/windows/zsyscall_windows.go b/vendor/golang.org/x/sys/windows/zsyscall_windows.go index 6c6f27ba5..96ba8559c 100644 --- a/vendor/golang.org/x/sys/windows/zsyscall_windows.go +++ b/vendor/golang.org/x/sys/windows/zsyscall_windows.go @@ -40,6 +40,7 @@ var ( modadvapi32 = NewLazySystemDLL("advapi32.dll") modcrypt32 = NewLazySystemDLL("crypt32.dll") moddnsapi = NewLazySystemDLL("dnsapi.dll") + moddwmapi = NewLazySystemDLL("dwmapi.dll") modiphlpapi = NewLazySystemDLL("iphlpapi.dll") modkernel32 = NewLazySystemDLL("kernel32.dll") modmswsock = NewLazySystemDLL("mswsock.dll") @@ -175,6 +176,8 @@ var ( procDnsNameCompare_W = moddnsapi.NewProc("DnsNameCompare_W") procDnsQuery_W = moddnsapi.NewProc("DnsQuery_W") procDnsRecordListFree = moddnsapi.NewProc("DnsRecordListFree") + procDwmGetWindowAttribute = moddwmapi.NewProc("DwmGetWindowAttribute") + procDwmSetWindowAttribute = moddwmapi.NewProc("DwmSetWindowAttribute") procGetAdaptersAddresses = modiphlpapi.NewProc("GetAdaptersAddresses") procGetAdaptersInfo = modiphlpapi.NewProc("GetAdaptersInfo") procGetBestInterfaceEx = modiphlpapi.NewProc("GetBestInterfaceEx") @@ -1534,6 +1537,22 @@ func DnsRecordListFree(rl *DNSRecord, freetype uint32) { return } +func DwmGetWindowAttribute(hwnd HWND, attribute uint32, value unsafe.Pointer, size uint32) (ret error) { + r0, _, _ := syscall.Syscall6(procDwmGetWindowAttribute.Addr(), 4, uintptr(hwnd), uintptr(attribute), uintptr(value), uintptr(size), 0, 0) + if r0 != 0 { + ret = syscall.Errno(r0) + } + return +} + +func DwmSetWindowAttribute(hwnd HWND, attribute uint32, value unsafe.Pointer, size uint32) (ret error) { + r0, _, _ := syscall.Syscall6(procDwmSetWindowAttribute.Addr(), 4, uintptr(hwnd), uintptr(attribute), uintptr(value), uintptr(size), 0, 0) + if r0 != 0 { + ret = syscall.Errno(r0) + } + return +} + func GetAdaptersAddresses(family uint32, flags uint32, reserved uintptr, adapterAddresses *IpAdapterAddresses, sizePointer *uint32) (errcode error) { r0, _, _ := syscall.Syscall6(procGetAdaptersAddresses.Addr(), 5, uintptr(family), uintptr(flags), uintptr(reserved), uintptr(unsafe.Pointer(adapterAddresses)), uintptr(unsafe.Pointer(sizePointer)), 0) if r0 != 0 { diff --git a/vendor/google.golang.org/api/internal/version.go b/vendor/google.golang.org/api/internal/version.go index 743154c54..08d3c916f 100644 --- a/vendor/google.golang.org/api/internal/version.go +++ b/vendor/google.golang.org/api/internal/version.go @@ -5,4 +5,4 @@ package internal // Version is the current tagged release of the library. -const Version = "0.95.0" +const Version = "0.96.0" diff --git a/vendor/modules.txt b/vendor/modules.txt index b02361808..7ef11b256 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -4,7 +4,7 @@ cloud.google.com/go/internal cloud.google.com/go/internal/optional cloud.google.com/go/internal/trace cloud.google.com/go/internal/version -# cloud.google.com/go/compute v1.9.0 +# cloud.google.com/go/compute v1.10.0 ## explicit; go 1.17 cloud.google.com/go/compute/metadata # cloud.google.com/go/iam v0.4.0 @@ -34,7 +34,7 @@ github.com/VictoriaMetrics/metricsql/binaryop # github.com/VividCortex/ewma v1.2.0 ## explicit; go 1.12 github.com/VividCortex/ewma -# github.com/aws/aws-sdk-go v1.44.96 +# github.com/aws/aws-sdk-go v1.44.100 ## explicit; go 1.11 github.com/aws/aws-sdk-go/aws github.com/aws/aws-sdk-go/aws/arn @@ -157,8 +157,8 @@ github.com/influxdata/influxdb/pkg/escape # github.com/jmespath/go-jmespath v0.4.0 ## explicit; go 1.14 github.com/jmespath/go-jmespath -# github.com/klauspost/compress v1.15.9 -## explicit; go 1.16 +# github.com/klauspost/compress v1.15.10 +## explicit; go 1.17 github.com/klauspost/compress github.com/klauspost/compress/flate github.com/klauspost/compress/fse @@ -305,7 +305,7 @@ golang.org/x/oauth2/jwt # golang.org/x/sync v0.0.0-20220907140024-f12130a52804 ## explicit golang.org/x/sync/errgroup -# golang.org/x/sys v0.0.0-20220913120320-3275c407cedc +# golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8 ## explicit; go 1.17 golang.org/x/sys/internal/unsafeheader golang.org/x/sys/unix @@ -320,7 +320,7 @@ golang.org/x/text/unicode/norm ## explicit; go 1.17 golang.org/x/xerrors golang.org/x/xerrors/internal -# google.golang.org/api v0.95.0 +# google.golang.org/api v0.96.0 ## explicit; go 1.15 google.golang.org/api/googleapi google.golang.org/api/googleapi/transport @@ -353,7 +353,7 @@ google.golang.org/appengine/internal/socket google.golang.org/appengine/internal/urlfetch google.golang.org/appengine/socket google.golang.org/appengine/urlfetch -# google.golang.org/genproto v0.0.0-20220909194730-69f6226f97e5 +# google.golang.org/genproto v0.0.0-20220916172020-2692e8806bfa ## explicit; go 1.19 google.golang.org/genproto/googleapis/api/annotations google.golang.org/genproto/googleapis/iam/v1 From 5e2fcd455daff851fdc11b359adf6a3321d5bb5b Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Mon, 19 Sep 2022 15:13:43 +0300 Subject: [PATCH 18/18] app/vmagent/remotewrite: `go fmt` after 2b55d167d787667017f6acf83bc3607aba7916db --- app/vmagent/remotewrite/pendingseries_test.go | 8 ++++---- app/vmagent/remotewrite/pendingseries_timing_test.go | 2 +- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/app/vmagent/remotewrite/pendingseries_test.go b/app/vmagent/remotewrite/pendingseries_test.go index 74900a13b..be99ea9a8 100644 --- a/app/vmagent/remotewrite/pendingseries_test.go +++ b/app/vmagent/remotewrite/pendingseries_test.go @@ -4,8 +4,8 @@ import ( "fmt" "testing" - "github.com/golang/snappy" "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" + "github.com/golang/snappy" ) func TestPushWriteRequest(t *testing.T) { @@ -44,7 +44,7 @@ func newTestWriteRequest(seriesCount, labelsCount int) *prompbmarshal.WriteReque var labels []prompbmarshal.Label for j := 0; j < labelsCount; j++ { labels = append(labels, prompbmarshal.Label{ - Name: fmt.Sprintf("label_%d_%d", i, j), + Name: fmt.Sprintf("label_%d_%d", i, j), Value: fmt.Sprintf("value_%d_%d", i, j), }) } @@ -52,8 +52,8 @@ func newTestWriteRequest(seriesCount, labelsCount int) *prompbmarshal.WriteReque Labels: labels, Samples: []prompbmarshal.Sample{ { - Value: float64(i), - Timestamp: 1000*int64(i), + Value: float64(i), + Timestamp: 1000 * int64(i), }, }, }) diff --git a/app/vmagent/remotewrite/pendingseries_timing_test.go b/app/vmagent/remotewrite/pendingseries_timing_test.go index 72bfcb1ff..8fbd4224f 100644 --- a/app/vmagent/remotewrite/pendingseries_timing_test.go +++ b/app/vmagent/remotewrite/pendingseries_timing_test.go @@ -4,9 +4,9 @@ import ( "fmt" "testing" + "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" "github.com/golang/snappy" "github.com/klauspost/compress/s2" - "github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal" ) func BenchmarkCompressWriteRequestSnappy(b *testing.B) {
`) -//line app/vmalert/web.qtpl:450 +//line app/vmalert/web.qtpl:454 qw422016.E().S(u.time.Format(time.RFC3339)) -//line app/vmalert/web.qtpl:450 +//line app/vmalert/web.qtpl:454 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:452 +//line app/vmalert/web.qtpl:456 qw422016.N().D(u.samples) -//line app/vmalert/web.qtpl:452 +//line app/vmalert/web.qtpl:456 qw422016.N().S(` `) -//line app/vmalert/web.qtpl:453 +//line app/vmalert/web.qtpl:457 qw422016.N().FPrec(u.duration.Seconds(), 3) -//line app/vmalert/web.qtpl:453 +//line app/vmalert/web.qtpl:457 qw422016.N().S(`s `) -//line app/vmalert/web.qtpl:454 +//line app/vmalert/web.qtpl:458 qw422016.E().S(u.at.Format(time.RFC3339)) -//line app/vmalert/web.qtpl:454 +//line app/vmalert/web.qtpl:458 qw422016.N().S(`
+ `) -//line app/vmalert/web.qtpl:463 +//line app/vmalert/web.qtpl:467 qw422016.E().V(u.err) -//line app/vmalert/web.qtpl:463 +//line app/vmalert/web.qtpl:467 qw422016.N().S(`