#    This file is part of Metasm, the Ruby assembly manipulation suite
#    Copyright (C) 2006-2009 Yoann GUILLOT
#
#    Licence is LGPL, see LICENCE in the top-level directory


require 'metasm/arm/opcodes'
require 'metasm/encode'

module Metasm
class ARM
	def encode_instr_op(section, instr, op)
		base = op.bin
		set_field = lambda { |f, v|
			v = v.reduce if v.kind_of? Expression
			case f
			when :i8_12
				base = Expression[base, :|, [[v, :&, 0xf], :|, [[v, :<<, 4], :&, 0xf00]]]
				next
			when :stype; v = [:lsl, :lsr, :asr, :ror].index(v)
			when :u; v = [:-, :+].index(v)
			end
			base = Expression[base, :|, [[v, :&, @fields_mask[f]], :<<, @fields_shift[f]]]
		}

		val, mask, shift = 0, 0, 0

		if op.props[:cond]
			coff = op.props[:cond_name_off] || op.name.length
			cd = instr.opname[coff, 2]
			cdi = %w[eq ne cs cc mi pl vs vc hi ls ge lt gt le al].index(cd) || 14	# default = al
			set_field[:cond, cdi]
		end

		op.args.zip(instr.args).each { |sym, arg|
			case sym
			when :rd, :rs, :rn, :rm; set_field[sym, arg.i]
			when :rm_rs
				set_field[:rm, arg.i]
				set_field[:stype, arg.stype]
				set_field[:rs, arg.shift.i]
			when :rm_is
				set_field[:rm, arg.i]
				set_field[:stype, arg.stype]
				set_field[:shifti, arg.shift/2]
			when :mem_rn_rm, :mem_rn_rms, :mem_rn_i8_12, :mem_rn_i12
				set_field[:rn, arg.base.i]
				case sym
				when :mem_rn_rm
					set_field[:rm, arg.offset.i]
				when :mem_rn_rms
					set_field[:rm, arg.offset.i]
					set_field[:stype, arg.offset.stype]
					set_field[:rs, arg.offset.shift.i]
				when :mem_rn_i8_12
					set_field[:i8_12, arg.offset]
				when :mem_rn_i12
					set_field[:i12, arg.offset]
				end
				# TODO set_field[:u] etc
			when :reglist
				set_field[sym, arg.list.inject(0) { |rl, r| rl | (1 << r.i) }]
			when :i8_r
				# XXX doublecheck this
				b = arg.reduce & 0xffffffff
				r = (0..15).find { next true if b < 0x10 ; b = (b >> 2) | ((b & 3) << 30) }
				set_field[:i8, b]
				set_field[:rotate, r]
			when :i16, :i24
				val, mask, shift = arg, @fields_mask[sym], @fields_shift[sym]
			end
		}

		Expression[base, :|, [[val, :<<, shift], :&, mask]].encode(:u32, @endianness)
	end
end
end