# frozen_string_literal: true # # ronin-code-asm - A Ruby DSL for crafting Assembly programs and shellcode. # # Copyright (c) 2007-2022 Hal Brodigan (postmodern.mod3 at gmail.com) # # ronin-code-asm is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published # by the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # ronin-code-asm is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with ronin-code-asm. If not, see . # require 'ronin/code/asm/syntax/common' module Ronin module Code module ASM module Syntax class Intel < Common # Data sizes and their identifiers WIDTHS = { 1 => 'BYTE', 2 => 'WORD', 4 => 'DWORD', 8 => 'QWORD' } # # Emits a register. # # @param [Register] reg # The register. # # @return [String] # The register name. # def self.emit_register(reg) reg.name.to_s end # # Emits an immediate operand. # # @param [ImmediateOperand] op # The operand. # # @return [String] # The formatted immediate operand. # def self.emit_immediate_operand(op) "#{WIDTHS[op.width]} #{emit_integer(op.value)}" end # # Emits a memory operand. # # @param [MemoryOperand] op # The operand. # # @return [String] # The formatted memory operand. # def self.emit_memory_operand(op) asm = emit_register(op.base) if op.index asm << '+' << emit_register(op.index) asm << '*' << emit_integer(op.scale) if op.scale > 1 end if op.offset != 0 sign = if op.offset >= 0 then '+' else '-' end asm << sign << emit_integer(op.offset) end asm = "[#{asm}]" unless op.width == op.base.width asm = "#{WIDTHS[op.width]} #{asm}" end return asm end # # Emits an instruction. # # @param [Instruction] ins # The instruction. # # @return [String] # The formatted instruction. # def self.emit_instruction(ins) line = emit_keyword(ins.name) unless ins.operands.empty? line << "\t" << emit_operands(ins.operands) end return line end # # Emits a section name. # # @param [Symbol] name # The section name. # # @return [String] # The formatted section name. # # @since 0.2.0 # def self.emit_section(name) "section .#{name}" end # # Emits the program's prologue. # # @param [Program] program # The program. # # @return [String] # The formatted prologue. # # @since 0.2.0 # def self.emit_prologue(program) "BITS #{BITS[program.arch]}" end end end end end end