APM32芯得EP.40|玩点不一样的2,写个能读取APM32F411内存的小程序

极海Geehy 2024-09-19 10:00


《APM32芯得》系列内容为用户使用APM32系列产品的经验总结,均转载自21ic论坛极海半导体专区,全文未作任何修改,未经原文作者授权禁止转载。




1 背景


之前用Python+pyocd配合APM32F411 tinyboard 板卡上的Geehylink对内存里面的数据进行读取然后再用波形描绘出来(可以看这里:https://bbs.21ic.com/icview-3344910-1-1.html)。完成这个之后我就想着是不是也可以直接读取APM32F411的内存里面的内容然后直接显示出来,想保存的时候就直接保存呢?这样子就不用一个个敲命令了(因为我懒,敲命令多了有点累O(∩_∩)O哈哈~)


说干就干,这里给大家分享一下我把我这个想法实现的一个小过程,权当抛砖引玉,给大家启迪思路。


2 技术选择


2.1 原型设计


基于我的功能,我对程序界面基本原型设计如下:



Byte、Halfword、Word是用来控制读取到数据显示长度的(参考优秀的J-Flash的设计)。


2.2 Python的GUI


读取APM32F411的内存里面的数据出来后,我们需要显示出来。我们可以利用Python自带的Tkinter图形用户界面库。当然我们也可以选择QT5的图形库来画UI,但是我为撒不用咧?因为我初学这个,想先挑简单的来进行。


Tkinter是Python的标准GUI库,它提供了丰富的组件和布局管理器,能够帮助我们快速地创建图形用户界面应用程序。它无需安装第三方库即可使用。QT5的话是需要安装其依赖库的,Tkinter,Python安装的时候就自带了。


虽然它有点简陋,但是提供了按钮、画布、条目、框架等基本组件,基本能够满足我现阶段的需求。网上有许多这个玩意的介绍及使用,这里我就不赘述,大家百度即可。


3 程序设计思路


3.1 获取连接的Geehy-Link设备


一个PC不会只连接一个仿真器,我这里想着能够在程序的界面上显示连接到PC的设备,并能够使用下拉框进行选择。

获取设备这里用的是:`ConnectHelper.get_all_connected_probes`,这是ConnectHelper类中的一个静态方法。它用于搜索并返回当前连接到计算机的所有调试探针(Debug Probes)列表(包括Jlink,DAPLink等)。


这里我封装一下: 


def update_device_list():

    probes = ConnectHelper.get_all_connected_probes()

    device_list = [probe.unique_id for probe in probes]

    device_cb['values'] = device_list

    if device_list:

        device_cb.current(0)

    else:

        device_cb.set('')


这样我们就可以`update_device_list()`来更新获取我们的设备列表。


3.2 读取数据的保存


由于我设计想着能够切换显示格式:可以选择字节、半字或字的显示方式。不能每次切换显示方式就读一次APM32F411的内存,这样会造成不必要的消耗。


我这里直接设计一个全局变量,用来保存读取到的数据,即相当于一个缓存,切换显示格式我们就直接对缓存里面的数据重新排列就好。 


# 全局缓存区,保存读取到的内容

memory_cache = {

    'data': None,

    'address': 0x08000000,

    'size': 0x1000,

}


当然我这里也一并保存了需要读取的起始地址和读取的数据长度。


3.3 进度的显示


读大块的内容时可能需要花费一些时间,这个我想着用一个小地方显示读取的进度以及另存为bin文件的结果显示。这里就涉及到了一个线程的操作。


1. UI界面做一个主线程

2. 读取数据的操作做一个线程,并且读取的过程中计算百分比,然后把百分比推送至UI界面进行显示。


为什么要这样操作?回到刚刚说的,读取大块内容花费时间较长,若读取数据和UI在一个线程,那就会造成读取的时候UI有点“卡卡”。


读取的线程操作如下: 


def read_memory_thread():

    try:

        # 更新标签显示读取正在进行(在主线程中执行)

        output_label.config(text="Read 0%")

        root.update_idletasks()  # 强制更新UI

        

        selected_probe = device_cb.get()

        address = int(address_entry.get() or "08000000", 16)

        size = int(size_entry.get() or "1000", 16)

        memory_cache['address'] = address

        memory_cache['size'] = size


        # 按块读取内存并更新百分比

        block_size = size // 20  # 计算每5%需要读取的大小

        if block_size == 0:

            block_size = size

        memory_cache['data'] = []

        with ConnectHelper.session_with_chosen_probe(unique_id=selected_probe) as session:

            target = session.board.target

            for i in range(0, size, block_size):

                end_address = i + block_size

                if end_address > size:

                    end_address = size

                memory_cache['data'] += target.read_memory_block8(address + i, end_address - i)

                # 计算并更新百分比

                percent_complete = (i + block_size) * 100 // size

                if percent_complete > 100:

                    percent_complete = 100

                root.after(0, lambda p=percent_complete: output_label.config(text=f"Read {p}%"))

                root.update_idletasks()  # 更新UI以显示百分比

        

        # 在主线程中更新UI显示读取成功

        root.after(0, lambda: output_label.config(text="Read successfully"))

        root.after(0, update_memory_display)

    except ValueError as e:

        root.after(0, lambda: output_label.config(text="Error: Please enter valid hexadecimal address and size."))

    except Exception as e:

        root.after(0, lambda: output_label.config(text=f"Error: {str(e)}"))


def read_memory():

    # 创建并启动后台线程进行内存读取

    threading.Thread(target=read_memory_thread, daemon=True).start()


3.4 保存到文件


这一步就比较简单,只是简单的文件的操作,需要注意的是,我们读取到的数据其实是16进制的,保存成bin文件需要是2进制的(需要用`bytes`转换一下)。 


def save_memory_to_file():

    if memory_cache['data'] is None:

        output_label.config(text="Error: No data to save. Please read memory first.")

        return


    file_path = filedialog.asksaveasfilename(

        defaultextension=".bin",

        filetypes=[("Binary files", "*.bin"), ("All files", "*.*")],

        title="Save memory as binary file"

    )


    if not file_path:

        # User cancelled the file dialog

        return


    with open(file_path, 'wb') as file:

        file.write(bytes(memory_cache['data']))

        output_label.config(text=f"Memory saved to {file_path} successfully")


文件的打开与保存就需要进行一些判断了:保存前判断一下有没有数据,用户是否取消保存等。


3.5 界面的设计


界面的设计主要是利用Tkinter控件,我们要清楚需要的组件:


1. 设备选择——下拉框;

2. 地址的设置、读取长度的设置——文本框;

3. 读取、另存为、显示切换——按钮;

4. 显示数据——视图控件;

5. 输出信息/提示内容等——标签;


然后考虑位置及大小等信息。


代码参考如下:


1. update_memory_display函数更新内存树视图控件,将读取的数据以十六进制和ASCII格式展示。

2. change_display_format函数允许用户改变内存数据的显示格式(字节、半字、字)。

3. GUI部分设置了窗口、框架、输入框、下拉列表、按钮和标签等控件。


def update_memory_display():

   memory_tree.delete(*memory_tree.get_children())

    display_format = display_format_var.get()


    if not memory_cache['data']:

        return


    read_data = memory_cache['data']

    bytes_per_row = 0x10

    num_cols = bytes_per_row >> {'byte': 0, 'halfword': 1, 'word': 2}[display_format]

    row_format = {'byte': '{:02X}', 'halfword': '{:04X}', 'word': '{:08X}'}[display_format]


    for i in range(0, len(read_data), bytes_per_row):

        row_data = read_data[i:i + bytes_per_row]

        ascii_representation = ''.join(chr(b) if 0x20 <= b <= 0x7E else '.' for b in row_data)

        if display_format == 'halfword':

            row_data = [int.from_bytes(row_data[j:j + 2], byteorder='little') for j in range(0, len(row_data), 2)]

        elif display_format == 'word':

            row_data = [int.from_bytes(row_data[j:j + 4], byteorder='little') for j in range(0, len(row_data), 4)]


        addr = f"{memory_cache['address'] + i:08X}"

        hex_data = ' '.join(row_format.format(byte).rjust(8 if display_format == 'word' else 5) for byte in row_data[:num_cols])

        memory_tree.insert('', 'end', text=addr, values=[hex_data, ascii_representation])


def change_display_format(new_format):

    display_format_var.set(new_format)

    update_memory_display()


# Set up the GUI

root = tk.Tk()

root.title("Memory Reader for APM32F411VC TinyBoard")

root.rowconfigure(1, weight=1)

root.columnconfigure(0, weight=1)


frame = ttk.Frame(root, padding="10")

frame.grid(row=0, column=0, sticky=(tk.W, tk.E, tk.N, tk.S))

frame.columnconfigure(1, weight=1)


# Device selection combobox

device_label = ttk.Label(frame, text="DAPlink Device:")

device_label.grid(row=0, column=0, sticky=tk.W)

device_cb = ttk.Combobox(frame, width=10, postcommand=update_device_list)

device_cb.grid(row=0, column=1, sticky=(tk.W, tk.E))


# 启动便获取一次支持的设备列表,

update_device_list()


# Address input entry with default placeholder text

address_label = ttk.Label(frame, text="Address (hex):")

address_label.grid(row=1, column=0, sticky=tk.W)

address_entry = ttk.Entry(frame, width=10)

address_entry.insert(0, "08000000")

# address_entry.grid(row=1, column=1, sticky=(tk.W, tk.E))


address_entry.grid(row=1, column=1, sticky=(tk.W, tk.E))


# Size input entry with default placeholder text

size_label = ttk.Label(frame, text="Size (hex):")

size_label.grid(row=1, column=2, sticky=tk.W)

size_entry = ttk.Entry(frame, width=10)

size_entry.insert(0, "1000")

size_entry.grid(row=1, column=3, sticky=(tk.W, tk.E))


# Read memory button

read_button = ttk.Button(frame, text="Read Memory", command=read_memory)

read_button.grid(row=1,  column=4, columnspan=2)


# Output label for messages

frame.rowconfigure(2, minsize=10)

frame.rowconfigure(4, minsize=10)


output_label = ttk.Label(frame, text="",anchor="e")

output_label.grid(row=3, column=0,sticky="we", columnspan=10)


# Save memory to file button

save_button = ttk.Button(frame, text="Save as", command=save_memory_to_file)

save_button.grid(row=1, column=7, columnspan=2, sticky=(tk.E, tk.W))


# Display format buttons

display_format_var = tk.StringVar(value='byte')

formats_frame = ttk.Frame(frame)

formats_frame.grid(row=5, column=0 , columnspan=20,sticky="e")

byte_btn = ttk.Button(formats_frame, text="Byte View", command=lambda: change_display_format('byte'))

byte_btn.pack(side=tk.LEFT, padx=5)

halfword_btn = ttk.Button(formats_frame, text="Halfword View", command=lambda: change_display_format('halfword'))

halfword_btn.pack(side=tk.LEFT, padx=5)

word_btn = ttk.Button(formats_frame, text="Word View", command=lambda: change_display_format('word'))

word_btn.pack(side=tk.LEFT, padx=5)


# Treeview widget for memory display with an extra ASCII column

memory_tree = ttk.Treeview(root, columns=('data', 'ascii'), show='tree headings')

memory_tree.heading('data', text='Data (Hexadecimal)')

memory_tree.heading('ascii', text='ASCII')

memory_tree.column('data', width=300, stretch=True)

memory_tree.column('ascii', width=150, stretch=True)

memory_tree.grid(row=1, column=0, sticky=(tk.W, tk.E, tk.N, tk.S))


# Make sure the data is displayed in a monospaced font

style = ttk.Style()

style.configure('Treeview', font=('Courier', 10))


# Scrollbar for the Treeview widget

scrollbar = ttk.Scrollbar(root, orient='vertical', command=memory_tree.yview)

scrollbar.grid(row=1, column=1, sticky=(tk.N, tk.S))

memory_tree.configure(yscrollcommand=scrollbar.set)


4 运行效果


完成以上代码后,我们点击运行。


读取并保存:



读取GPIOE的寄存器:



5 总结


这个程序也是心血来潮,一边学习一边完成的,里面少考虑了很多异常情况或者需求:


1. 支持的芯片读取区域的校验,超越区域需要保持;

2. 保存成hex、s19等文件;

3. 支持芯片选型。 



注:文章作者在原帖中提供了工程文件,有需要请至原文21ic论坛下载


原文地址:https://bbs.21ic.com/icview-3345102-1-1.html

或点击下方 阅读原文 跳转


↑↑↑ 点击上方卡片关注极海 ↑↑↑

极海Geehy 极海半导体是一家致力于开发工业级/车规级MCU、模拟与混合信号IC及系统级芯片的集成电路设计型企业
评论
  • PNT、GNSS、GPS均是卫星定位和导航相关领域中的常见缩写词,他们经常会被用到,且在很多情况下会被等同使用或替换使用。我们会把定位导航功能测试叫做PNT性能测试,也会叫做GNSS性能测试。我们会把定位导航终端叫做GNSS模块,也会叫做GPS模块。但是实际上他们之间是有一些重要的区别。伴随着技术发展与越发深入,我们有必要对这三个词汇做以清晰的区分。一、什么是GPS?GPS是Global Positioning System(全球定位系统)的缩写,它是美国建立的全球卫星定位导航系统,是GNSS概
    德思特测试测量 2025-01-13 15:42 469浏览
  • 随着全球向绿色能源转型的加速,对高效、可靠和环保元件的需求从未如此强烈。在这种背景下,国产固态继电器(SSR)在实现太阳能逆变器、风力涡轮机和储能系统等关键技术方面发挥着关键作用。本文探讨了绿色能源系统背景下中国固态继电器行业的前景,并强调了2025年的前景。 1.对绿色能源解决方案日益增长的需求绿色能源系统依靠先进的电源管理技术来最大限度地提高效率并最大限度地减少损失。固态继电器以其耐用性、快速开关速度和抗机械磨损而闻名,正日益成为传统机电继电器的首选。可再生能源(尤其是太阳能和风能
    克里雅半导体科技 2025-01-10 16:18 319浏览
  • 新年伊始,又到了对去年做总结,对今年做展望的时刻 不知道你在2024年初立的Flag都实现了吗? 2025年对自己又有什么新的期待呢? 2024年注定是不平凡的一年, 一年里我测评了50余块开发板, 写出了很多科普文章, 从一个小小的工作室成长为科工公司。 展望2025年, 中国香河英茂科工, 会继续深耕于,具身机器人、飞行器、物联网等方面的研发, 我觉得,要向未来学习未来, 未来是什么? 是掌握在孩子们生活中的发现,和精历, 把最好的技术带给孩子,
    丙丁先生 2025-01-11 11:35 447浏览
  • Snyk 是一家为开发人员提供安全平台的公司,致力于协助他们构建安全的应用程序,并为安全团队提供应对数字世界挑战的工具。以下为 Snyk 如何通过 CircleCI 实现其“交付”使命的案例分析。一、Snyk 的挑战随着客户对安全工具需求的不断增长,Snyk 的开发团队面临多重挑战:加速交付的需求:Snyk 的核心目标是为开发者提供更快、更可靠的安全解决方案,但他们的现有 CI/CD 工具(TravisCI)运行缓慢,无法满足快速开发和部署的要求。扩展能力不足:随着团队规模和代码库的不断扩大,S
    艾体宝IT 2025-01-10 15:52 162浏览
  • 随着数字化的不断推进,LED显示屏行业对4K、8K等超高清画质的需求日益提升。与此同时,Mini及Micro LED技术的日益成熟,推动了间距小于1.2 Pitch的Mini、Micro LED显示屏的快速发展。这类显示屏不仅画质卓越,而且尺寸适中,通常在110至1000英寸之间,非常适合应用于电影院、监控中心、大型会议、以及电影拍摄等多种室内场景。鉴于室内LED显示屏与用户距离较近,因此对于噪音控制、体积小型化、冗余备份能力及电气安全性的要求尤为严格。为满足这一市场需求,开关电源技术推出了专为
    晶台光耦 2025-01-13 10:42 487浏览
  • 根据Global Info Research(环洋市场咨询)项目团队最新调研,预计2030年全球无人机电池和电源产值达到2834百万美元,2024-2030年期间年复合增长率CAGR为10.1%。 无人机电池是为无人机提供动力并使其飞行的关键。无人机使用的电池类型因无人机的大小和型号而异。一些常见的无人机电池类型包括锂聚合物(LiPo)电池、锂离子电池和镍氢(NiMH)电池。锂聚合物电池是最常用的无人机电池类型,因为其能量密度高、设计轻巧。这些电池以输出功率大、飞行时间长而著称。不过,它们需要
    GIRtina 2025-01-13 10:49 167浏览
  •   在信号处理过程中,由于信号的时域截断会导致频谱扩展泄露现象。那么导致频谱泄露发生的根本原因是什么?又该采取什么样的改善方法。本文以ADC性能指标的测试场景为例,探讨了对ADC的输出结果进行非周期截断所带来的影响及问题总结。 两个点   为了更好的分析或处理信号,实际应用时需要从频域而非时域的角度观察原信号。但物理意义上只能直接获取信号的时域信息,为了得到信号的频域信息需要利用傅里叶变换这个工具计算出原信号的频谱函数。但对于计算机来说实现这种计算需要面对两个问题: 1.
    TIAN301 2025-01-14 14:15 100浏览
  • 说到福特,就要从亨利·福特(Henry Ford)这个人物说起。在发明大王爱迪生的电气工厂担任工程师的福特下班后,总是在自家仓库里努力研究和开发汽车。1896年,福特终于成功制造出一辆三轮车,开启了福特汽车的传奇。最初几年,福特都是独自制造汽车并同时进行销售。 (今天很多人都知道的精益管理中的5S方法,或多或少地受到了福特 CANDO方法的影响。)1903年,福特从牧师、律师、银行家、会计师等十一位股东那里筹集了十万美元,并在自家庭院成立了美国第五百零三家汽车公司——福特汽车公司(Fo
    优思学院 2025-01-10 11:21 51浏览
  • 电动汽车(EV)正在改变交通运输,为传统内燃机提供更清洁、更高效的替代方案。这种转变的核心是电力电子和能源管理方面的创新,而光耦合器在其中发挥着关键作用。这些不起眼的组件可实现可靠的通信、增强安全性并优化电动汽车系统的性能,使其成为正在进行的革命中不可或缺的一部分。光耦合器,也称为光隔离器,是一种使用光传输电信号的设备。通过隔离高压和低压电路,光耦合器可确保安全性、减少干扰并保持信号完整性。这些特性对于电动汽车至关重要,因为精确控制和安全性至关重要。 光耦合器在电动汽车中的作用1.电池
    腾恩科技-彭工 2025-01-10 16:14 72浏览
  • 流量传感器是实现对燃气、废气、生活用水、污水、冷却液、石油等各种流体流量精准计量的关键手段。但随着工业自动化、数字化、智能化与低碳化进程的不断加速,采用传统机械式检测方式的流量传感器已不能满足当代流体计量行业对于测量精度、测量范围、使用寿命与维护成本等方面的精细需求。流量传感器的应用场景(部分)超声波流量传感器,是一种利用超声波技术测量流体流量的新型传感器,其主要通过发射超声波信号并接收反射回来的信号,根据超声波在流体中传播的时间、幅度或相位变化等参数,间接计算流体的流量,具有非侵入式测量、高精
    华普微HOPERF 2025-01-13 14:18 465浏览
  • 在不断发展的电子元件领域,继电器——作为切换电路的关键设备,正在经历前所未有的技术变革。固态继电器(SSR)和机械继电器之间的争论由来已久。然而,从未来发展的角度来看,固态继电器正逐渐占据上风。本文将从耐用性、速度和能效三个方面,全面剖析固态继电器为何更具优势,并探讨其在行业中的应用与发展趋势。1. 耐用性:经久耐用的设计机械继电器:机械继电器依靠物理触点完成电路切换。然而,随着时间的推移,这些触点因电弧、氧化和材料老化而逐渐磨损,导致其使用寿命有限。因此,它们更适合低频或对切换耐久性要求不高的
    腾恩科技-彭工 2025-01-10 16:15 97浏览
  • 01. 什么是过程能力分析?过程能力研究利用生产过程中初始一批产品的数据,预测制造过程是否能够稳定地生产符合规格的产品。可以把它想象成一种预测。通过历史数据的分析,推断未来是否可以依赖该工艺持续生产高质量产品。客户可能会要求将过程能力研究作为生产件批准程序 (PPAP) 的一部分。这是为了确保制造过程能够持续稳定地生产合格的产品。02. 基本概念在定义制造过程时,目标是确保生产的零件符合上下规格限 (USL 和 LSL)。过程能力衡量制造过程能多大程度上稳定地生产符合规格的产品。核心概念很简单:
    优思学院 2025-01-12 15:43 499浏览
  • 随着通信技术的迅速发展,现代通信设备需要更高效、可靠且紧凑的解决方案来应对日益复杂的系统。中国自主研发和制造的国产接口芯片,正逐渐成为通信设备(从5G基站到工业通信模块)中的重要基石。这些芯片凭借卓越性能、成本效益及灵活性,满足了现代通信基础设施的多样化需求。 1. 接口芯片在通信设备中的关键作用接口芯片作为数据交互的桥梁,是通信设备中不可或缺的核心组件。它们在设备内的各种子系统之间实现无缝数据传输,支持高速数据交换、协议转换和信号调节等功能。无论是5G基站中的数据处理,还是物联网网关
    克里雅半导体科技 2025-01-10 16:20 436浏览
  • ARMv8-A是ARM公司为满足新需求而重新设计的一个架构,是近20年来ARM架构变动最大的一次。以下是对ARMv8-A的详细介绍: 1. 背景介绍    ARM公司最初并未涉足PC市场,其产品主要针对功耗敏感的移动设备。     随着技术的发展和市场需求的变化,ARM开始扩展到企业设备、服务器等领域,这要求其架构能够支持更大的内存和更复杂的计算任务。 2. 架构特点    ARMv8-A引入了Execution State(执行状
    丙丁先生 2025-01-12 10:30 457浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦