mirror of
https://github.com/RetroDECK/Duckstation.git
synced 2025-01-23 00:35:37 +00:00
624 lines
19 KiB
C++
624 lines
19 KiB
C++
/*
|
|
* Copyright (C) 2014 Patrick Mours
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
#include "effect_lexer.hpp"
|
|
#include "effect_codegen.hpp"
|
|
#include <cmath> // fmod
|
|
#include <cassert>
|
|
#include <cstring> // memcpy, memset
|
|
#include <algorithm> // std::min, std::max
|
|
|
|
reshadefx::type reshadefx::type::merge(const type &lhs, const type &rhs)
|
|
{
|
|
type result = { std::max(lhs.base, rhs.base) };
|
|
|
|
// Non-numeric types cannot be vectors or matrices
|
|
if (!result.is_numeric())
|
|
{
|
|
result.rows = 0;
|
|
result.cols = 0;
|
|
}
|
|
// If one side of the expression is scalar, it needs to be promoted to the same dimension as the other side
|
|
else if ((lhs.rows == 1 && lhs.cols == 1) || (rhs.rows == 1 && rhs.cols == 1))
|
|
{
|
|
result.rows = std::max(lhs.rows, rhs.rows);
|
|
result.cols = std::max(lhs.cols, rhs.cols);
|
|
}
|
|
else // Otherwise dimensions match or one side is truncated to match the other one
|
|
{
|
|
result.rows = std::min(lhs.rows, rhs.rows);
|
|
result.cols = std::min(lhs.cols, rhs.cols);
|
|
}
|
|
|
|
// Some qualifiers propagate to the result
|
|
result.qualifiers = (lhs.qualifiers & type::q_precise) | (rhs.qualifiers & type::q_precise);
|
|
|
|
// In case this is a structure, assume they are the same
|
|
result.definition = rhs.definition;
|
|
assert(lhs.definition == rhs.definition || lhs.definition == 0);
|
|
assert(lhs.array_length == 0 && rhs.array_length == 0);
|
|
|
|
return result;
|
|
}
|
|
|
|
std::string reshadefx::type::description() const
|
|
{
|
|
std::string result;
|
|
switch (base)
|
|
{
|
|
case reshadefx::type::t_void:
|
|
result = "void";
|
|
break;
|
|
case reshadefx::type::t_bool:
|
|
result = "bool";
|
|
break;
|
|
case reshadefx::type::t_min16int:
|
|
result = "min16int";
|
|
break;
|
|
case reshadefx::type::t_int:
|
|
result = "int";
|
|
break;
|
|
case reshadefx::type::t_min16uint:
|
|
result = "min16uint";
|
|
break;
|
|
case reshadefx::type::t_uint:
|
|
result = "uint";
|
|
break;
|
|
case reshadefx::type::t_min16float:
|
|
result = "min16float";
|
|
break;
|
|
case reshadefx::type::t_float:
|
|
result = "float";
|
|
break;
|
|
case reshadefx::type::t_string:
|
|
result = "string";
|
|
break;
|
|
case reshadefx::type::t_struct:
|
|
result = "struct";
|
|
break;
|
|
case reshadefx::type::t_texture1d:
|
|
result = "texture1D";
|
|
break;
|
|
case reshadefx::type::t_texture2d:
|
|
result = "texture2D";
|
|
break;
|
|
case reshadefx::type::t_texture3d:
|
|
result = "texture3D";
|
|
break;
|
|
case reshadefx::type::t_sampler1d_int:
|
|
result = "sampler1D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler2d_int:
|
|
result = "sampler2D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler3d_int:
|
|
result = "sampler3D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler1d_uint:
|
|
result = "sampler1D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler2d_uint:
|
|
result = "sampler2D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler3d_uint:
|
|
result = "sampler3D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler1d_float:
|
|
result = "sampler1D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler2d_float:
|
|
result = "sampler2D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_sampler3d_float:
|
|
result = "sampler3D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage1d_int:
|
|
result = "storage1D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage2d_int:
|
|
result = "storage2D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage3d_int:
|
|
result = "storage3D<int" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage1d_uint:
|
|
result = "storage1D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage2d_uint:
|
|
result = "storage2D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage3d_uint:
|
|
result = "storage3D<uint" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage1d_float:
|
|
result = "storage1D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage2d_float:
|
|
result = "storage2D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_storage3d_float:
|
|
result = "storage3D<float" + std::to_string(rows) + '>';
|
|
break;
|
|
case reshadefx::type::t_function:
|
|
result = "function";
|
|
break;
|
|
}
|
|
|
|
if (is_numeric())
|
|
{
|
|
if (rows > 1 || cols > 1)
|
|
result += std::to_string(rows);
|
|
if (cols > 1)
|
|
result += 'x' + std::to_string(cols);
|
|
}
|
|
|
|
if (is_array())
|
|
{
|
|
result += '[';
|
|
if (array_length > 0)
|
|
result += std::to_string(array_length);
|
|
result += ']';
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void reshadefx::expression::reset_to_lvalue(const reshadefx::location &loc, uint32_t in_base, const reshadefx::type &in_type)
|
|
{
|
|
type = in_type;
|
|
base = in_base;
|
|
location = loc;
|
|
is_lvalue = true;
|
|
is_constant = false;
|
|
chain.clear();
|
|
|
|
// Make sure uniform l-values cannot be assigned to by making them constant
|
|
if (in_type.has(type::q_uniform))
|
|
type.qualifiers |= type::q_const;
|
|
|
|
// Strip away global variable qualifiers
|
|
type.qualifiers &= ~(reshadefx::type::q_extern | reshadefx::type::q_static | reshadefx::type::q_uniform | reshadefx::type::q_groupshared);
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue(const reshadefx::location &loc, uint32_t in_base, const reshadefx::type &in_type)
|
|
{
|
|
type = in_type;
|
|
type.qualifiers |= type::q_const;
|
|
base = in_base;
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = false;
|
|
chain.clear();
|
|
|
|
// Strip away global variable qualifiers
|
|
type.qualifiers &= ~(reshadefx::type::q_extern | reshadefx::type::q_static | reshadefx::type::q_uniform | reshadefx::type::q_groupshared);
|
|
}
|
|
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, bool data)
|
|
{
|
|
type = { type::t_bool, 1, 1, type::q_const };
|
|
base = 0; constant = {}; constant.as_uint[0] = data;
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, float data)
|
|
{
|
|
type = { type::t_float, 1, 1, type::q_const };
|
|
base = 0; constant = {}; constant.as_float[0] = data;
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, int32_t data)
|
|
{
|
|
type = { type::t_int, 1, 1, type::q_const };
|
|
base = 0; constant = {}; constant.as_int[0] = data;
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, uint32_t data)
|
|
{
|
|
type = { type::t_uint, 1, 1, type::q_const };
|
|
base = 0; constant = {}; constant.as_uint[0] = data;
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, std::string data)
|
|
{
|
|
type = { type::t_string, 0, 0, type::q_const };
|
|
base = 0; constant = {}; constant.string_data = std::move(data);
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
void reshadefx::expression::reset_to_rvalue_constant(const reshadefx::location &loc, reshadefx::constant data, const reshadefx::type &in_type)
|
|
{
|
|
type = in_type;
|
|
type.qualifiers |= type::q_const;
|
|
base = 0; constant = std::move(data);
|
|
location = loc;
|
|
is_lvalue = false;
|
|
is_constant = true;
|
|
chain.clear();
|
|
}
|
|
|
|
void reshadefx::expression::add_cast_operation(const reshadefx::type &cast_type)
|
|
{
|
|
// First try to simplify the cast with a swizzle operation (only works with scalars and vectors)
|
|
if (type.cols == 1 && cast_type.cols == 1 && type.rows != cast_type.rows)
|
|
{
|
|
signed char swizzle[] = { 0, 1, 2, 3 };
|
|
// Ignore components in a demotion cast
|
|
for (unsigned int i = cast_type.rows; i < 4; ++i)
|
|
swizzle[i] = -1;
|
|
// Use the last component to fill in a promotion cast
|
|
for (unsigned int i = type.rows; i < cast_type.rows; ++i)
|
|
swizzle[i] = swizzle[type.rows - 1];
|
|
|
|
add_swizzle_access(swizzle, cast_type.rows);
|
|
}
|
|
|
|
if (type == cast_type)
|
|
return; // There is nothing more to do if the expression is already of the target type at this point
|
|
|
|
if (is_constant)
|
|
{
|
|
const auto cast_constant = [](reshadefx::constant &constant, const reshadefx::type &from, const reshadefx::type &to) {
|
|
// Handle scalar to vector promotion first
|
|
if (from.is_scalar() && !to.is_scalar())
|
|
for (unsigned int i = 1; i < to.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[0];
|
|
|
|
// Next check whether the type needs casting as well (and don't convert between signed/unsigned, since that is handled by the union)
|
|
if (from.base == to.base || from.is_floating_point() == to.is_floating_point())
|
|
return;
|
|
|
|
if (!to.is_floating_point())
|
|
for (unsigned int i = 0; i < to.components(); ++i)
|
|
constant.as_uint[i] = static_cast<int>(constant.as_float[i]);
|
|
else
|
|
for (unsigned int i = 0; i < to.components(); ++i)
|
|
constant.as_float[i] = static_cast<float>(constant.as_int[i]);
|
|
};
|
|
|
|
for (auto &element : constant.array_data)
|
|
cast_constant(element, type, cast_type);
|
|
|
|
cast_constant(constant, type, cast_type);
|
|
}
|
|
else
|
|
{
|
|
assert(!type.is_array() && !cast_type.is_array());
|
|
|
|
chain.push_back({ operation::op_cast, type, cast_type });
|
|
}
|
|
|
|
type = cast_type;
|
|
type.qualifiers |= type::q_const; // Casting always makes expression not modifiable
|
|
}
|
|
void reshadefx::expression::add_member_access(unsigned int index, const reshadefx::type &in_type)
|
|
{
|
|
assert(type.is_struct());
|
|
|
|
chain.push_back({ operation::op_member, type, in_type, index });
|
|
|
|
// The type is now the type of the member that was accessed
|
|
type = in_type;
|
|
is_constant = false;
|
|
}
|
|
void reshadefx::expression::add_dynamic_index_access(uint32_t index_expression)
|
|
{
|
|
assert(!is_constant); // Cannot have dynamic indexing into constant in SPIR-V
|
|
assert(type.is_array() || (type.is_numeric() && !type.is_scalar()));
|
|
|
|
auto prev_type = type;
|
|
|
|
if (type.is_array())
|
|
{
|
|
type.array_length = 0;
|
|
}
|
|
else if (type.is_matrix())
|
|
{
|
|
type.rows = type.cols;
|
|
type.cols = 1;
|
|
}
|
|
else if (type.is_vector())
|
|
{
|
|
type.rows = 1;
|
|
}
|
|
|
|
chain.push_back({ operation::op_dynamic_index, prev_type, type, index_expression });
|
|
}
|
|
void reshadefx::expression::add_constant_index_access(unsigned int index)
|
|
{
|
|
assert(type.is_array() || (type.is_numeric() && !type.is_scalar()));
|
|
|
|
auto prev_type = type;
|
|
|
|
if (type.is_array())
|
|
{
|
|
assert(type.array_length < 0 || index < static_cast<unsigned int>(type.array_length));
|
|
|
|
type.array_length = 0;
|
|
}
|
|
else if (type.is_matrix())
|
|
{
|
|
assert(index < type.components());
|
|
|
|
type.rows = type.cols;
|
|
type.cols = 1;
|
|
}
|
|
else if (type.is_vector())
|
|
{
|
|
assert(index < type.components());
|
|
|
|
type.rows = 1;
|
|
}
|
|
|
|
if (is_constant)
|
|
{
|
|
if (prev_type.is_array())
|
|
{
|
|
constant = constant.array_data[index];
|
|
}
|
|
else if (prev_type.is_matrix()) // Indexing into a matrix returns a row of it as a vector
|
|
{
|
|
for (unsigned int i = 0; i < prev_type.cols; ++i)
|
|
constant.as_uint[i] = constant.as_uint[index * prev_type.cols + i];
|
|
}
|
|
else // Indexing into a vector returns the element as a scalar
|
|
{
|
|
constant.as_uint[0] = constant.as_uint[index];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
chain.push_back({ operation::op_constant_index, prev_type, type, index });
|
|
}
|
|
}
|
|
void reshadefx::expression::add_swizzle_access(const signed char swizzle[4], unsigned int length)
|
|
{
|
|
assert(type.is_numeric() && !type.is_array());
|
|
|
|
const auto prev_type = type;
|
|
|
|
type.rows = length;
|
|
type.cols = 1;
|
|
|
|
if (is_constant)
|
|
{
|
|
assert(constant.array_data.empty());
|
|
|
|
uint32_t data[16];
|
|
std::memcpy(data, &constant.as_uint[0], sizeof(data));
|
|
for (unsigned int i = 0; i < length; ++i)
|
|
constant.as_uint[i] = data[swizzle[i]];
|
|
std::memset(&constant.as_uint[length], 0, sizeof(uint32_t) * (16 - length)); // Clear the rest of the constant
|
|
}
|
|
else if (length == 1 && prev_type.is_vector()) // Use indexing when possible since the code generation logic is simpler in SPIR-V
|
|
{
|
|
chain.push_back({ operation::op_constant_index, prev_type, type, static_cast<uint32_t>(swizzle[0]) });
|
|
}
|
|
else
|
|
{
|
|
chain.push_back({ operation::op_swizzle, prev_type, type, 0, { swizzle[0], swizzle[1], swizzle[2], swizzle[3] } });
|
|
}
|
|
}
|
|
|
|
bool reshadefx::expression::evaluate_constant_expression(reshadefx::tokenid op)
|
|
{
|
|
if (!is_constant)
|
|
return false;
|
|
|
|
switch (op)
|
|
{
|
|
case tokenid::exclaim:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = !constant.as_uint[i];
|
|
break;
|
|
case tokenid::minus:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_float[i] = -constant.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_int[i] = -constant.as_int[i];
|
|
break;
|
|
case tokenid::tilde:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = ~constant.as_uint[i];
|
|
break;
|
|
default:
|
|
// Unknown operator token, so nothing to do
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
bool reshadefx::expression::evaluate_constant_expression(reshadefx::tokenid op, const reshadefx::constant &rhs)
|
|
{
|
|
if (!is_constant)
|
|
return false;
|
|
|
|
switch (op)
|
|
{
|
|
case tokenid::percent:
|
|
if (type.is_floating_point()) {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
// Floating point modulo with zero is defined and results in NaN
|
|
if (rhs.as_float[i] == 0)
|
|
constant.as_float[i] = std::numeric_limits<float>::quiet_NaN();
|
|
else
|
|
constant.as_float[i] = std::fmod(constant.as_float[i], rhs.as_float[i]);
|
|
}
|
|
else if (type.is_signed()) {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
// Integer modulo with zero on the other hand is not defined, so do not fold this expression in that case
|
|
if (rhs.as_int[i] == 0)
|
|
return false;
|
|
else
|
|
constant.as_int[i] %= rhs.as_int[i];
|
|
}
|
|
else {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
if (rhs.as_uint[i] == 0)
|
|
return false;
|
|
else
|
|
constant.as_uint[i] %= rhs.as_uint[i];
|
|
}
|
|
break;
|
|
case tokenid::star:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_float[i] *= rhs.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] *= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::plus:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_float[i] += rhs.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] += rhs.as_uint[i];
|
|
break;
|
|
case tokenid::minus:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_float[i] -= rhs.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] -= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::slash:
|
|
if (type.is_floating_point()) {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
// Floating point division by zero is well defined and results in infinity or NaN
|
|
constant.as_float[i] /= rhs.as_float[i];
|
|
}
|
|
else if (type.is_signed()) {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
// Integer division by zero on the other hand is not defined, so do not fold this expression in that case
|
|
if (rhs.as_int[i] == 0)
|
|
return false;
|
|
else
|
|
constant.as_int[i] /= rhs.as_int[i];
|
|
}
|
|
else {
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
if (rhs.as_uint[i] == 0)
|
|
return false;
|
|
else
|
|
constant.as_uint[i] /= rhs.as_uint[i];
|
|
}
|
|
break;
|
|
case tokenid::ampersand:
|
|
case tokenid::ampersand_ampersand:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] &= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::pipe:
|
|
case tokenid::pipe_pipe:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] |= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::caret:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] ^= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::less:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] < rhs.as_float[i];
|
|
else if (type.is_signed())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_int[i] < rhs.as_int[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] < rhs.as_uint[i];
|
|
type.base = type::t_bool; // Logic operations change the type to boolean
|
|
break;
|
|
case tokenid::less_equal:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] <= rhs.as_float[i];
|
|
else if (type.is_signed())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_int[i] <= rhs.as_int[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] <= rhs.as_uint[i];
|
|
type.base = type::t_bool;
|
|
break;
|
|
case tokenid::greater:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] > rhs.as_float[i];
|
|
else if (type.is_signed())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_int[i] > rhs.as_int[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] > rhs.as_uint[i];
|
|
type.base = type::t_bool;
|
|
break;
|
|
case tokenid::greater_equal:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] >= rhs.as_float[i];
|
|
else if (type.is_signed())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_int[i] >= rhs.as_int[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] >= rhs.as_uint[i];
|
|
type.base = type::t_bool;
|
|
break;
|
|
case tokenid::equal_equal:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] == rhs.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] == rhs.as_uint[i];
|
|
type.base = type::t_bool;
|
|
break;
|
|
case tokenid::exclaim_equal:
|
|
if (type.is_floating_point())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_float[i] != rhs.as_float[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] = constant.as_uint[i] != rhs.as_uint[i];
|
|
type.base = type::t_bool;
|
|
break;
|
|
case tokenid::less_less:
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] <<= rhs.as_uint[i];
|
|
break;
|
|
case tokenid::greater_greater:
|
|
if (type.is_signed())
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_int[i] >>= rhs.as_int[i];
|
|
else
|
|
for (unsigned int i = 0; i < type.components(); ++i)
|
|
constant.as_uint[i] >>= rhs.as_uint[i];
|
|
break;
|
|
default:
|
|
// Unknown operator token, so nothing to do
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|