File | % covered | Lines | Relevant Lines | Lines covered | Lines missed | Avg. Hits / Line |
---|---|---|---|---|---|---|
lib/origen_memory_image/binary.rb | 87.50 % | 69 | 32 | 28 | 4 | 53.84 |
lib/origen_memory_image/intel_hex.rb | 94.64 % | 98 | 56 | 53 | 3 | 8.29 |
lib/origen_memory_image/base.rb | 97.62 % | 96 | 42 | 41 | 1 | 45.55 |
lib/origen_memory_image.rb | 100.00 % | 43 | 23 | 23 | 0 | 9.78 |
lib/origen_memory_image/hex.rb | 100.00 % | 60 | 30 | 30 | 0 | 27.27 |
lib/origen_memory_image/s_record.rb | 100.00 % | 239 | 51 | 51 | 0 | 36.94 |
- 1
require 'origen'
- 1
require_relative '../config/application.rb'
- 1
module OrigenMemoryImage
- 1
autoload :Base, 'origen_memory_image/base'
- 1
autoload :SRecord, 'origen_memory_image/s_record'
- 1
autoload :Hex, 'origen_memory_image/hex'
- 1
autoload :Binary, 'origen_memory_image/binary'
- 1
autoload :IntelHex, 'origen_memory_image/intel_hex'
- 1
def self.new(file, options = {})
- 31
unless options[:source] == String
- 29
file = Origen.file_handler.clean_path_to(file)
end
- 31
find_type(file, options).new(file, options)
end
# Returns the class of the image manager for the given file
- 1
def self.find_type(file, options = {})
# Read first 10 lines
- 31
if options[:source] == String
- 2
snippet = file.split("\n")
else
- 29
snippet = File.foreach(file.to_s).first(10)
end
case
# Always do the binary first since the others won't be able to process
# a binary snippet
- 31
when options[:type] == :binary || (options[:source] != String && Binary.match?(file))
- 1
Binary
when options[:source] == String && Binary.match?(snippet, true)
- 1
Binary
when options[:type] == :srecord || SRecord.match?(snippet)
- 17
SRecord
when options[:type] == :intel_hex || IntelHex.match?(snippet)
- 5
IntelHex
when options[:type] == :hex || Hex.match?(snippet)
- 6
Hex
else
- 1
fail "Unknown format for image file: #{file}"
end
end
end
- 1
module OrigenMemoryImage
- 1
class Base
- 1
attr_reader :file, :source
- 1
def initialize(file, options = {})
- 30
if options[:source] == String
- 2
@source = file
else
- 28
@file = file
end
- 30
@ljust_partial_data = options[:ljust_partial_data]
end
# Returns the code execution start address as an int
- 1
def start_address
fail "#{self.class} has not implemented the start_address method!"
end
# Returns true if a start (jump address) record exists
- 1
def has_start_record
- 9
start_address unless @start_address
- 9
@start_record_found = false if @start_record_found.nil?
- 9
@start_record_found
end
# Returns the s-record as an array of addresses and data
#
# @param [hash] options, allows the selection of endianness swapping - ie the output will have the endianness changed
#
# The output is a 2D array, with each element being an array with element zero being the
# address of the data and element one being one word of data
# like this [[ADDR0, DATA0], [ADDR1, DATA1], [ADDR2, DATA2]...]
#
# The block header data and end of block value are not interpreted in any way and
# the checksum bits are disregarded
- 1
def to_a(options = {})
options = {
- 29
flip_endianness: false,
data_width_in_bytes: 4,
crop: []
}.merge(options)
- 29
data = extract_addr_data(options)
- 28
if options[:crop].count > 0
- 7
cropped_data = []
- 7
data.each do |addr, data|
- 129
case options[:crop].count
when 1
- 64
cropped_data.push([addr, data]) if addr >= options[:crop][0]
when 2
- 64
cropped_data.push([addr, data]) if addr >= options[:crop][0] && addr <= options[:crop][1]
else
- 1
fail 'crop option can only be array of size 1 or 2'
end
end
- 6
data = cropped_data
end
- 27
if options[:flip_endianness] || options[:endianness_change]
- 5
data.map do |v|
- 94
[v[0], flip_endianness(v[1], options[:data_width_in_bytes])]
end
else
- 22
data
end
end
- 1
alias_method :to_array, :to_a
# Reverse the endianness of the given data value, the width of it in bytes must
# be supplied as the second argument
#
# @example
# flip_endianness(0x12345678, 4) # => 0x78563412
- 1
def flip_endianness(data, width_in_bytes)
- 94
v = 0
- 94
width_in_bytes.times do |i|
# data[7:0] => data[15:8]
- 448
start = 8 * i
- 448
v += data[(start + 7)..start] << ((width_in_bytes - i - 1) * 8)
end
- 94
v
end
- 1
def file_name
- 3
file || 'From source string'
end
- 1
def lines
- 46
if file
- 43
File.readlines(file)
else
- 3
source.split("\n")
end
end
end
end
- 1
module OrigenMemoryImage
- 1
class Binary < Base
- 1
def self.match?(file, snippet = false)
- 31
if snippet
- 7
file.all? { |l| l.strip =~ /^[01]*$/ }
else
# detect whether the data is mostly not alpha numeric
- 29
filedata = (File.read(file, 256) || '')
- 29
(filedata.gsub(/\s+/, '').gsub(/\w/, '').length.to_f / filedata.length.to_f) > 0.3
end
end
# Always returns 0 since binary files do not contain addresses
- 1
def start_address
- 3
0
end
- 1
def create_test_file
data = [
0x1EE0021C, 0x22401BE0, 0x021C2243,
0x18E0021C, 0x5A780A43, 0x03E0034B,
0xF7215A78, 0x0A400020, 0x22E08442,
0x22D31FE0, 0x84421FD9, 0x1CE08442,
0x002B20D1, 0x03E0012A, 0x01D1002B,
0x1BD00223, 0x2340022A, 0x02D1002B,
0x15D103E0, 0x032A01D1, 0x78000018,
0x7C000018, 0x82000018, 0x88000018
]
data = data.map { |d| d.to_s(2).rjust(32, '0') }.join
File.open('examples/bin1.bin', 'wb') do |output|
output.write [data].pack('B*')
end
end
- 1
private
# Returns an array containing all address/data from the given s-record
# No address manipulation is performed, that is left to the caller to apply
# any scrambling as required by the target system
- 1
def extract_addr_data(options = {})
options = {
- 9
data_width_in_bytes: 4
}.merge(options)
- 9
result = []
- 9
width = options[:data_width_in_bytes]
- 9
address = 0
- 9
if file
- 8
raw = File.binread(file)
- 8
bytes = raw.unpack('C*')
else
- 1
raw = lines.map(&:strip).join
- 17
bytes = raw.scan(/.{1,8}/).map { |s| s.to_i(2) }
end
- 9
bytes.each_slice(width) do |d|
- 184
v = 0
- 184
width.times do |i|
- 784
v |= d[i] << ((width - 1 - i) * 8) if d[i]
end
- 184
result << [address, v]
- 184
address += width
end
- 9
result
end
end
end
- 1
module OrigenMemoryImage
- 1
class Hex < Base
- 1
def self.match?(snippet)
- 6
snippet.any? do |line|
# Match a line like:
# @180000F0
- 15
line =~ /^@[0-9a-fA-F]+\s?$/
end
end
# The first in the file will be taken as the start address
- 1
def start_address
- 4
@start_address ||= begin
- 4
lines.each do |line|
- 4
if line =~ /^@([0-9a-fA-F]+)\s?$/
- 4
return Regexp.last_match[1].to_i(16)
end
end
end
end
- 1
private
# Returns an array containing all address/data from the given s-record
# No address manipulation is performed, that is left to the caller to apply
# any scrambling as required by the target system
- 1
def extract_addr_data(options = {})
options = {
- 9
data_width_in_bytes: 4
}.merge(options)
- 9
result = []
- 9
lines.each do |line|
# Only if the line is an s-record with data...
- 67
if line =~ /^@([0-9a-fA-F]+)\s?$/
- 22
@address = Regexp.last_match[1].to_i(16)
- 45
elsif line =~ /^[0-9A-F]/
- 45
unless @address
- 1
fail "Hex data found before an @address line in #{file_name}"
end
- 44
data = line.strip.gsub(/\s/, '')
- 44
data_matcher = '\w\w' * options[:data_width_in_bytes]
- 44
data.scan(/#{data_matcher}/).each do |data_packet|
- 190
result << [@address, data_packet.to_i(16)]
- 190
@address += options[:data_width_in_bytes]
end
# If a partial word is left over
- 44
if (remainder = data.length % (2 * options[:data_width_in_bytes])) > 0
- 2
if @ljust_partial_data
- 1
result << [@address, data[data.length - remainder..data.length].ljust(options[:data_width_in_bytes] * 2, '0').to_i(16)]
else
- 1
result << [@address, data[data.length - remainder..data.length].to_i(16)]
end
end
end
end
- 8
result
end
end
end
- 1
module OrigenMemoryImage
- 1
class IntelHex < Base
- 1
def self.match?(snippet)
- 12
snippet.all? do |line|
- 41
line.empty? || line =~ /^:[0-9A-Fa-f]{6}0[0-5]/
end
end
- 1
def start_address
- 3
@start_address ||= begin
- 3
addrs = []
- 3
lines.each do |line|
- 21
line = line.strip
- 21
if start_linear_address?(line)
- 2
addrs << decode(line)[:data].to_i(16)
end
end
- 3
addrs.last || 0
end
end
- 1
private
- 1
def decode(line)
- 12
d = {}
- 12
if line =~ /^:([0-9A-Fa-f]{2})([0-9A-Fa-f]{4})(\d\d)([0-9A-Fa-f]+)([0-9A-Fa-f]{2})$/
- 12
d[:byte_count] = Regexp.last_match(1).to_i(16)
- 12
d[:address] = Regexp.last_match(2).to_i(16)
- 12
d[:record_type] = Regexp.last_match(3).to_i(16)
- 12
d[:data] = Regexp.last_match(4)
- 12
d[:checksum] = Regexp.last_match(5).to_i(16)
else
fail "Invalid line encountered in Intel Hex formatted file: #{line}"
end
- 12
d
end
- 1
def data?(line)
- 11
!!(line =~ /^:[0-9A-Fa-f]{6}00/)
end
- 1
def extended_segment_address?(line)
- 13
!!(line =~ /^:[0-9A-Fa-f]{6}02/)
end
- 1
def extended_linear_address?(line)
- 13
!!(line =~ /^:[0-9A-Fa-f]{6}04/)
end
- 1
def start_linear_address?(line)
- 21
!!(line =~ /^:[0-9A-Fa-f]{6}05/)
end
- 1
def upper_addr
- 8
@upper_addr || 0
end
- 1
def segment_address
- 8
@segment_address || 0
end
# Returns an array containing all address/data from the given s-record
# No address manipulation is performed, that is left to the caller to apply
# any scrambling as required by the target system
- 1
def extract_addr_data(options = {})
options = {
- 2
data_width_in_bytes: 4
}.merge(options)
- 2
result = []
- 2
lines.each do |line|
- 13
line = line.strip
- 13
if extended_segment_address?(line)
@segment_address = decode(line)[:data].to_i(16) * 16
- 13
elsif extended_linear_address?(line)
- 2
@upper_addr = (decode(line)[:data].to_i(16)) << 16
- 11
elsif data?(line)
- 8
d = decode(line)
- 8
addr = d[:address] + segment_address + upper_addr
- 8
data = d[:data]
- 8
data_matcher = '\w\w' * options[:data_width_in_bytes]
- 8
data.scan(/#{data_matcher}/).each do |data_packet|
- 32
result << [addr, data_packet.to_i(16)]
- 32
addr += options[:data_width_in_bytes]
end
# If a partial word is left over
- 8
if (remainder = data.length % (2 * options[:data_width_in_bytes])) > 0
result << [addr, data[data.length - remainder..data.length].to_i(16)]
end
end
end
- 2
result
end
end
end
- 1
module OrigenMemoryImage
# An S-record file consists of a sequence of specially formatted ASCII character strings. An S-record will
# be less than or equal to 78 bytes in length.
# The order of S-records within a file is of no significance and no particular order may be assumed.
#
# The general format of an S-record follows:
#
# +-------------------//------------------//-----------------------+
# | type | count | address | data | checksum |
# +-------------------//------------------//-----------------------+
#
# type
# : A char[2] field. These characters describe the type of record (S0, S1, S2, S3, S5, S7, S8, or S9).
#
# count
# : A char[2] field. These characters when paired and interpreted as a hexadecimal value, display
# the count of remaining character pairs in the record.
#
# address
# : A char[4,6, or 8] field. These characters grouped and interpreted as a hexadecimal value,
# display the address at which the data field is to be loaded into memory. The length of the field depends
# on the number of bytes necessary to hold the address. A 2-byte address uses 4 characters, a 3-byte
# address uses 6 characters, and a 4-byte address uses 8 characters.
#
# data
# : A char [0-64] field. These characters when paired and interpreted as hexadecimal values represent
# the memory loadable data or descriptive information.
#
# checksum
# : A char[2] field. These characters when paired and interpreted as a hexadecimal value display
# the least significant byte of the ones complement of the sum of the byte values represented by the pairs
# of characters making up the count, the address, and the data fields.
#
# Each record is terminated with a line feed. If any additional or different record terminator(s) or delay
# characters are needed during transmission to the target system it is the responsibility of the
# transmitting program to provide them.
#
# #### S0 Record
#
# The type of record is 'S0' (0x5330). The address field is unused and will be filled with zeros
# (0x0000). The header information within the data field is divided into the following subfields.
#
# * mname is char[20] and is the module name.
# * ver is char[2] and is the version number.
# * rev is char[2] and is the revision number.
# * description is char[0-36] and is a text comment.
#
# Each of the subfields is composed of ASCII bytes whose associated characters, when paired, represent one
# byte hexadecimal values in the case of the version and revision numbers, or represent the hexadecimal
# values of the ASCII characters comprising the module name and description.
#
# #### S1 Record
#
# The type of record field is 'S1' (0x5331). The address field is intrepreted as a 2-byte
# address. The data field is composed of memory loadable data.
#
# #### S2 Record
#
# The type of record field is 'S2' (0x5332). The address field is intrepreted as a 3-byte
# address. The data field is composed of memory loadable data.
#
# #### S3 Record
#
# The type of record field is 'S3' (0x5333). The address field is intrepreted as a 4-byte
# address. The data field is composed of memory loadable data.
#
# #### S5 Record
#
# The type of record field is 'S5' (0x5335). The address field is intrepreted as a 2-byte value
# and contains the count of S1, S2, and S3 records previously transmitted. There is no data field.
#
# #### S7 Record
#
# The type of record field is 'S7' (0x5337). The address field contains the starting execution
# address and is intrepreted as 4-byte address. There is no data field.
#
# #### S8 Record
#
# The type of record field is 'S8' (0x5338). The address field contains the starting execution
# address and is intrepreted as 3-byte address. There is no data field.
#
# #### S9 Record
#
# The type of record field is 'S9' (0x5339). The address field contains the starting execution
# address and is intrepreted as 2-byte address. There is no data field.
#
# ### Example
#
# Shown below is a typical S-record format file.
#
# S00600004844521B
# S1130000285F245F2212226A000424290008237C2A
# S11300100002000800082629001853812341001813
# S113002041E900084E42234300182342000824A952
# S107003000144ED492
# S5030004F8
# S9030000FC
#
# The file consists of one S0 record, four S1 records, one S5 record and an S9 record.
#
# The S0 record is comprised as follows:
#
# * S0 S-record type S0, indicating it is a header record.
# * 06 Hexadecimal 06 (decimal 6), indicating that six character pairs (or ASCII bytes) follow.
# * 00 00 Four character 2-byte address field, zeroes in this example.
# * 48 44 52 ASCII H, D, and R - "HDR".
# * 1B The checksum.
#
# The first S1 record is comprised as follows:
#
# * S1 S-record type S1, indicating it is a data record to be loaded at a 2-byte address.
# * 13 Hexadecimal 13 (decimal 19), indicating that nineteen character pairs, representing a 2 byte address,
# * 16 bytes of binary data, and a 1 byte checksum, follow.
# * 00 00 Four character 2-byte address field; hexidecimal address 0x0000, where the data which follows is to
# be loaded.
# * 28 5F 24 5F 22 12 22 6A 00 04 24 29 00 08 23 7C Sixteen character pairs representing the actual binary
# data.
# * 2A The checksum.
# * The second and third S1 records each contain 0x13 (19) character pairs and are ended with checksums of 13
# and 52, respectively. The fourth S1 record contains 07 character pairs and has a checksum of 92.
#
# The S5 record is comprised as follows:
#
# * S5 S-record type S5, indicating it is a count record indicating the number of S1 records
# * 03 Hexadecimal 03 (decimal 3), indicating that three character pairs follow.
# * 00 04 Hexadecimal 0004 (decimal 4), indicating that there are four data records previous to this record.
# * F8 The checksum.
#
# The S9 record is comprised as follows:
#
# * S9 S-record type S9, indicating it is a termination record.
# * 03 Hexadecimal 03 (decimal 3), indicating that three character pairs follow.
# * 00 00 The address field, hexadecimal 0 (decimal 0) indicating the starting execution address.
# * FC The checksum.
#
# ### Additional Notes
#
# There isn't any evidence that Motorola ever has made use of the header information within the data field
# of the S0 record, as described above. This must have been used by some third party vendors.
# This is the only place that a 78-byte limit on total record length or 64-byte limit on data length is
# documented. These values shouldn't be trusted for the general case.
#
# The count field can have values in the range of 0x3 (2 bytes of address + 1 byte checksum = 3, a not
# very useful record) to 0xff; this is the count of remaining character pairs, including checksum.
# If you write code to convert S-Records, you should always assume that a record can be as long as 514
# (decimal) characters in length (255 * 2 = 510, plus 4 characters for the type and count fields), plus
# any terminating character(s).
#
# That is, in establishing an input buffer in C, you would declare it to be
# an array of 515 chars, thus leaving room for the terminating null character.
- 1
class SRecord < Base
- 1
def self.match?(snippet)
- 29
snippet.all? do |line|
- 129
line.empty? || line =~ /^S[01235789]/
end
end
- 1
def start_address
- 18
if @call_order_warn
- 1
Origen.log.warn 'Previously srec.start_address returned the lowest address when to_a was called first. Now the start record is always returned if present.'
- 1
@call_order_warn = false
end
- 18
lowest_address = nil
- 18
@start_address ||= begin
- 18
lines.each do |line|
- 124
if line =~ /^S([789])(.*)/
- 16
@start_record_found = true
- 16
type = Regexp.last_match[1]
- 16
case type
when '7'
- 10
return line.slice(4, 8).to_i(16)
when '8'
- 4
return line.slice(4, 6).to_i(16)
when '9'
- 2
return line.slice(4, 4).to_i(16)
end
end
- 108
if line =~ /^S([1-3])/
- 90
type = Regexp.last_match[1].to_i(16) # S-record type, 1-3
# Set the matcher to capture x number of bytes dependent on the s-rec type
- 90
addr_matcher = '\w\w' * (1 + type)
- 90
line.strip =~ /^S\d\w\w(#{addr_matcher})(\w*)\w\w$/ # $1 = address, $2 = data
- 90
addr = Regexp.last_match[1].to_i(16)
- 90
lowest_address ||= addr
- 90
lowest_address = addr if addr < lowest_address
end
end
# if no start_address record is found, return lowest address
- 2
@start_record_found = false
- 2
lowest_address
end
end
- 1
private
# Returns an array containing all address/data from the given s-record
# No address manipulation is performed, that is left to the caller to apply
# any scrambling as required by the target system
- 1
def extract_addr_data(options = {})
options = {
- 9
data_width_in_bytes: 4
}.merge(options)
# guarantee that the start_address will be the jump address if provided
- 9
if @start_address.nil?
- 9
start_address
- 9
@call_order_warn = @start_record_found ? true : false
end
- 9
result = []
- 9
lines.each do |line|
# Only if the line is an s-record with data...
- 63
if line =~ /^S([1-3])/
- 45
type = Regexp.last_match[1].to_i(16) # S-record type, 1-3
# Set the matcher to capture x number of bytes dependent on the s-rec type
- 45
addr_matcher = '\w\w' * (1 + type)
- 45
line.strip =~ /^S\d\w\w(#{addr_matcher})(\w*)\w\w$/ # $1 = address, $2 = data
- 45
addr = Regexp.last_match[1].to_i(16)
- 45
data = Regexp.last_match[2]
- 45
data_matcher = '\w\w' * options[:data_width_in_bytes]
- 45
data.scan(/#{data_matcher}/).each do |data_packet|
- 158
result << [addr, data_packet.to_i(16)]
- 158
addr += options[:data_width_in_bytes]
end
# If a partial word is left over
- 45
if (remainder = data.length % (2 * options[:data_width_in_bytes])) > 0
- 2
if @ljust_partial_data
- 1
result << [addr, data[data.length - remainder..data.length].ljust(options[:data_width_in_bytes] * 2, '0').to_i(16)]
else
- 1
result << [addr, data[data.length - remainder..data.length].to_i(16)]
end
end
end
end
- 9
result
end
end
end