blob: 43627f7c0bacb2d8ff22a3fa0f8503e6940fab7c [file] [log] [blame]
/**************************************************************************
*
* Copyright 2010 Luca Barbieri
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice (including the
* next paragraph) shall be included in all copies or substantial
* portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE COPYRIGHT OWNER(S) AND/OR ITS SUPPLIERS BE
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
**************************************************************************/
/* Header for Shader Model 4.0, 4.1 and 5.0 */
#ifndef SM4_H_
#define SM4_H_
#include <stdint.h>
#include <string.h>
#include <stdlib.h>
#include <memory>
#include <vector>
#include <map>
#include <iostream>
#include "le32.h"
#include "sm4_defs.h"
extern const char* sm4_opcode_names[];
extern const char* sm4_file_names[];
extern const char* sm4_shortfile_names[];
extern const char* sm4_target_names[];
extern const char* sm4_interpolation_names[];
extern const char* sm4_sv_names[];
struct sm4_token_version
{
unsigned minor : 4;
unsigned major : 4;
unsigned format : 8;
unsigned type : 16;
};
struct sm4_token_instruction
{
// we don't make it an union directly because unions can't be inherited from
union
{
// length and extended are always present, but they are only here to reduce duplication
struct
{
unsigned opcode : 11;
unsigned _11_23 : 13;
unsigned length : 7;
unsigned extended : 1;
};
struct
{
unsigned opcode : 11;
unsigned resinfo_return_type : 2;
unsigned sat : 1;
unsigned _14_17 : 4;
unsigned test_nz : 1; // bit 18
unsigned precise_mask : 4;
unsigned _23 : 1;
unsigned length : 7;
unsigned extended : 1;
} insn;
struct
{
unsigned opcode : 11;
unsigned threads_in_group : 1;
unsigned shared_memory : 1;
unsigned uav_group : 1;
unsigned uav_global : 1;
unsigned _15_17 : 3;
} sync;
struct
{
unsigned opcode : 11;
unsigned allow_refactoring : 1;
unsigned fp64 : 1;
unsigned early_depth_stencil : 1;
unsigned enable_raw_and_structured_in_non_cs : 1;
} dcl_global_flags;
struct
{
unsigned opcode : 11;
unsigned target : 5;
unsigned nr_samples : 7;
} dcl_resource;
struct
{
unsigned opcode : 11;
unsigned shadow : 1;
unsigned mono : 1;
} dcl_sampler;
struct
{
unsigned opcode : 11;
unsigned interpolation : 5;
} dcl_input_ps;
struct
{
unsigned opcode : 11;
unsigned dynamic : 1;
} dcl_constant_buffer;
struct
{
unsigned opcode : 11;
unsigned primitive : 6;
} dcl_gs_input_primitive;
struct
{
unsigned opcode : 11;
unsigned primitive_topology : 7;
} dcl_gs_output_primitive_topology;
struct
{
unsigned opcode : 11;
unsigned control_points : 6;
} dcl_input_control_point_count;
struct
{
unsigned opcode : 11;
unsigned control_points : 6;
} dcl_output_control_point_count;
struct
{
unsigned opcode : 11;
unsigned domain : 3; /* D3D_TESSELLATOR_DOMAIN */
} dcl_tess_domain;
struct
{
unsigned opcode : 11;
unsigned partitioning : 3; /* D3D_TESSELLATOR_PARTITIONING */
} dcl_tess_partitioning;
struct
{
unsigned opcode : 11;
unsigned primitive : 3; /* D3D_TESSELLATOR_OUTPUT_PRIMITIVE */
} dcl_tess_output_primitive;
};
};
union sm4_token_instruction_extended
{
struct
{
unsigned type : 6;
unsigned _6_30 : 25;
unsigned extended :1;
};
struct
{
unsigned type : 6;
unsigned _6_8 : 3;
int offset_u : 4;
int offset_v : 4;
int offset_w : 4;
} sample_controls;
struct
{
unsigned type : 6;
unsigned target : 5;
} resource_target;
struct
{
unsigned type : 6;
unsigned x : 4;
unsigned y : 4;
unsigned z : 4;
unsigned w : 4;
} resource_return_type;
};
struct sm4_token_resource_return_type
{
unsigned x : 4;
unsigned y : 4;
unsigned z : 4;
unsigned w : 4;
};
struct sm4_token_operand
{
unsigned comps_enum : 2; /* sm4_operands_comps */
unsigned mode : 2; /* sm4_operand_mode */
unsigned sel : 8;
unsigned file : 8; /* sm4_file */
unsigned num_indices : 2;
unsigned index0_repr : 3; /* sm4_operand_index_repr */
unsigned index1_repr : 3; /* sm4_operand_index_repr */
unsigned index2_repr : 3; /* sm4_operand_index_repr */
unsigned extended : 1;
};
#define SM4_OPERAND_SEL_MASK(sel) ((sel) & 0xf)
#define SM4_OPERAND_SEL_SWZ(sel, i) (((sel) >> ((i) * 2)) & 3)
#define SM4_OPERAND_SEL_SCALAR(sel) ((sel) & 3)
struct sm4_token_operand_extended
{
unsigned type : 6;
unsigned neg : 1;
unsigned abs : 1;
};
union sm4_any
{
double f64;
float f32;
int64_t i64;
int32_t i32;
uint64_t u64;
int64_t u32;
};
struct sm4_op;
struct sm4_insn;
struct sm4_dcl;
struct sm4_program;
std::ostream& operator <<(std::ostream& out, const sm4_op& op);
std::ostream& operator <<(std::ostream& out, const sm4_insn& op);
std::ostream& operator <<(std::ostream& out, const sm4_dcl& op);
std::ostream& operator <<(std::ostream& out, const sm4_program& op);
struct sm4_op
{
uint8_t mode;
uint8_t comps;
uint8_t mask;
uint8_t num_indices;
uint8_t swizzle[4];
sm4_file file;
sm4_any imm_values[4];
bool neg;
bool abs;
struct
{
int64_t disp;
std::auto_ptr<sm4_op> reg;
} indices[3];
bool is_index_simple(unsigned i) const
{
return !indices[i].reg.get() && indices[i].disp >= 0 && (int64_t)(int32_t)indices[i].disp == indices[i].disp;
}
bool has_simple_index() const
{
return num_indices == 1 && is_index_simple(0);
}
sm4_op()
{
memset(this, 0, sizeof(*this));
}
void dump();
private:
sm4_op(const sm4_op& op)
{}
};
/* for sample_d */
#define SM4_MAX_OPS 6
struct sm4_insn : public sm4_token_instruction
{
int8_t sample_offset[3];
uint8_t resource_target;
uint8_t resource_return_type[4];
unsigned num;
unsigned num_ops;
std::auto_ptr<sm4_op> ops[SM4_MAX_OPS];
sm4_insn()
{
memset(this, 0, sizeof(*this));
}
void dump();
private:
sm4_insn(const sm4_insn& op)
{}
};
struct sm4_dcl : public sm4_token_instruction
{
std::auto_ptr<sm4_op> op;
union
{
unsigned num;
float f32;
sm4_sv sv;
struct
{
unsigned id;
unsigned expected_function_table_length;
unsigned table_length;
unsigned array_length;
} intf;
unsigned thread_group_size[3];
sm4_token_resource_return_type rrt;
struct
{
unsigned num;
unsigned comps;
} indexable_temp;
struct
{
unsigned stride;
unsigned count;
} structured;
};
void* data;
sm4_dcl()
{
memset(this, 0, sizeof(*this));
}
~sm4_dcl()
{
free(data);
}
void dump();
private:
sm4_dcl(const sm4_dcl& op)
{}
};
struct _D3D11_SIGNATURE_PARAMETER_DESC;
struct sm4_program
{
sm4_token_version version;
std::vector<sm4_dcl*> dcls;
std::vector<sm4_insn*> insns;
_D3D11_SIGNATURE_PARAMETER_DESC* params_in;
_D3D11_SIGNATURE_PARAMETER_DESC* params_out;
_D3D11_SIGNATURE_PARAMETER_DESC* params_patch;
unsigned num_params_in;
unsigned num_params_out;
unsigned num_params_patch;
/* for ifs, the insn number of the else or endif if there is no else
* for elses, the insn number of the endif
* for endifs, the insn number of the if
* for loops, the insn number of the endloop
* for endloops, the insn number of the loop
* for all others, -1
*/
std::vector<int> cf_insn_linked;
bool labels_found;
std::vector<int> label_to_insn_num;
sm4_program()
{
memset(&version, 0, sizeof(version));
labels_found = false;
num_params_in = num_params_out = num_params_patch = 0;
}
~sm4_program()
{
for(std::vector<sm4_dcl*>::iterator i = dcls.begin(), e = dcls.end(); i != e; ++i)
delete *i;
for(std::vector<sm4_insn*>::iterator i = insns.begin(), e = insns.end(); i != e; ++i)
delete *i;
if(num_params_in)
free(params_in);
if(num_params_out)
free(params_out);
if(num_params_patch)
free(params_patch);
}
void dump();
private:
sm4_program(const sm4_dcl& op)
{}
};
sm4_program* sm4_parse(void* tokens, int size);
bool sm4_link_cf_insns(sm4_program& program);
bool sm4_find_labels(sm4_program& program);
#endif /* SM4_H_ */