// -*- mode:c++ -*- // Copyright (c) 2009 The University of Edinburgh // 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 the copyright holders 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. // // Authors: Timothy M. Jones //////////////////////////////////////////////////////////////////// // // Integer ALU instructions // // Instruction class constructor template when Rc is set. def template IntRcConstructor {{ %(class_name)s::%(class_name)s(ExtMachInst machInst) : %(base_class)s("%(mnemonic)s", machInst, %(op_class)s) { %(constructor)s; rcSet = true; } }}; // Instruction class constructor template when OE is set. def template IntOeConstructor {{ %(class_name)s::%(class_name)s(ExtMachInst machInst) : %(base_class)s("%(mnemonic)s", machInst, %(op_class)s) { %(constructor)s; oeSet = true; } }}; // Instruction class constructor template when both Rc and OE are set. def template IntRcOeConstructor {{ %(class_name)s::%(class_name)s(ExtMachInst machInst) : %(base_class)s("%(mnemonic)s", machInst, %(op_class)s) { %(constructor)s; rcSet = true; oeSet = true; } }}; let {{ readXERCode = 'Xer xer = XER;' setXERCode = 'XER = xer;' computeCR0Code = ''' Cr cr = CR; cr.cr0 = makeCRField((int32_t)%(result)s, (int32_t)0, xer.so); CR = cr; ''' computeCACode = ''' if (findCarry(32, %(result)s, %(inputa)s, %(inputb)s)) { xer.ca = 1; } else { xer.ca = 0; } ''' computeOVCode = ''' if (findOverflow(32, %(result)s, %(inputa)s, %(inputb)s)) { xer.ov = 1; xer.so = 1; } else { xer.ov = 0; } ''' computeDivOVCode = ''' if (divSetOV) { xer.ov = 1; xer.so = 1; } else { if (findOverflow(32, %(result)s, %(inputa)s, %(inputb)s)) { xer.ov = 1; xer.so = 1; } else { xer.ov = 0; } } ''' }}; // A basic integer instruction. def format IntOp(code, inst_flags = []) {{ (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntOp', code, inst_flags, BasicDecode, BasicConstructor) }}; // Integer instructions with immediate (signed or unsigned). def format IntImmOp(code, inst_flags = []) {{ (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntImmOp', code, inst_flags, BasicDecode, BasicConstructor) }}; // Integer instructions with immediate that perform arithmetic. // These instructions all write to Rt and use an altered form of the // value in source register Ra, hence the use of src to hold the actual // value. The control flags include the use of code to compute the // carry bit or the CR0 code. def format IntImmArithOp(code, ctrl_flags = [], inst_flags = []) {{ # Set up the dictionary and deal with control flags dict = {'result':'Rt', 'inputa':'src', 'inputb':'imm'} if ctrl_flags: code += readXERCode for val in ctrl_flags: if val == 'computeCA': code += computeCACode % dict + setXERCode elif val == 'computeCR0': code += computeCR0Code % dict # Generate the class (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntImmOp', code, inst_flags, BasicDecode, BasicConstructor) }}; // Integer instructions with immediate that perform arithmetic but use // the value 0 when Ra == 0. We generate two versions of each instruction // corresponding to these two different scenarios. The correct version is // determined at decode (see the CheckRaDecode template). def format IntImmArithCheckRaOp(code, code_ra0, inst_flags = []) {{ # First the version where Ra is non-zero (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntImmOp', code, inst_flags, CheckRaDecode, BasicConstructor) # Now another version where Ra == 0 (header_output_ra0, decoder_output_ra0, _, exec_output_ra0) = \ GenAluOp(name, Name + 'RaZero', 'IntImmOp', code_ra0, inst_flags, CheckRaDecode, BasicConstructor) # Finally, add to the other outputs header_output += header_output_ra0 decoder_output += decoder_output_ra0 exec_output += exec_output_ra0 }}; // Integer instructions with immediate that perform logic operations. // All instructions write to Ra and use Rs as a source register. Some // also compute the CR0 code too. def format IntImmLogicOp(code, computeCR0 = 0, inst_flags = []) {{ # Set up the dictionary and deal with computing CR0 dict = {'result':'Ra'} if computeCR0: code += readXERCode + computeCR0Code % dict # Generate the class (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntImmOp', code, inst_flags, BasicDecode, BasicConstructor) }}; // Integer instructions that perform logic operations. The result is // always written into Ra. All instructions have 2 versions depending on // whether the Rc bit is set to compute the CR0 code. This is determined // at decode as before. def format IntLogicOp(code, inst_flags = []) {{ dict = {'result':'Ra'} # Code when Rc is set code_rc1 = code + readXERCode + computeCR0Code % dict # Generate the first class (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntOp', code, inst_flags, CheckRcDecode, BasicConstructor) # Generate the second class (header_output_rc1, decoder_output_rc1, _, exec_output_rc1) = \ GenAluOp(name, Name + 'RcSet', 'IntOp', code_rc1, inst_flags, CheckRcDecode, IntRcConstructor) # Finally, add to the other outputs header_output += header_output_rc1 decoder_output += decoder_output_rc1 exec_output += exec_output_rc1 }}; // Integer instructions with a shift amount. As above, except inheriting // from the IntShiftOp class. def format IntShiftOp(code, inst_flags = []) {{ dict = {'result':'Ra'} # Code when Rc is set code_rc1 = code + readXERCode + computeCR0Code % dict # Generate the first class (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntShiftOp', code, inst_flags, CheckRcDecode, BasicConstructor) # Generate the second class (header_output_rc1, decoder_output_rc1, _, exec_output_rc1) = \ GenAluOp(name, Name + 'RcSet', 'IntShiftOp', code_rc1, inst_flags, CheckRcDecode, IntRcConstructor) # Finally, add to the other outputs header_output += header_output_rc1 decoder_output += decoder_output_rc1 exec_output += exec_output_rc1 }}; // Instructions in this format are all reduced to the form Rt = src1 + src2, // therefore we just give src1 and src2 definitions. In working out the // template we first put in the definitions of the variables and then // the code for the addition. We also deal with computing the carry flag // if required. // // We generate 4 versions of each instruction. This correspond to the // different combinations of having the OE bit set or unset (which controls // whether the overflow flag is computed) and the Rc bit set or unset too // (which controls whether the CR0 code is computed). def format IntSumOp(src1, src2, ca = {{ 0 }}, computeCA = 0, inst_flags = []) {{ # The result is always in Rt, but the source values vary dict = {'result':'Rt', 'inputa':'src1', 'inputb':'src2'} # Add code to set up variables and do the sum code = 'uint32_t src1 = ' + src1 + ';\n' code += 'uint32_t src2 = ' + src2 + ';\n' code += 'uint32_t ca = ' + ca + ';\n' code += 'Rt = src1 + src2 + ca;\n' # Add code for calculating the carry, if needed if computeCA: code += computeCACode % dict + setXERCode # Setup the 4 code versions and add code to access XER if necessary code_rc1 = readXERCode + code code_oe1 = readXERCode + code + computeOVCode % dict + setXERCode code_rc1_oe1 = readXERCode + code + computeOVCode % dict + setXERCode if (computeCA or ca == 'xer.ca'): code = readXERCode + code code_rc1 += computeCR0Code % dict code_rc1_oe1 += computeCR0Code % dict # Generate the classes (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntOp', code, inst_flags, CheckRcOeDecode, BasicConstructor) (header_output_rc1, decoder_output_rc1, _, exec_output_rc1) = \ GenAluOp(name, Name + 'RcSet', 'IntOp', code_rc1, inst_flags, CheckRcOeDecode, IntRcConstructor) (header_output_oe1, decoder_output_oe1, _, exec_output_oe1) = \ GenAluOp(name, Name + 'OeSet', 'IntOp', code_oe1, inst_flags, CheckRcOeDecode, IntOeConstructor) (header_output_rc1_oe1, decoder_output_rc1_oe1, _, exec_output_rc1_oe1) = \ GenAluOp(name, Name + 'RcSetOeSet', 'IntOp', code_rc1_oe1, inst_flags, CheckRcOeDecode, IntRcOeConstructor) # Finally, add to the other outputs header_output += \ header_output_rc1 + header_output_oe1 + header_output_rc1_oe1 decoder_output += \ decoder_output_rc1 + decoder_output_oe1 + decoder_output_rc1_oe1 exec_output += \ exec_output_rc1 + exec_output_oe1 + exec_output_rc1_oe1 }}; // Instructions that use source registers Ra and Rb, with the result // placed into Rt. Basically multiply and divide instructions. The // carry bit is never set, but overflow can be calculated. Division // explicitly sets the overflow bit in certain situations and this is // dealt with using the 'divSetOV' boolean in decoder.isa. We generate // two versions of each instruction to deal with the Rc bit. def format IntArithOp(code, computeOV = 0, inst_flags = []) {{ # The result is always in Rt, but the source values vary dict = {'result':'Rt', 'inputa':'src1', 'inputb':'src2'} # Deal with setting the overflow flag if computeOV: code = 'bool divSetOV = false;\n' + code code += computeDivOVCode % dict + setXERCode # Setup the 2 code versions and add code to access XER if necessary code_rc1 = readXERCode + code + computeCR0Code % dict if computeOV: code = readXERCode + code # Generate the classes (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntOp', code, inst_flags, CheckRcDecode, BasicConstructor) # Generate the second class (header_output_rc1, decoder_output_rc1, _, exec_output_rc1) = \ GenAluOp(name, Name + 'RcSet', 'IntOp', code_rc1, inst_flags, CheckRcDecode, IntRcConstructor) # Finally, add to the other outputs header_output += header_output_rc1 decoder_output += decoder_output_rc1 exec_output += exec_output_rc1 }}; // A special format for rotate instructions which use certain fields // from the instruction's binary encoding. We need two versions for each // instruction to deal with the Rc bit. def format IntRotateOp(code, inst_flags = []) {{ # The result is always in Ra dict = {'result':'Ra'} # Setup the code for when Rc is set code_rc1 = readXERCode + code + computeCR0Code % dict # Generate the first class (header_output, decoder_output, decode_block, exec_output) = \ GenAluOp(name, Name, 'IntRotateOp', code, inst_flags, CheckRcDecode, BasicConstructor) # Generate the second class (header_output_rc1, decoder_output_rc1, _, exec_output_rc1) = \ GenAluOp(name, Name + 'RcSet', 'IntRotateOp', code_rc1, inst_flags, CheckRcDecode, IntRcConstructor) # Finally, add to the other outputs header_output += header_output_rc1 decoder_output += decoder_output_rc1 exec_output += exec_output_rc1 }};