commit
a0f5559a86
|
@ -20,7 +20,7 @@
|
|||
<option id="cdt.managedbuild.option.gnu.cross.prefix.1771691886" name="Prefix" superClass="cdt.managedbuild.option.gnu.cross.prefix"/>
|
||||
<option id="cdt.managedbuild.option.gnu.cross.path.1831805518" name="Path" superClass="cdt.managedbuild.option.gnu.cross.path"/>
|
||||
<targetPlatform archList="all" binaryParser="org.eclipse.cdt.core.ELF" id="cdt.managedbuild.targetPlatform.gnu.cross.2055058514" isAbstract="false" osList="all" superClass="cdt.managedbuild.targetPlatform.gnu.cross"/>
|
||||
<builder command="gen_misc.bat" enableAutoBuild="false" enabledIncrementalBuild="true" id="cdt.managedbuild.builder.gnu.cross.415628675" incrementalBuildTarget="" keepEnvironmentInBuildfile="false" managedBuildOn="false" name="Gnu Make Builder" parallelBuildOn="false" superClass="cdt.managedbuild.builder.gnu.cross">
|
||||
<builder enableAutoBuild="false" enabledIncrementalBuild="true" id="cdt.managedbuild.builder.gnu.cross.415628675" incrementalBuildTarget="all" keepEnvironmentInBuildfile="false" managedBuildOn="false" name="Gnu Make Builder" parallelBuildOn="false" superClass="cdt.managedbuild.builder.gnu.cross">
|
||||
<outputEntries>
|
||||
<entry flags="VALUE_WORKSPACE_PATH|RESOLVED" kind="outputPath" name=""/>
|
||||
<entry flags="VALUE_WORKSPACE_PATH|RESOLVED" kind="outputPath" name="bin"/>
|
||||
|
@ -69,4 +69,32 @@
|
|||
<autodiscovery enabled="true" problemReportingEnabled="true" selectedProfileId=""/>
|
||||
</scannerConfigBuildInfo>
|
||||
</storageModule>
|
||||
<storageModule moduleId="org.eclipse.cdt.make.core.buildtargets">
|
||||
<buildTargets>
|
||||
<target name="all" path="" targetID="org.eclipse.cdt.build.MakeTargetBuilder">
|
||||
<buildCommand>${PWD}/gen_misc.bat</buildCommand>
|
||||
<buildArguments/>
|
||||
<buildTarget>all</buildTarget>
|
||||
<stopOnError>true</stopOnError>
|
||||
<useDefaultCommand>false</useDefaultCommand>
|
||||
<runAllBuilders>true</runAllBuilders>
|
||||
</target>
|
||||
<target name="clean" path="" targetID="org.eclipse.cdt.build.MakeTargetBuilder">
|
||||
<buildCommand>make</buildCommand>
|
||||
<buildArguments/>
|
||||
<buildTarget>clean</buildTarget>
|
||||
<stopOnError>true</stopOnError>
|
||||
<useDefaultCommand>true</useDefaultCommand>
|
||||
<runAllBuilders>true</runAllBuilders>
|
||||
</target>
|
||||
<target name="flash" path="" targetID="org.eclipse.cdt.build.MakeTargetBuilder">
|
||||
<buildCommand>${PWD}/flash.bat</buildCommand>
|
||||
<buildArguments/>
|
||||
<buildTarget/>
|
||||
<stopOnError>true</stopOnError>
|
||||
<useDefaultCommand>false</useDefaultCommand>
|
||||
<runAllBuilders>true</runAllBuilders>
|
||||
</target>
|
||||
</buildTargets>
|
||||
</storageModule>
|
||||
</cproject>
|
||||
|
|
|
@ -5,10 +5,6 @@
|
|||
<provider copy-of="extension" id="org.eclipse.cdt.ui.UserLanguageSettingsProvider"/>
|
||||
<provider-reference id="org.eclipse.cdt.core.ReferencedProjectsLanguageSettingsProvider" ref="shared-provider"/>
|
||||
<provider copy-of="extension" id="org.eclipse.cdt.managedbuilder.core.GCCBuildCommandParser"/>
|
||||
<provider class="org.eclipse.cdt.internal.build.crossgcc.CrossGCCBuiltinSpecsDetector" console="false" env-hash="-1433312569745558301" id="org.eclipse.cdt.build.crossgcc.CrossGCCBuiltinSpecsDetector" keep-relative-paths="false" name="CDT Cross GCC Built-in Compiler Settings" parameter="${COMMAND} ${FLAGS} -E -P -v -dD "${INPUTS}"" prefer-non-shared="true">
|
||||
<language-scope id="org.eclipse.cdt.core.gcc"/>
|
||||
<language-scope id="org.eclipse.cdt.core.g++"/>
|
||||
</provider>
|
||||
<provider-reference id="org.eclipse.cdt.managedbuilder.core.MBSLanguageSettingsProvider" ref="shared-provider"/>
|
||||
</extension>
|
||||
</configuration>
|
||||
|
|
|
@ -0,0 +1,19 @@
|
|||
@echo off
|
||||
set BACKPATH=%PATH%
|
||||
set PATH=%BACKPATH%;%CD%\..\tools
|
||||
@echo on
|
||||
|
||||
rem gen_misc.bat
|
||||
rem eagle.app.v6.flash.bin: 0x00000
|
||||
rem eagle.app.v6.irom0text.bin: 0x10000
|
||||
rem esp_init_data_default.bin: 0x7c000
|
||||
rem blank.bin: 0x7e000
|
||||
|
||||
cd ..\bin
|
||||
esptool.py -p com1 write_flash 0x00000 eagle.app.v6.flash.bin 0x10000 eagle.app.v6.irom0text.bin 0x7c000 esp_init_data_default.bin 0x7e000 blank.bin
|
||||
echo ************************* flash end *************************************
|
||||
cd ..\app
|
||||
|
||||
@echo off
|
||||
set PATH=%BACKPATH%
|
||||
@echo on
|
|
@ -0,0 +1,11 @@
|
|||
#!/bin/bash -x
|
||||
### gen_misc.sh
|
||||
### eagle.app.v6.flash.bin: 0x00000
|
||||
### eagle.app.v6.irom0text.bin: 0x10000
|
||||
### esp_init_data_default.bin: 0x7c000
|
||||
### blank.bin: 0x7e000
|
||||
|
||||
cd ../bin
|
||||
../../../../../tools/esptool.py --port /dev/ttyUSB0 write_flash 0x00000 eagle.app.v6.flash.bin 0x10000 eagle.app.v6.irom0text.bin 0x7c000 esp_init_data_default.bin 0x7e000 blank.bin
|
||||
echo "************************* flash end *************************************"
|
||||
cd ../app
|
|
@ -38,7 +38,7 @@
|
|||
#define ICACHE_STORE_TYPEDEF_ATTR __attribute__((aligned(4),packed))
|
||||
#define ICACHE_STORE_ATTR __attribute__((aligned(4)))
|
||||
#define ICACHE_RAM_ATTR __attribute__((section(".iram0.text")))
|
||||
#define ICACHE_RODATA_ATTR __attribute__((section(".rodata.text")))
|
||||
#define ICACHE_RODATA_ATTR __attribute__((section(".rodata2.text")))
|
||||
|
||||
#define CLIENT_SSL_ENABLE
|
||||
#define GPIO_INTERRUPT_ENABLE
|
||||
|
|
|
@ -280,6 +280,7 @@ tmr.alarm(6,2000,1,function() print("alarm 6") end)
|
|||
tmr.stop(6)
|
||||
|
||||
for k,v in pairs(_G.package.loaded) do print(k) end
|
||||
for k,v in pairs(_G) do print(k) end
|
||||
for k,v in pairs(d) do print("n:"..k..", s:"..v) end
|
||||
|
||||
a="pin=9"
|
||||
|
|
|
@ -70,8 +70,8 @@ SECTIONS
|
|||
{
|
||||
_irom0_text_start = ABSOLUTE(.);
|
||||
*(.irom0.literal .irom.literal .irom.text.literal .irom0.text .irom.text)
|
||||
*(.literal.* .text.*)
|
||||
*(.rodata.text)
|
||||
*(.literal.* .text.*)
|
||||
*(.rodata2.text)
|
||||
_irom0_text_end = ABSOLUTE(.);
|
||||
_flash_used_end = ABSOLUTE(.);
|
||||
} >irom0_0_seg :irom0_0_phdr
|
||||
|
@ -85,7 +85,7 @@ SECTIONS
|
|||
*(.init)
|
||||
/* *(.literal .text .literal.* .text.* .stub .gnu.warning .gnu.linkonce.literal.* .gnu.linkonce.t.*.literal .gnu.linkonce.t.*) */
|
||||
*(.literal .text .iram0.text .stub .gnu.warning .gnu.linkonce.literal.* .gnu.linkonce.t.*.literal .gnu.linkonce.t.*)
|
||||
*(.fini.literal)
|
||||
*(.fini.literal)
|
||||
*(.fini)
|
||||
*(.gnu.version)
|
||||
_text_end = ABSOLUTE(.);
|
||||
|
|
|
@ -0,0 +1,486 @@
|
|||
#!/usr/bin/env python2.7
|
||||
#
|
||||
# ESP8266 ROM Bootloader Utility
|
||||
# https://github.com/themadinventor/esptool
|
||||
#
|
||||
# Copyright (C) 2014 Fredrik Ahlberg
|
||||
#
|
||||
# This program is free software; you can redistribute it and/or modify it under
|
||||
# the terms of the GNU General Public License as published by the Free Software
|
||||
# Foundation; either version 2 of the License, or (at your option) any later version.
|
||||
#
|
||||
# This program 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 General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License along with
|
||||
# this program; if not, write to the Free Software Foundation, Inc., 51 Franklin
|
||||
# Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
|
||||
import sys
|
||||
import struct
|
||||
import serial
|
||||
import math
|
||||
import time
|
||||
import argparse
|
||||
import os
|
||||
import subprocess
|
||||
|
||||
class ESPROM:
|
||||
|
||||
# These are the currently known commands supported by the ROM
|
||||
ESP_FLASH_BEGIN = 0x02
|
||||
ESP_FLASH_DATA = 0x03
|
||||
ESP_FLASH_END = 0x04
|
||||
ESP_MEM_BEGIN = 0x05
|
||||
ESP_MEM_END = 0x06
|
||||
ESP_MEM_DATA = 0x07
|
||||
ESP_SYNC = 0x08
|
||||
ESP_WRITE_REG = 0x09
|
||||
ESP_READ_REG = 0x0a
|
||||
|
||||
# Maximum block sized for RAM and Flash writes, respectively.
|
||||
ESP_RAM_BLOCK = 0x1800
|
||||
ESP_FLASH_BLOCK = 0x100
|
||||
|
||||
# Default baudrate. The ROM auto-bauds, so we can use more or less whatever we want.
|
||||
ESP_ROM_BAUD = 115200
|
||||
|
||||
# First byte of the application image
|
||||
ESP_IMAGE_MAGIC = 0xe9
|
||||
|
||||
# Initial state for the checksum routine
|
||||
ESP_CHECKSUM_MAGIC = 0xef
|
||||
|
||||
# OTP ROM addresses
|
||||
ESP_OTP_MAC0 = 0x3ff00050
|
||||
ESP_OTP_MAC1 = 0x3ff00054
|
||||
|
||||
def __init__(self, port = 0, baud = ESP_ROM_BAUD):
|
||||
self._port = serial.Serial(port, baud)
|
||||
|
||||
""" Read bytes from the serial port while performing SLIP unescaping """
|
||||
def read(self, length = 1):
|
||||
b = ''
|
||||
while len(b) < length:
|
||||
c = self._port.read(1)
|
||||
if c == '\xdb':
|
||||
c = self._port.read(1)
|
||||
if c == '\xdc':
|
||||
b = b + '\xc0'
|
||||
elif c == '\xdd':
|
||||
b = b + '\xdb'
|
||||
else:
|
||||
raise Exception('Invalid SLIP escape')
|
||||
else:
|
||||
b = b + c
|
||||
return b
|
||||
|
||||
""" Write bytes to the serial port while performing SLIP escaping """
|
||||
def write(self, packet):
|
||||
buf = '\xc0'
|
||||
for b in packet:
|
||||
if b == '\xc0':
|
||||
buf += '\xdb\xdc'
|
||||
elif b == '\xdb':
|
||||
buf += '\xdb\xdd'
|
||||
else:
|
||||
buf += b
|
||||
buf += '\xc0'
|
||||
self._port.write(buf)
|
||||
|
||||
""" Calculate checksum of a blob, as it is defined by the ROM """
|
||||
@staticmethod
|
||||
def checksum(data, state = ESP_CHECKSUM_MAGIC):
|
||||
for b in data:
|
||||
state ^= ord(b)
|
||||
return state
|
||||
|
||||
""" Send a request and read the response """
|
||||
def command(self, op = None, data = None, chk = 0):
|
||||
if op:
|
||||
# Construct and send request
|
||||
pkt = struct.pack('<BBHI', 0x00, op, len(data), chk) + data
|
||||
self.write(pkt)
|
||||
|
||||
# Read header of response and parse
|
||||
if self._port.read(1) != '\xc0':
|
||||
raise Exception('Invalid head of packet')
|
||||
hdr = self.read(8)
|
||||
(resp, op_ret, len_ret, val) = struct.unpack('<BBHI', hdr)
|
||||
if resp != 0x01 or (op and op_ret != op):
|
||||
raise Exception('Invalid response')
|
||||
|
||||
# The variable-length body
|
||||
body = self.read(len_ret)
|
||||
|
||||
# Terminating byte
|
||||
if self._port.read(1) != chr(0xc0):
|
||||
raise Exception('Invalid end of packet')
|
||||
|
||||
return val, body
|
||||
|
||||
""" Perform a connection test """
|
||||
def sync(self):
|
||||
self.command(ESPROM.ESP_SYNC, '\x07\x07\x12\x20'+32*'\x55')
|
||||
for i in xrange(7):
|
||||
self.command()
|
||||
|
||||
""" Try connecting repeatedly until successful, or giving up """
|
||||
def connect(self):
|
||||
print 'Connecting...'
|
||||
|
||||
# RTS = CH_PD (i.e reset)
|
||||
# DTR = GPIO0
|
||||
self._port.setRTS(True)
|
||||
self._port.setDTR(True)
|
||||
self._port.setRTS(False)
|
||||
time.sleep(0.1)
|
||||
self._port.setDTR(False)
|
||||
|
||||
self._port.timeout = 0.5
|
||||
for i in xrange(10):
|
||||
try:
|
||||
self._port.flushInput()
|
||||
self._port.flushOutput()
|
||||
self.sync()
|
||||
self._port.timeout = 5
|
||||
return
|
||||
except:
|
||||
time.sleep(0.1)
|
||||
raise Exception('Failed to connect')
|
||||
|
||||
""" Read memory address in target """
|
||||
def read_reg(self, addr):
|
||||
res = self.command(ESPROM.ESP_READ_REG, struct.pack('<I', addr))
|
||||
if res[1] != "\0\0":
|
||||
raise Exception('Failed to read target memory')
|
||||
return res[0]
|
||||
|
||||
""" Write to memory address in target """
|
||||
def write_reg(self, addr, value, mask, delay_us = 0):
|
||||
if self.command(ESPROM.ESP_WRITE_REG,
|
||||
struct.pack('<IIII', addr, value, mask, delay_us))[1] != "\0\0":
|
||||
raise Exception('Failed to write target memory')
|
||||
|
||||
""" Start downloading an application image to RAM """
|
||||
def mem_begin(self, size, blocks, blocksize, offset):
|
||||
if self.command(ESPROM.ESP_MEM_BEGIN,
|
||||
struct.pack('<IIII', size, blocks, blocksize, offset))[1] != "\0\0":
|
||||
raise Exception('Failed to enter RAM download mode')
|
||||
|
||||
""" Send a block of an image to RAM """
|
||||
def mem_block(self, data, seq):
|
||||
if self.command(ESPROM.ESP_MEM_DATA,
|
||||
struct.pack('<IIII', len(data), seq, 0, 0)+data, ESPROM.checksum(data))[1] != "\0\0":
|
||||
raise Exception('Failed to write to target RAM')
|
||||
|
||||
""" Leave download mode and run the application """
|
||||
def mem_finish(self, entrypoint = 0):
|
||||
if self.command(ESPROM.ESP_MEM_END,
|
||||
struct.pack('<II', int(entrypoint == 0), entrypoint))[1] != "\0\0":
|
||||
raise Exception('Failed to leave RAM download mode')
|
||||
|
||||
""" Start downloading to Flash (performs an erase) """
|
||||
def flash_begin(self, size, offset):
|
||||
old_tmo = self._port.timeout
|
||||
num_blocks = (size + ESPROM.ESP_FLASH_BLOCK - 1) / ESPROM.ESP_FLASH_BLOCK
|
||||
self._port.timeout = 10
|
||||
if self.command(ESPROM.ESP_FLASH_BEGIN,
|
||||
struct.pack('<IIII', size, num_blocks, ESPROM.ESP_FLASH_BLOCK, offset))[1] != "\0\0":
|
||||
raise Exception('Failed to enter Flash download mode')
|
||||
self._port.timeout = old_tmo
|
||||
|
||||
""" Write block to flash """
|
||||
def flash_block(self, data, seq):
|
||||
if self.command(ESPROM.ESP_FLASH_DATA,
|
||||
struct.pack('<IIII', len(data), seq, 0, 0)+data, ESPROM.checksum(data))[1] != "\0\0":
|
||||
raise Exception('Failed to write to target Flash')
|
||||
|
||||
""" Leave flash mode and run/reboot """
|
||||
def flash_finish(self, reboot = False):
|
||||
pkt = struct.pack('<I', int(not reboot))
|
||||
if self.command(ESPROM.ESP_FLASH_END, pkt)[1] != "\0\0":
|
||||
raise Exception('Failed to leave Flash mode')
|
||||
|
||||
""" Run application code in flash """
|
||||
def run(self, reboot = False):
|
||||
# Fake flash begin immediately followed by flash end
|
||||
self.flash_begin(0, 0)
|
||||
self.flash_finish(reboot)
|
||||
|
||||
|
||||
class ESPFirmwareImage:
|
||||
|
||||
def __init__(self, filename = None):
|
||||
self.segments = []
|
||||
self.entrypoint = 0
|
||||
|
||||
if filename is not None:
|
||||
f = file(filename, 'rb')
|
||||
(magic, segments, _, _, self.entrypoint) = struct.unpack('<BBBBI', f.read(8))
|
||||
|
||||
# some sanity check
|
||||
if magic != ESPROM.ESP_IMAGE_MAGIC or segments > 16:
|
||||
raise Exception('Invalid firmware image')
|
||||
|
||||
for i in xrange(segments):
|
||||
(offset, size) = struct.unpack('<II', f.read(8))
|
||||
if offset > 0x40200000 or offset < 0x3ffe0000 or size > 65536:
|
||||
raise Exception('Suspicious segment %x,%d' % (offset, size))
|
||||
self.segments.append((offset, size, f.read(size)))
|
||||
|
||||
# Skip the padding. The checksum is stored in the last byte so that the
|
||||
# file is a multiple of 16 bytes.
|
||||
align = 15-(f.tell() % 16)
|
||||
f.seek(align, 1)
|
||||
|
||||
self.checksum = ord(f.read(1))
|
||||
|
||||
def add_segment(self, addr, data):
|
||||
# Data should be aligned on word boundary
|
||||
l = len(data)
|
||||
if l % 4:
|
||||
data += b"\x00" * (4 - l % 4)
|
||||
self.segments.append((addr, len(data), data))
|
||||
|
||||
def save(self, filename):
|
||||
f = file(filename, 'wb')
|
||||
f.write(struct.pack('<BBBBI', ESPROM.ESP_IMAGE_MAGIC, len(self.segments), 0, 0, self.entrypoint))
|
||||
|
||||
checksum = ESPROM.ESP_CHECKSUM_MAGIC
|
||||
for (offset, size, data) in self.segments:
|
||||
f.write(struct.pack('<II', offset, size))
|
||||
f.write(data)
|
||||
checksum = ESPROM.checksum(data, checksum)
|
||||
|
||||
align = 15-(f.tell() % 16)
|
||||
f.seek(align, 1)
|
||||
f.write(struct.pack('B', checksum))
|
||||
|
||||
|
||||
class ELFFile:
|
||||
|
||||
def __init__(self, name):
|
||||
self.name = name
|
||||
self.symbols = None
|
||||
|
||||
def _fetch_symbols(self):
|
||||
if self.symbols is not None:
|
||||
return
|
||||
self.symbols = {}
|
||||
try:
|
||||
proc = subprocess.Popen(["xtensa-lx106-elf-nm", self.name], stdout=subprocess.PIPE)
|
||||
except OSError:
|
||||
print "Error calling xtensa-lx106-elf-nm, do you have Xtensa toolchain in PATH?"
|
||||
sys.exit(1)
|
||||
for l in proc.stdout:
|
||||
fields = l.strip().split()
|
||||
self.symbols[fields[2]] = int(fields[0], 16)
|
||||
|
||||
def get_symbol_addr(self, sym):
|
||||
self._fetch_symbols()
|
||||
return self.symbols[sym]
|
||||
|
||||
def load_section(self, section):
|
||||
subprocess.check_call(["xtensa-lx106-elf-objcopy", "--only-section", section, "-Obinary", self.name, ".tmp.section"])
|
||||
f = open(".tmp.section", "rb")
|
||||
data = f.read()
|
||||
f.close()
|
||||
os.remove(".tmp.section")
|
||||
return data
|
||||
|
||||
|
||||
def arg_auto_int(x):
|
||||
return int(x, 0)
|
||||
|
||||
if __name__ == '__main__':
|
||||
parser = argparse.ArgumentParser(description = 'ESP8266 ROM Bootloader Utility', prog = 'esptool')
|
||||
|
||||
parser.add_argument(
|
||||
'--port', '-p',
|
||||
help = 'Serial port device',
|
||||
default = '/dev/ttyUSB0')
|
||||
|
||||
parser.add_argument(
|
||||
'--baud', '-b',
|
||||
help = 'Serial port baud rate',
|
||||
type = arg_auto_int,
|
||||
default = ESPROM.ESP_ROM_BAUD)
|
||||
|
||||
subparsers = parser.add_subparsers(
|
||||
dest = 'operation',
|
||||
help = 'Run esptool {command} -h for additional help')
|
||||
|
||||
parser_load_ram = subparsers.add_parser(
|
||||
'load_ram',
|
||||
help = 'Download an image to RAM and execute')
|
||||
parser_load_ram.add_argument('filename', help = 'Firmware image')
|
||||
|
||||
parser_dump_mem = subparsers.add_parser(
|
||||
'dump_mem',
|
||||
help = 'Dump arbitrary memory to disk')
|
||||
parser_dump_mem.add_argument('address', help = 'Base address', type = arg_auto_int)
|
||||
parser_dump_mem.add_argument('size', help = 'Size of region to dump', type = arg_auto_int)
|
||||
parser_dump_mem.add_argument('filename', help = 'Name of binary dump')
|
||||
|
||||
parser_read_mem = subparsers.add_parser(
|
||||
'read_mem',
|
||||
help = 'Read arbitrary memory location')
|
||||
parser_read_mem.add_argument('address', help = 'Address to read', type = arg_auto_int)
|
||||
|
||||
parser_write_mem = subparsers.add_parser(
|
||||
'write_mem',
|
||||
help = 'Read-modify-write to arbitrary memory location')
|
||||
parser_write_mem.add_argument('address', help = 'Address to write', type = arg_auto_int)
|
||||
parser_write_mem.add_argument('value', help = 'Value', type = arg_auto_int)
|
||||
parser_write_mem.add_argument('mask', help = 'Mask of bits to write', type = arg_auto_int)
|
||||
|
||||
parser_write_flash = subparsers.add_parser(
|
||||
'write_flash',
|
||||
help = 'Write a binary blob to flash')
|
||||
parser_write_flash.add_argument('addr_filename', nargs = '+', help = 'Address and binary file to write there, separated by space')
|
||||
|
||||
parser_run = subparsers.add_parser(
|
||||
'run',
|
||||
help = 'Run application code in flash')
|
||||
|
||||
parser_image_info = subparsers.add_parser(
|
||||
'image_info',
|
||||
help = 'Dump headers from an application image')
|
||||
parser_image_info.add_argument('filename', help = 'Image file to parse')
|
||||
|
||||
parser_make_image = subparsers.add_parser(
|
||||
'make_image',
|
||||
help = 'Create an application image from binary files')
|
||||
parser_make_image.add_argument('output', help = 'Output image file')
|
||||
parser_make_image.add_argument('--segfile', '-f', action = 'append', help = 'Segment input file')
|
||||
parser_make_image.add_argument('--segaddr', '-a', action = 'append', help = 'Segment base address', type = arg_auto_int)
|
||||
parser_make_image.add_argument('--entrypoint', '-e', help = 'Address of entry point', type = arg_auto_int, default = 0)
|
||||
|
||||
parser_elf2image = subparsers.add_parser(
|
||||
'elf2image',
|
||||
help = 'Create an application image from ELF file')
|
||||
parser_elf2image.add_argument('input', help = 'Input ELF file')
|
||||
parser_elf2image.add_argument('--output', '-o', help = 'Output filename prefix', type = str)
|
||||
|
||||
parser_read_mac = subparsers.add_parser(
|
||||
'read_mac',
|
||||
help = 'Read MAC address from OTP ROM')
|
||||
|
||||
args = parser.parse_args()
|
||||
|
||||
# Create the ESPROM connection object, if needed
|
||||
esp = None
|
||||
if args.operation not in ('image_info','make_image','elf2image'):
|
||||
esp = ESPROM(args.port, args.baud)
|
||||
esp.connect()
|
||||
|
||||
# Do the actual work. Should probably be split into separate functions.
|
||||
if args.operation == 'load_ram':
|
||||
image = ESPFirmwareImage(args.filename)
|
||||
|
||||
print 'RAM boot...'
|
||||
for (offset, size, data) in image.segments:
|
||||
print 'Downloading %d bytes at %08x...' % (size, offset),
|
||||
sys.stdout.flush()
|
||||
esp.mem_begin(size, math.ceil(size / float(esp.ESP_RAM_BLOCK)), esp.ESP_RAM_BLOCK, offset)
|
||||
|
||||
seq = 0
|
||||
while len(data) > 0:
|
||||
esp.mem_block(data[0:esp.ESP_RAM_BLOCK], seq)
|
||||
data = data[esp.ESP_RAM_BLOCK:]
|
||||
seq += 1
|
||||
print 'done!'
|
||||
|
||||
print 'All segments done, executing at %08x' % image.entrypoint
|
||||
esp.mem_finish(image.entrypoint)
|
||||
|
||||
elif args.operation == 'read_mem':
|
||||
print '0x%08x = 0x%08x' % (args.address, esp.read_reg(args.address))
|
||||
|
||||
elif args.operation == 'write_mem':
|
||||
esp.write_reg(args.address, args.value, args.mask, 0)
|
||||
print 'Wrote %08x, mask %08x to %08x' % (args.value, args.mask, args.address)
|
||||
|
||||
elif args.operation == 'dump_mem':
|
||||
f = file(args.filename, 'wb')
|
||||
for i in xrange(args.size/4):
|
||||
d = esp.read_reg(args.address+(i*4))
|
||||
f.write(struct.pack('<I', d))
|
||||
if f.tell() % 1024 == 0:
|
||||
print '\r%d bytes read... (%d %%)' % (f.tell(), f.tell()*100/args.size),
|
||||
sys.stdout.flush()
|
||||
print 'Done!'
|
||||
|
||||
elif args.operation == 'write_flash':
|
||||
assert len(args.addr_filename) % 2 == 0
|
||||
while args.addr_filename:
|
||||
address = int(args.addr_filename[0], 0)
|
||||
filename = args.addr_filename[1]
|
||||
args.addr_filename = args.addr_filename[2:]
|
||||
image = file(filename, 'rb').read()
|
||||
print 'Erasing flash...'
|
||||
blocks = math.ceil(len(image)/float(esp.ESP_FLASH_BLOCK))
|
||||
esp.flash_begin(blocks*esp.ESP_FLASH_BLOCK, address)
|
||||
seq = 0
|
||||
while len(image) > 0:
|
||||
print '\rWriting at 0x%08x... (%d %%)' % (address + seq*esp.ESP_FLASH_BLOCK, 100*(seq+1)/blocks),
|
||||
sys.stdout.flush()
|
||||
block = image[0:esp.ESP_FLASH_BLOCK]
|
||||
block = block + '\xe0' * (esp.ESP_FLASH_BLOCK-len(block))
|
||||
esp.flash_block(block, seq)
|
||||
image = image[esp.ESP_FLASH_BLOCK:]
|
||||
seq += 1
|
||||
print
|
||||
print '\nLeaving...'
|
||||
esp.flash_finish(False)
|
||||
|
||||
elif args.operation == 'run':
|
||||
esp.run()
|
||||
|
||||
elif args.operation == 'image_info':
|
||||
image = ESPFirmwareImage(args.filename)
|
||||
print ('Entry point: %08x' % image.entrypoint) if image.entrypoint != 0 else 'Entry point not set'
|
||||
print '%d segments' % len(image.segments)
|
||||
print
|
||||
checksum = ESPROM.ESP_CHECKSUM_MAGIC
|
||||
for (idx, (offset, size, data)) in enumerate(image.segments):
|
||||
print 'Segment %d: %5d bytes at %08x' % (idx+1, size, offset)
|
||||
checksum = ESPROM.checksum(data, checksum)
|
||||
print
|
||||
print 'Checksum: %02x (%s)' % (image.checksum, 'valid' if image.checksum == checksum else 'invalid!')
|
||||
|
||||
elif args.operation == 'make_image':
|
||||
image = ESPFirmwareImage()
|
||||
if len(args.segfile) == 0:
|
||||
raise Exception('No segments specified')
|
||||
if len(args.segfile) != len(args.segaddr):
|
||||
raise Exception('Number of specified files does not match number of specified addresses')
|
||||
for (seg, addr) in zip(args.segfile, args.segaddr):
|
||||
data = file(seg, 'rb').read()
|
||||
image.add_segment(addr, data)
|
||||
image.entrypoint = args.entrypoint
|
||||
image.save(args.output)
|
||||
|
||||
elif args.operation == 'elf2image':
|
||||
if args.output is None:
|
||||
args.output = args.input + '-'
|
||||
e = ELFFile(args.input)
|
||||
image = ESPFirmwareImage()
|
||||
image.entrypoint = e.get_symbol_addr("call_user_start")
|
||||
for section, start in ((".text", "_text_start"), (".data", "_data_start"), (".rodata", "_rodata_start")):
|
||||
data = e.load_section(section)
|
||||
image.add_segment(e.get_symbol_addr(start), data)
|
||||
image.save(args.output + "0x00000.bin")
|
||||
data = e.load_section(".irom0.text")
|
||||
off = e.get_symbol_addr("_irom0_text_start") - 0x40200000
|
||||
assert off >= 0
|
||||
f = open(args.output + "0x%05x.bin" % off, "wb")
|
||||
f.write(data)
|
||||
f.close()
|
||||
|
||||
elif args.operation == 'read_mac':
|
||||
mac0 = esp.read_reg(esp.ESP_OTP_MAC0)
|
||||
mac1 = esp.read_reg(esp.ESP_OTP_MAC1)
|
||||
print 'MAC: 18:fe:34:%02x:%02x:%02x' % ((mac1 >> 8) & 0xff, mac1 & 0xff, (mac0 >> 24) & 0xff)
|
Loading…
Reference in New Issue