2021-06-01 23:43:28 +08:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// ----------------------------------------------------------------------------
|
2022-04-24 19:15:08 +08:00
|
|
|
// Copyright 2011-2022 Arm Limited
|
2021-06-01 23:43:28 +08:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
|
|
|
// use this file except in compliance with the License. You may obtain a copy
|
|
|
|
// of the License at:
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
// License for the specific language governing permissions and limitations
|
|
|
|
// under the License.
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Functions for finding dominant direction of a set of colors.
|
|
|
|
*/
|
|
|
|
#if !defined(ASTCENC_DECOMPRESS_ONLY)
|
|
|
|
|
|
|
|
#include "astcenc_internal.h"
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_avgs_and_dirs_4_comp(
|
|
|
|
const partition_info& pi,
|
|
|
|
const image_block& blk,
|
|
|
|
partition_metrics pm[BLOCK_MAX_PARTITIONS]
|
|
|
|
) {
|
2022-04-24 19:15:08 +08:00
|
|
|
float texel_weight = hadd_s(blk.channel_weight) / 4.0f;
|
|
|
|
|
2021-06-01 23:43:28 +08:00
|
|
|
int partition_count = pi.partition_count;
|
|
|
|
promise(partition_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 average = blk.data_mean;
|
|
|
|
|
2021-06-01 23:43:28 +08:00
|
|
|
for (int partition = 0; partition < partition_count; partition++)
|
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
const uint8_t *texel_indexes = pi.texels_of_partition[partition];
|
|
|
|
unsigned int texel_count = pi.partition_texel_count[partition];
|
2021-06-01 23:43:28 +08:00
|
|
|
promise(texel_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
// Only compute a partition mean if more than one partition
|
|
|
|
if (partition_count > 1)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
average = vfloat4::zero();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
|
|
|
int iwt = texel_indexes[i];
|
|
|
|
average += blk.texel(iwt);
|
|
|
|
}
|
|
|
|
|
|
|
|
average = average * (1.0f / static_cast<float>(texel_count));
|
2021-06-01 23:43:28 +08:00
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
pm[partition].avg = average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 sum_xp = vfloat4::zero();
|
|
|
|
vfloat4 sum_yp = vfloat4::zero();
|
|
|
|
vfloat4 sum_zp = vfloat4::zero();
|
|
|
|
vfloat4 sum_wp = vfloat4::zero();
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
unsigned int iwt = texel_indexes[i];
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat4 texel_datum = blk.texel(iwt);
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_datum = texel_datum - average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 zero = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm0 = texel_datum.swz<0,0,0,0>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_xp += select(zero, texel_datum, tdm0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm1 = texel_datum.swz<1,1,1,1>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_yp += select(zero, texel_datum, tdm1);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm2 = texel_datum.swz<2,2,2,2>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_zp += select(zero, texel_datum, tdm2);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm3 = texel_datum.swz<3,3,3,3>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_wp += select(zero, texel_datum, tdm3);
|
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
sum_xp = sum_xp * texel_weight;
|
|
|
|
sum_yp = sum_yp * texel_weight;
|
|
|
|
sum_zp = sum_zp * texel_weight;
|
|
|
|
sum_wp = sum_wp * texel_weight;
|
|
|
|
|
|
|
|
vfloat4 prod_xp = dot(sum_xp, sum_xp);
|
|
|
|
vfloat4 prod_yp = dot(sum_yp, sum_yp);
|
|
|
|
vfloat4 prod_zp = dot(sum_zp, sum_zp);
|
|
|
|
vfloat4 prod_wp = dot(sum_wp, sum_wp);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 best_vector = sum_xp;
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 best_sum = prod_xp;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 mask = prod_yp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_yp, mask);
|
|
|
|
best_sum = select(best_sum, prod_yp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
mask = prod_zp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_zp, mask);
|
|
|
|
best_sum = select(best_sum, prod_zp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
mask = prod_wp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_wp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
pm[partition].dir = best_vector;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_avgs_and_dirs_3_comp(
|
|
|
|
const partition_info& pi,
|
|
|
|
const image_block& blk,
|
|
|
|
unsigned int omitted_component,
|
|
|
|
partition_metrics pm[BLOCK_MAX_PARTITIONS]
|
|
|
|
) {
|
2022-04-24 19:15:08 +08:00
|
|
|
float texel_weight = hadd_s(blk.channel_weight.swz<0, 1, 2>()) / 3.0f;
|
|
|
|
vfloat4 average = blk.data_mean.swz<0, 1, 2>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
const float* data_vr = blk.data_r;
|
|
|
|
const float* data_vg = blk.data_g;
|
|
|
|
const float* data_vb = blk.data_b;
|
|
|
|
|
|
|
|
if (omitted_component == 0)
|
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<1, 2, 3>()) / 3.0f;
|
|
|
|
average = blk.data_mean.swz<1, 2, 3>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vr = blk.data_g;
|
|
|
|
data_vg = blk.data_b;
|
|
|
|
data_vb = blk.data_a;
|
|
|
|
}
|
|
|
|
else if (omitted_component == 1)
|
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<0, 2, 3>()) / 3.0f;
|
|
|
|
average = blk.data_mean.swz<0, 2, 3>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vg = blk.data_b;
|
|
|
|
data_vb = blk.data_a;
|
|
|
|
}
|
|
|
|
else if (omitted_component == 2)
|
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<0, 1, 3>()) / 3.0f;
|
|
|
|
average = blk.data_mean.swz<0, 1, 3>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vb = blk.data_a;
|
|
|
|
}
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
unsigned int partition_count = pi.partition_count;
|
2021-06-01 23:43:28 +08:00
|
|
|
promise(partition_count > 0);
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
for (unsigned int partition = 0; partition < partition_count; partition++)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
const uint8_t *texel_indexes = pi.texels_of_partition[partition];
|
|
|
|
unsigned int texel_count = pi.partition_texel_count[partition];
|
2021-06-01 23:43:28 +08:00
|
|
|
promise(texel_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
// Only compute a partition mean if more than one partition
|
|
|
|
if (partition_count > 1)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
average = vfloat4::zero();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
|
|
|
unsigned int iwt = texel_indexes[i];
|
|
|
|
average += vfloat3(data_vr[iwt], data_vg[iwt], data_vb[iwt]);
|
|
|
|
}
|
|
|
|
|
|
|
|
average = average * (1.0f / static_cast<float>(texel_count));
|
2021-06-01 23:43:28 +08:00
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
pm[partition].avg = average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 sum_xp = vfloat4::zero();
|
|
|
|
vfloat4 sum_yp = vfloat4::zero();
|
|
|
|
vfloat4 sum_zp = vfloat4::zero();
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
unsigned int iwt = texel_indexes[i];
|
|
|
|
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat4 texel_datum = vfloat3(data_vr[iwt],
|
|
|
|
data_vg[iwt],
|
|
|
|
data_vb[iwt]);
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_datum = texel_datum - average;
|
2021-07-02 00:18:02 +08:00
|
|
|
|
|
|
|
vfloat4 zero = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm0 = texel_datum.swz<0,0,0,0>() > zero;
|
2021-07-02 00:18:02 +08:00
|
|
|
sum_xp += select(zero, texel_datum, tdm0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm1 = texel_datum.swz<1,1,1,1>() > zero;
|
2021-07-02 00:18:02 +08:00
|
|
|
sum_yp += select(zero, texel_datum, tdm1);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm2 = texel_datum.swz<2,2,2,2>() > zero;
|
2021-07-02 00:18:02 +08:00
|
|
|
sum_zp += select(zero, texel_datum, tdm2);
|
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
sum_xp = sum_xp * texel_weight;
|
|
|
|
sum_yp = sum_yp * texel_weight;
|
|
|
|
sum_zp = sum_zp * texel_weight;
|
|
|
|
|
|
|
|
vfloat4 prod_xp = dot(sum_xp, sum_xp);
|
|
|
|
vfloat4 prod_yp = dot(sum_yp, sum_yp);
|
|
|
|
vfloat4 prod_zp = dot(sum_zp, sum_zp);
|
2021-07-02 00:18:02 +08:00
|
|
|
|
|
|
|
vfloat4 best_vector = sum_xp;
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 best_sum = prod_xp;
|
2021-07-02 00:18:02 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 mask = prod_yp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_yp, mask);
|
|
|
|
best_sum = select(best_sum, prod_yp, mask);
|
2021-07-02 00:18:02 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
mask = prod_zp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_zp, mask);
|
2021-07-02 00:18:02 +08:00
|
|
|
|
|
|
|
pm[partition].dir = best_vector;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_avgs_and_dirs_3_comp_rgb(
|
|
|
|
const partition_info& pi,
|
|
|
|
const image_block& blk,
|
|
|
|
partition_metrics pm[BLOCK_MAX_PARTITIONS]
|
|
|
|
) {
|
2022-04-24 19:15:08 +08:00
|
|
|
float texel_weight = hadd_s(blk.channel_weight.swz<0, 1, 2>()) / 3;
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
unsigned int partition_count = pi.partition_count;
|
|
|
|
promise(partition_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 average = blk.data_mean.swz<0, 1, 2>();
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
for (unsigned int partition = 0; partition < partition_count; partition++)
|
|
|
|
{
|
|
|
|
const uint8_t *texel_indexes = pi.texels_of_partition[partition];
|
|
|
|
unsigned int texel_count = pi.partition_texel_count[partition];
|
|
|
|
promise(texel_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
// Only compute a partition mean if more than one partition
|
|
|
|
if (partition_count > 1)
|
2021-07-02 00:18:02 +08:00
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
average = vfloat4::zero();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
|
|
|
unsigned int iwt = texel_indexes[i];
|
|
|
|
average += blk.texel3(iwt);
|
|
|
|
}
|
|
|
|
|
|
|
|
average = average * (1.0f / static_cast<float>(texel_count));
|
2021-07-02 00:18:02 +08:00
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
pm[partition].avg = average;
|
2021-07-02 00:18:02 +08:00
|
|
|
|
|
|
|
vfloat4 sum_xp = vfloat4::zero();
|
|
|
|
vfloat4 sum_yp = vfloat4::zero();
|
|
|
|
vfloat4 sum_zp = vfloat4::zero();
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
|
|
|
unsigned int iwt = texel_indexes[i];
|
|
|
|
|
|
|
|
vfloat4 texel_datum = blk.texel3(iwt);
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_datum = texel_datum - average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 zero = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm0 = texel_datum.swz<0,0,0,0>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_xp += select(zero, texel_datum, tdm0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm1 = texel_datum.swz<1,1,1,1>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_yp += select(zero, texel_datum, tdm1);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm2 = texel_datum.swz<2,2,2,2>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_zp += select(zero, texel_datum, tdm2);
|
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
sum_xp = sum_xp * texel_weight;
|
|
|
|
sum_yp = sum_yp * texel_weight;
|
|
|
|
sum_zp = sum_zp * texel_weight;
|
|
|
|
|
|
|
|
vfloat4 prod_xp = dot(sum_xp, sum_xp);
|
|
|
|
vfloat4 prod_yp = dot(sum_yp, sum_yp);
|
|
|
|
vfloat4 prod_zp = dot(sum_zp, sum_zp);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 best_vector = sum_xp;
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 best_sum = prod_xp;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 mask = prod_yp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_yp, mask);
|
|
|
|
best_sum = select(best_sum, prod_yp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
mask = prod_zp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_zp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
pm[partition].dir = best_vector;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_avgs_and_dirs_2_comp(
|
|
|
|
const partition_info& pt,
|
|
|
|
const image_block& blk,
|
|
|
|
unsigned int component1,
|
|
|
|
unsigned int component2,
|
|
|
|
partition_metrics pm[BLOCK_MAX_PARTITIONS]
|
|
|
|
) {
|
2022-04-24 19:15:08 +08:00
|
|
|
float texel_weight;
|
|
|
|
vfloat4 average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
const float* data_vr = nullptr;
|
|
|
|
const float* data_vg = nullptr;
|
|
|
|
|
|
|
|
if (component1 == 0 && component2 == 1)
|
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<0, 1>()) / 2.0f;
|
|
|
|
average = blk.data_mean.swz<0, 1>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vr = blk.data_r;
|
|
|
|
data_vg = blk.data_g;
|
|
|
|
}
|
|
|
|
else if (component1 == 0 && component2 == 2)
|
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<0, 2>()) / 2.0f;
|
|
|
|
average = blk.data_mean.swz<0, 2>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vr = blk.data_r;
|
|
|
|
data_vg = blk.data_b;
|
|
|
|
}
|
|
|
|
else // (component1 == 1 && component2 == 2)
|
|
|
|
{
|
|
|
|
assert(component1 == 1 && component2 == 2);
|
2022-04-24 19:15:08 +08:00
|
|
|
|
|
|
|
texel_weight = hadd_s(blk.channel_weight.swz<1, 2>()) / 2.0f;
|
|
|
|
average = blk.data_mean.swz<1, 2>();
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
data_vr = blk.data_g;
|
|
|
|
data_vg = blk.data_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int partition_count = pt.partition_count;
|
|
|
|
promise(partition_count > 0);
|
|
|
|
|
|
|
|
for (unsigned int partition = 0; partition < partition_count; partition++)
|
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
const uint8_t *texel_indexes = pt.texels_of_partition[partition];
|
2021-06-01 23:43:28 +08:00
|
|
|
unsigned int texel_count = pt.partition_texel_count[partition];
|
|
|
|
promise(texel_count > 0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
// Only compute a partition mean if more than one partition
|
|
|
|
if (partition_count > 1)
|
2021-06-01 23:43:28 +08:00
|
|
|
{
|
2022-04-24 19:15:08 +08:00
|
|
|
average = vfloat4::zero();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
|
|
|
unsigned int iwt = texel_indexes[i];
|
|
|
|
average += vfloat2(data_vr[iwt], data_vg[iwt]);
|
|
|
|
}
|
|
|
|
|
|
|
|
average = average * (1.0f / static_cast<float>(texel_count));
|
2021-06-01 23:43:28 +08:00
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
pm[partition].avg = average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 sum_xp = vfloat4::zero();
|
|
|
|
vfloat4 sum_yp = vfloat4::zero();
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < texel_count; i++)
|
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
unsigned int iwt = texel_indexes[i];
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat4 texel_datum = vfloat2(data_vr[iwt], data_vg[iwt]);
|
2022-04-24 19:15:08 +08:00
|
|
|
texel_datum = texel_datum - average;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 zero = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm0 = texel_datum.swz<0,0,0,0>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_xp += select(zero, texel_datum, tdm0);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 tdm1 = texel_datum.swz<1,1,1,1>() > zero;
|
2021-06-01 23:43:28 +08:00
|
|
|
sum_yp += select(zero, texel_datum, tdm1);
|
|
|
|
}
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
sum_xp = sum_xp * texel_weight;
|
|
|
|
sum_yp = sum_yp * texel_weight;
|
|
|
|
|
|
|
|
vfloat4 prod_xp = dot(sum_xp, sum_xp);
|
|
|
|
vfloat4 prod_yp = dot(sum_yp, sum_yp);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat4 best_vector = sum_xp;
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat4 best_sum = prod_xp;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vmask4 mask = prod_yp > best_sum;
|
|
|
|
best_vector = select(best_vector, sum_yp, mask);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
pm[partition].dir = best_vector;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_error_squared_rgba(
|
|
|
|
const partition_info& pi,
|
|
|
|
const image_block& blk,
|
|
|
|
const processed_line4 uncor_plines[BLOCK_MAX_PARTITIONS],
|
|
|
|
const processed_line4 samec_plines[BLOCK_MAX_PARTITIONS],
|
|
|
|
float uncor_lengths[BLOCK_MAX_PARTITIONS],
|
|
|
|
float samec_lengths[BLOCK_MAX_PARTITIONS],
|
|
|
|
float& uncor_error,
|
|
|
|
float& samec_error
|
|
|
|
) {
|
|
|
|
unsigned int partition_count = pi.partition_count;
|
|
|
|
promise(partition_count > 0);
|
|
|
|
|
|
|
|
uncor_error = 0.0f;
|
|
|
|
samec_error = 0.0f;
|
|
|
|
|
|
|
|
for (unsigned int partition = 0; partition < partition_count; partition++)
|
|
|
|
{
|
2021-07-02 00:18:02 +08:00
|
|
|
const uint8_t *texel_indexes = pi.texels_of_partition[partition];
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
float uncor_loparam = 1e10f;
|
|
|
|
float uncor_hiparam = -1e10f;
|
|
|
|
|
|
|
|
float samec_loparam = 1e10f;
|
|
|
|
float samec_hiparam = -1e10f;
|
|
|
|
|
|
|
|
processed_line4 l_uncor = uncor_plines[partition];
|
|
|
|
processed_line4 l_samec = samec_plines[partition];
|
|
|
|
|
|
|
|
unsigned int texel_count = pi.partition_texel_count[partition];
|
|
|
|
promise(texel_count > 0);
|
|
|
|
|
|
|
|
// Vectorize some useful scalar inputs
|
|
|
|
vfloat l_uncor_bs0(l_uncor.bs.lane<0>());
|
|
|
|
vfloat l_uncor_bs1(l_uncor.bs.lane<1>());
|
|
|
|
vfloat l_uncor_bs2(l_uncor.bs.lane<2>());
|
|
|
|
vfloat l_uncor_bs3(l_uncor.bs.lane<3>());
|
|
|
|
|
|
|
|
vfloat l_uncor_amod0(l_uncor.amod.lane<0>());
|
|
|
|
vfloat l_uncor_amod1(l_uncor.amod.lane<1>());
|
|
|
|
vfloat l_uncor_amod2(l_uncor.amod.lane<2>());
|
|
|
|
vfloat l_uncor_amod3(l_uncor.amod.lane<3>());
|
|
|
|
|
|
|
|
vfloat l_samec_bs0(l_samec.bs.lane<0>());
|
|
|
|
vfloat l_samec_bs1(l_samec.bs.lane<1>());
|
|
|
|
vfloat l_samec_bs2(l_samec.bs.lane<2>());
|
|
|
|
vfloat l_samec_bs3(l_samec.bs.lane<3>());
|
|
|
|
|
|
|
|
assert(all(l_samec.amod == vfloat4(0.0f)));
|
|
|
|
|
|
|
|
vfloat uncor_loparamv(1e10f);
|
|
|
|
vfloat uncor_hiparamv(-1e10f);
|
|
|
|
vfloat4 uncor_errorsumv = vfloat4::zero();
|
|
|
|
|
|
|
|
vfloat samec_loparamv(1e10f);
|
|
|
|
vfloat samec_hiparamv(-1e10f);
|
|
|
|
vfloat4 samec_errorsumv = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat ew_r(blk.channel_weight.lane<0>());
|
|
|
|
vfloat ew_g(blk.channel_weight.lane<1>());
|
|
|
|
vfloat ew_b(blk.channel_weight.lane<2>());
|
|
|
|
vfloat ew_a(blk.channel_weight.lane<3>());
|
|
|
|
|
2021-07-02 00:18:02 +08:00
|
|
|
// This implementation over-shoots, but this is safe as we initialize the texel_indexes
|
|
|
|
// array to extend the last value. This means min/max are not impacted, but we need to mask
|
2021-06-01 23:43:28 +08:00
|
|
|
// out the dummy values when we compute the line weighting.
|
|
|
|
vint lane_ids = vint::lane_id();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i += ASTCENC_SIMD_WIDTH)
|
|
|
|
{
|
|
|
|
vmask mask = lane_ids < vint(texel_count);
|
2021-07-02 00:18:02 +08:00
|
|
|
vint texel_idxs(&(texel_indexes[i]));
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat data_r = gatherf(blk.data_r, texel_idxs);
|
|
|
|
vfloat data_g = gatherf(blk.data_g, texel_idxs);
|
|
|
|
vfloat data_b = gatherf(blk.data_b, texel_idxs);
|
|
|
|
vfloat data_a = gatherf(blk.data_a, texel_idxs);
|
|
|
|
|
|
|
|
vfloat uncor_param = (data_r * l_uncor_bs0)
|
|
|
|
+ (data_g * l_uncor_bs1)
|
|
|
|
+ (data_b * l_uncor_bs2)
|
|
|
|
+ (data_a * l_uncor_bs3);
|
|
|
|
|
|
|
|
uncor_loparamv = min(uncor_param, uncor_loparamv);
|
|
|
|
uncor_hiparamv = max(uncor_param, uncor_hiparamv);
|
|
|
|
|
|
|
|
vfloat uncor_dist0 = (l_uncor_amod0 - data_r)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs0);
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat uncor_dist1 = (l_uncor_amod1 - data_g)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs1);
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat uncor_dist2 = (l_uncor_amod2 - data_b)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs2);
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat uncor_dist3 = (l_uncor_amod3 - data_a)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs3);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat uncor_err = (ew_r * uncor_dist0 * uncor_dist0)
|
|
|
|
+ (ew_g * uncor_dist1 * uncor_dist1)
|
|
|
|
+ (ew_b * uncor_dist2 * uncor_dist2)
|
|
|
|
+ (ew_a * uncor_dist3 * uncor_dist3);
|
|
|
|
|
|
|
|
uncor_err = select(vfloat::zero(), uncor_err, mask);
|
|
|
|
haccumulate(uncor_errorsumv, uncor_err);
|
|
|
|
|
|
|
|
// Process samechroma data
|
|
|
|
vfloat samec_param = (data_r * l_samec_bs0)
|
|
|
|
+ (data_g * l_samec_bs1)
|
|
|
|
+ (data_b * l_samec_bs2)
|
|
|
|
+ (data_a * l_samec_bs3);
|
|
|
|
|
|
|
|
samec_loparamv = min(samec_param, samec_loparamv);
|
|
|
|
samec_hiparamv = max(samec_param, samec_hiparamv);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat samec_dist0 = samec_param * l_samec_bs0 - data_r;
|
|
|
|
vfloat samec_dist1 = samec_param * l_samec_bs1 - data_g;
|
|
|
|
vfloat samec_dist2 = samec_param * l_samec_bs2 - data_b;
|
|
|
|
vfloat samec_dist3 = samec_param * l_samec_bs3 - data_a;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat samec_err = (ew_r * samec_dist0 * samec_dist0)
|
|
|
|
+ (ew_g * samec_dist1 * samec_dist1)
|
|
|
|
+ (ew_b * samec_dist2 * samec_dist2)
|
|
|
|
+ (ew_a * samec_dist3 * samec_dist3);
|
|
|
|
|
|
|
|
samec_err = select(vfloat::zero(), samec_err, mask);
|
|
|
|
haccumulate(samec_errorsumv, samec_err);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
lane_ids += vint(ASTCENC_SIMD_WIDTH);
|
2021-06-01 23:43:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
uncor_loparam = hmin_s(uncor_loparamv);
|
|
|
|
uncor_hiparam = hmax_s(uncor_hiparamv);
|
|
|
|
|
|
|
|
samec_loparam = hmin_s(samec_loparamv);
|
|
|
|
samec_hiparam = hmax_s(samec_hiparamv);
|
|
|
|
|
|
|
|
// Resolve the final scalar accumulator sum
|
|
|
|
haccumulate(uncor_error, uncor_errorsumv);
|
|
|
|
haccumulate(samec_error, samec_errorsumv);
|
|
|
|
|
|
|
|
float uncor_linelen = uncor_hiparam - uncor_loparam;
|
|
|
|
float samec_linelen = samec_hiparam - samec_loparam;
|
|
|
|
|
|
|
|
// Turn very small numbers and NaNs into a small number
|
|
|
|
uncor_lengths[partition] = astc::max(uncor_linelen, 1e-7f);
|
|
|
|
samec_lengths[partition] = astc::max(samec_linelen, 1e-7f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See header for documentation. */
|
|
|
|
void compute_error_squared_rgb(
|
|
|
|
const partition_info& pi,
|
|
|
|
const image_block& blk,
|
|
|
|
partition_lines3 plines[BLOCK_MAX_PARTITIONS],
|
|
|
|
float& uncor_error,
|
|
|
|
float& samec_error
|
|
|
|
) {
|
|
|
|
unsigned int partition_count = pi.partition_count;
|
|
|
|
promise(partition_count > 0);
|
|
|
|
|
|
|
|
uncor_error = 0.0f;
|
|
|
|
samec_error = 0.0f;
|
|
|
|
|
|
|
|
for (unsigned int partition = 0; partition < partition_count; partition++)
|
|
|
|
{
|
|
|
|
partition_lines3& pl = plines[partition];
|
2021-07-02 00:18:02 +08:00
|
|
|
const uint8_t *texel_indexes = pi.texels_of_partition[partition];
|
2021-06-01 23:43:28 +08:00
|
|
|
unsigned int texel_count = pi.partition_texel_count[partition];
|
|
|
|
promise(texel_count > 0);
|
|
|
|
|
|
|
|
float uncor_loparam = 1e10f;
|
|
|
|
float uncor_hiparam = -1e10f;
|
|
|
|
|
|
|
|
float samec_loparam = 1e10f;
|
|
|
|
float samec_hiparam = -1e10f;
|
|
|
|
|
|
|
|
processed_line3 l_uncor = pl.uncor_pline;
|
|
|
|
processed_line3 l_samec = pl.samec_pline;
|
|
|
|
|
|
|
|
// This implementation is an example vectorization of this function.
|
|
|
|
// It works for - the codec is a 2-4% faster than not vectorizing - but
|
|
|
|
// the benefit is limited by the use of gathers and register pressure
|
|
|
|
|
|
|
|
// Vectorize some useful scalar inputs
|
|
|
|
vfloat l_uncor_bs0(l_uncor.bs.lane<0>());
|
|
|
|
vfloat l_uncor_bs1(l_uncor.bs.lane<1>());
|
|
|
|
vfloat l_uncor_bs2(l_uncor.bs.lane<2>());
|
|
|
|
|
|
|
|
vfloat l_uncor_amod0(l_uncor.amod.lane<0>());
|
|
|
|
vfloat l_uncor_amod1(l_uncor.amod.lane<1>());
|
|
|
|
vfloat l_uncor_amod2(l_uncor.amod.lane<2>());
|
|
|
|
|
|
|
|
vfloat l_samec_bs0(l_samec.bs.lane<0>());
|
|
|
|
vfloat l_samec_bs1(l_samec.bs.lane<1>());
|
|
|
|
vfloat l_samec_bs2(l_samec.bs.lane<2>());
|
|
|
|
|
|
|
|
assert(all(l_samec.amod == vfloat4(0.0f)));
|
|
|
|
|
|
|
|
vfloat uncor_loparamv(1e10f);
|
|
|
|
vfloat uncor_hiparamv(-1e10f);
|
|
|
|
vfloat4 uncor_errorsumv = vfloat4::zero();
|
|
|
|
|
|
|
|
vfloat samec_loparamv(1e10f);
|
|
|
|
vfloat samec_hiparamv(-1e10f);
|
|
|
|
vfloat4 samec_errorsumv = vfloat4::zero();
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat ew_r(blk.channel_weight.lane<0>());
|
|
|
|
vfloat ew_g(blk.channel_weight.lane<1>());
|
|
|
|
vfloat ew_b(blk.channel_weight.lane<2>());
|
|
|
|
|
2021-06-01 23:43:28 +08:00
|
|
|
// This implementation over-shoots, but this is safe as we initialize the weights array
|
|
|
|
// to extend the last value. This means min/max are not impacted, but we need to mask
|
|
|
|
// out the dummy values when we compute the line weighting.
|
|
|
|
vint lane_ids = vint::lane_id();
|
|
|
|
for (unsigned int i = 0; i < texel_count; i += ASTCENC_SIMD_WIDTH)
|
|
|
|
{
|
|
|
|
vmask mask = lane_ids < vint(texel_count);
|
2021-07-02 00:18:02 +08:00
|
|
|
vint texel_idxs(&(texel_indexes[i]));
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat data_r = gatherf(blk.data_r, texel_idxs);
|
|
|
|
vfloat data_g = gatherf(blk.data_g, texel_idxs);
|
|
|
|
vfloat data_b = gatherf(blk.data_b, texel_idxs);
|
|
|
|
|
|
|
|
vfloat uncor_param = (data_r * l_uncor_bs0)
|
|
|
|
+ (data_g * l_uncor_bs1)
|
|
|
|
+ (data_b * l_uncor_bs2);
|
|
|
|
|
|
|
|
uncor_loparamv = min(uncor_param, uncor_loparamv);
|
|
|
|
uncor_hiparamv = max(uncor_param, uncor_hiparamv);
|
|
|
|
|
|
|
|
vfloat uncor_dist0 = (l_uncor_amod0 - data_r)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs0);
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat uncor_dist1 = (l_uncor_amod1 - data_g)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs1);
|
2021-06-01 23:43:28 +08:00
|
|
|
vfloat uncor_dist2 = (l_uncor_amod2 - data_b)
|
2022-04-24 19:15:08 +08:00
|
|
|
+ (uncor_param * l_uncor_bs2);
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat uncor_err = (ew_r * uncor_dist0 * uncor_dist0)
|
|
|
|
+ (ew_g * uncor_dist1 * uncor_dist1)
|
|
|
|
+ (ew_b * uncor_dist2 * uncor_dist2);
|
|
|
|
|
|
|
|
uncor_err = select(vfloat::zero(), uncor_err, mask);
|
|
|
|
haccumulate(uncor_errorsumv, uncor_err);
|
|
|
|
|
|
|
|
// Process samechroma data
|
|
|
|
vfloat samec_param = (data_r * l_samec_bs0)
|
|
|
|
+ (data_g * l_samec_bs1)
|
|
|
|
+ (data_b * l_samec_bs2);
|
|
|
|
|
|
|
|
samec_loparamv = min(samec_param, samec_loparamv);
|
|
|
|
samec_hiparamv = max(samec_param, samec_hiparamv);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
vfloat samec_dist0 = samec_param * l_samec_bs0 - data_r;
|
|
|
|
vfloat samec_dist1 = samec_param * l_samec_bs1 - data_g;
|
|
|
|
vfloat samec_dist2 = samec_param * l_samec_bs2 - data_b;
|
2021-06-01 23:43:28 +08:00
|
|
|
|
|
|
|
vfloat samec_err = (ew_r * samec_dist0 * samec_dist0)
|
|
|
|
+ (ew_g * samec_dist1 * samec_dist1)
|
|
|
|
+ (ew_b * samec_dist2 * samec_dist2);
|
|
|
|
|
|
|
|
samec_err = select(vfloat::zero(), samec_err, mask);
|
|
|
|
haccumulate(samec_errorsumv, samec_err);
|
|
|
|
|
2022-04-24 19:15:08 +08:00
|
|
|
lane_ids += vint(ASTCENC_SIMD_WIDTH);
|
2021-06-01 23:43:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
uncor_loparam = hmin_s(uncor_loparamv);
|
|
|
|
uncor_hiparam = hmax_s(uncor_hiparamv);
|
|
|
|
|
|
|
|
samec_loparam = hmin_s(samec_loparamv);
|
|
|
|
samec_hiparam = hmax_s(samec_hiparamv);
|
|
|
|
|
|
|
|
// Resolve the final scalar accumulator sum
|
|
|
|
haccumulate(uncor_error, uncor_errorsumv);
|
|
|
|
haccumulate(samec_error, samec_errorsumv);
|
|
|
|
|
|
|
|
float uncor_linelen = uncor_hiparam - uncor_loparam;
|
|
|
|
float samec_linelen = samec_hiparam - samec_loparam;
|
|
|
|
|
|
|
|
// Turn very small numbers and NaNs into a small number
|
|
|
|
pl.uncor_line_len = astc::max(uncor_linelen, 1e-7f);
|
|
|
|
pl.samec_line_len = astc::max(samec_linelen, 1e-7f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|