Keil一键添加.c文件和头文件路径脚本--可遍历添加整个文件夹

news2024/11/29 1:34:17

最近想移植个LVGL玩玩,发现文件实在是太多了,加的手疼都没搞完,实在不想搞了就去找脚本和工具,基本没找到一个。。。。。。

主要是自己也懒得去研究写脚本,偶然搜到了一个博主写的脚本,原博客地址:https://blog.csdn.net/riyue2044/article/details/139424599

但是有以下问题:

1.这个脚本的.h文件也加在了分组下面,这样一般是不对的,应该加在Target的C/C++的Include path里面

2.脚本没有重复添加检测,导致如果多次添加,会损坏工程文件

3.输入是命令行式的,使用者可能会忘了参数具体设置

之前没接触过XML,python也不熟,所以研究了一下,做以下修改

1.把之前的命令行式的输入改为先运行再输入,会提示具体的参数设置,有默认参数,是以我的工程包来写的

2.把.h文件路径直接加在了Target的C/C++的Include path里面

3.加入文件路径检测,重复添加不会导致文件损坏

4.加入更多提示

5.加入三种模式    0:.c文件和.h路径会一起添加 1:只加.c文件 2:只加.h路径

使用方法:需要安装python,或者用python打包成exe文件也可,命令参考:pyinstaller -F -i .\icon.ico .\keil_add_file.py,放个百度云的链接,里面有我打包好的,不过注意杀毒软件估计会报毒,请添加信任

链接:https://pan.baidu.com/s/1zC7kVboAtQwHZ2Zy5RFmIw?pwd=arzd
提取码:arzd

脚本需放在keil工程目录,需要添加的目录则以相对路径填充,比如"../../../external/lvgl",需要注意的是分组需要提前在keil里面创建好,这个懒得改了,有需要的朋友可以自行修改

脚本内容如下:

import os
import glob
import xml.etree.ElementTree as ET
import argparse

from multiprocessing import Event

def indent(elem, level=0):
    """ Helper function to indent the XML for pretty printing. """
    i = "\n" + level * "    "
    if len(elem):
        if not elem.text or not elem.text.strip():
            elem.text = i + "    "
        if not elem.tail or not elem.tail.strip():
            elem.tail = i
        for elem in elem:
            indent(elem, level + 1)
        if not elem.tail or not elem.tail.strip():
            elem.tail = i
    else:
        if level and (not elem.tail or not elem.tail.strip()):
            elem.tail = i
        if not elem.tail:
            elem.tail = "\n"

def add_files_to_group(uvprojx_file_path, mode,folder_path, group_name_target):
    # 改变文件扩展名从 .uvprojx 到 .xml
    base, ext = os.path.splitext(uvprojx_file_path)
    if ext != '.uvprojx':
        print("工程文件扩展名不正确")
        return

    xml_path = base + '.xml'
    os.rename(uvprojx_file_path, xml_path)

    try:
        #解析XML文件
        tree = ET.parse(xml_path)
        #获取根节点
        root = tree.getroot()

        if mode == 0 or mode == 1:
            # 找到指定GroupName的Group节点
            target_group = None
            for group in root.findall('.//Group'):
                group_name = group.find('GroupName')
                if group_name is not None and group_name.text == group_name_target:
                    target_group = group
                    break

            if target_group is None:
                print(f"未发现 '{group_name_target}' 分组,请先创建分组后再尝试")
                # 将文件扩展名改回 .uvprojx
                os.rename(xml_path, uvprojx_file_path)
                return

            # 找到目标 Group 节点下的 Files 节点,如果不存在则创建一个
            files_node = target_group.find('Files')
            if files_node is None:
                files_node = ET.SubElement(target_group, 'Files')

        #寻找头文件分组
        if mode == 0 or mode == 2:
            print("寻找头文件分组......")
            target_header = None
            heard_inc = None
            target_header = root.find('.//Cads')
            if target_header == None:
                print("未发现头文件分组Cads")
                return
            else:
                heard_inc = target_header.find('VariousControls')
                if heard_inc == None:
                    print("未发现头文件分组VariousControls")
                    return
                else:
                    heard_inc = heard_inc.find('IncludePath')
                    if heard_inc == None:
                        print("未发现头文件分组IncludePath")
                        return
                    else:
                        print("找到头文件分组")



        #下面没有节点
        if mode == 0 or mode == 1:
            creat_dot = 0 #是否需要创建节点标志,如果有重复则跳过
            init_creat = 0
            file_init = files_node.find('File')
            if file_init == None:
                creat_dot = 1
                init_creat = 1
                print("初始节点为空需要创建节点")

        # 遍历指定文件夹,查找所有 .c 文件
        if mode == 0 or mode == 2:
            #print(heard_inc.text)
            heard_data = heard_inc.text + ";"   #末尾需要先加一个分号

        for subdir, _, files in os.walk(folder_path):
            #.h路径
            if mode == 0 or mode == 2:
                dir_path = os.path.relpath(subdir, start=os.path.dirname(xml_path))
                if dir_path in heard_inc.text:
                    print("需要添加的头文件路径已存在本次跳过")
                else:
                    heard_data = heard_data + dir_path + ";"

            #.c添加到分组
            if mode == 0 or mode == 1:
                for file in files:
                    if file.endswith('.c'):
                        # 计算相对路径
                        file_path = os.path.relpath(os.path.join(subdir, file), start=os.path.dirname(xml_path))
                        #print("路径",file_path)
                        file_check = files_node.findall('File')
                        if init_creat == 0:
                            #print("长度",len(file_check))
                            #遍历当前分组下的节点,检测是否已经包含了该路径,如果有直接跳过
                            for i in range(len(file_check)):
                                if file_path in file_check[i].find("FilePath").text:
                                    print("节点已存在本次跳过")
                                    creat_dot = 0
                                    break
                                else:
                                    if i == len(file_check) - 1:
                                        creat_dot = 1
                                        print("节点不存在,创建节点")
                                    else:
                                       creat_dot = 0
                                    continue
                        if creat_dot == 1:
                            # 创建 File 节点并添加到 Files 节点下
                            file_node = ET.SubElement(files_node, 'File')
                            file_name_node = ET.SubElement(file_node, 'FileName')
                            file_name_node.text = file
                            file_type_node = ET.SubElement(file_node, 'FileType')
                            file_type_node.text = '1'  # .c 文件类型都为 1

                            file_path_node = ET.SubElement(file_node, 'FilePath')
                            file_path_node.text = file_path
                            creat_dot = 0
                            init_creat = 0

        if mode == 0 or mode == 2:
            heard_data = heard_data.rstrip(";") #移除最后一个多加的;
            heard_inc.text = heard_data
            #print(heard_inc.text)

        # 格式化 XML
        indent(root)

        # 保存修改后的 XML 文件
        tree.write(xml_path, encoding='utf-8', xml_declaration=True)
        print("已完成")

    except ET.ParseError as e:
        print(f"ParseError: {e}")
        with open(xml_path, 'r', encoding='utf-8') as file:
            lines = file.readlines()
            start = max(0, e.position[0] - 5)
            end = min(len(lines), e.position[0] + 5)
            print("Context around the error:")
            for i in range(start, end):
                print(f"{i+1}: {lines[i].strip()}")

    finally:
        # 将文件扩展名改回 .uvprojx
        os.rename(xml_path, uvprojx_file_path)

#寻找工程文件
def find_uvprojx_file():
    uvprojx_files = glob.glob("*.uvprojx")
    if not uvprojx_files:
        print("未找到工程文件,请把此文件放在keil工程目录下")
        return None
    elif len(uvprojx_files) > 1:
        print("在当前目录中找到多个.uvprojx文件:")
        for i, file in enumerate(uvprojx_files, start=1):
            print(f"{i}. {file}")
        print("请确保目录中只有一个.uvprojx文件")
        return None
    else:
        return uvprojx_files[0]

if __name__ == "__main__":
    print("keil一键添加文件和头文件路径脚本\n\
    需放在keil工程同级目录下\n\
    参数格式,参数用空格隔开\n\
    默认模式:0\n\
    默认路径:\"../../../external/lvgl\"\n\
    默认分组:\"lvgl\"\n\
    1.添加模式 0:全部添加(.c文件全添加到分组.h文件夹加入include路径里) 1:只添加.c文件到分组 2:只添加.h文件夹到include里\n\
    2.要添加的文件夹路径,请使用相对路径\n\
    3.要添加的分组名称,如果没有分组需要先去keil手动添加分组\n")

    param = input("请输入参数:")

    if param:
        #print(param)
        args = param.split()
        args[0] = int(args[0])
        print(args)
    else:
        args = [0,"../../../external/lvgl","lvgl"]
        print("使用默认参数:",args)
    uvprojx_file_path = find_uvprojx_file()
    if uvprojx_file_path:
        add_files_to_group(uvprojx_file_path, args[0],args[1],args[2])

    event = Event()
    event.wait()

复制代码

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1815759.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

前端传递bool型后端用int收不到

文章目录 背景模拟错误点解决方法 背景 我前几天遇到一个低级错误,就是我前端发一个请求,把参数送到后端,但是我参数里面无意间传的布尔型(刚开始一直没注意到,因为当时参数有十几个),但是我后…

CMS与AI的融合:构建万能表单小程序系统

引言: 随着人工智能技术的飞速发展,MyCMS作为一款功能强大的内容管理系统,通过集成AI技术,进一步拓展了其应用范围和智能化水平。本文将探讨如何利用MyCMS结合AI技术,构建一个能够将用户提交的万能表单数据转化为智能提…

【Nginx系列】基于请求头的分发

文章目录 一、HTTP请求头和响应头二、基于请求头的分发2.1、基于host分发2.2、基于域名的分发测试:2.3、基于开发语言分发2.4、基于浏览器分发2.5、基于源IP分发 🌈你好呀!我是 山顶风景独好 🎈欢迎踏入我的博客世界,能…

上新即爆品?2024小红书爆款黄金公式

5月,小红书正式上线了平台级新品营销IP——“宝藏新品”,旨在消费愈发审慎的当下,帮助品牌破除不确定性,达成新品的高质量生长。 本期千瓜将进一步解读「宝藏新品」策略,帮助品牌推新呈现更多样化的成长可能。 强种草…

PyTorch中配置CUDA

第零部分: 首先查看CUDA版本 打开cmd,输入:nvidia-smi 可以看到CUDA的版本,后面安装的pytorch相关的库不高于这个版本的就行。 这里CUDA的版本是12.5,但是由于pytorch目前最高支持的cuda版本是12.1(在下…

Spring Security实现用户认证四:使用JWT与Redis实现无状态认证

Spring Security实现用户认证四:使用JWT与Redis实现无状态认证 1 什么是无状态认证?2 什么是JWT?2.1 需要注意的事项2.2 JWT构成 3 Spring Security JWT实现无状态认证3.1 创建一个Spring Boot项目3.1.1 依赖3.1.2 Main3.1.3 application.ym…

劝你千万别碰CSGO/Steam游戏搬砖

添加链接描述 CSGO游戏搬砖这个项目,大家也不陌生了,所有人都在夸它如何如何好,如何如何赚钱,今天我们客观一点,不偏不倚,来聊聊CSGO游戏搬砖的弊端。 首先,这个项目最大的弊端就是不太适合大部…

pikachu靶场上的暴力破解

目录 一、暴力破解 基于表单的暴力破解 验证码绕过(on server) ​编辑 验证码绕过(on client) ​编辑 token防爆破? 二、暴力破解的相关知识点 (1)Burte Force(暴力破解)概述 (2)验证码的绕过原理 【验证码机制原理】 【客户端可能存在的安全…

debian系统apt 国内安装源

debian系统apt 国内安装源: 国内阿里镜像源: deb http://mirrors.aliyun.com/debian stable main non-free contrib deb-src http://mirrors.aliyun.com/debian stable main non-free contrib 打开源文件位置:/etc/apt/sources.list,原来的内…

证照之星 XE版软件最新版下载及详细安装教程

简介: 全新的证照之星XE版本内置500证件照规格,100套服装素材,新增AI算法,一键处理更智能,轻松将生活照变成证件照。 ​安 装 包 获 取 地 址: 证照之星 XE版:​​https://souurl.cn/4fuRmg​…

学习笔记——路由网络基础——路由度量值

3、路由度量值 (1)基本概念 路由度量值表示到达这条路由所指目的地址的代价。度量值数值越小越优先,度量值最小路由将会被添加到路由表中。度量值很多时候被称为开销(Cost)。 路由度量(路由开销 cost)对于同一个路由协议,当到达某目标网段有多条路由供…

海洋CMS admin_notify.php 远程代码执行漏洞复现(CVE-2024-30565)

0x01 产品简介 海洋CMS是一套专为不同需求的站长而设计的内容管理系统,灵活、方便、人性化设计、简单易用是最大的特色,可快速建立一个海量内容的专业网站。海洋CMS基于PHP+MySql技术开发,完全开源免费 、无任何加密代码。 0x02 漏洞概述 海洋CMS admin_notify.php 接口处…

Android本地Gradle Plugin的创建以及使用

有些Gradle插件,不想放到云端,本来也只是小功能而已,还放到云端,每次修改和发布都很麻烦,这种需求的插件放到本地还是合适的。 1.直接放到build.gradle 2.新建一个module 取名叫buildSrc(注意,一定要叫这个…

ATA-2088高压放大器在细胞分选中的作用是什么

细胞分选是生物医学研究中至关重要的一步,它允许科学家们从混合细胞群中分离出特定类型的细胞,以进行进一步的研究。高压放大器在细胞分选中发挥着关键作用,其主要任务是处理和放大细胞产生的电信号,使得这些信号能够被准确地检测…

全国产的香橙派 AIpro AI开发硬件,对标nvidia jetson硬件。太香了

1. 导语 国产化AI芯片的发展不仅反映了中国在信息技术产业的战略布局,更是对全球科技竞争格局的积极回应。随着人工智能与大数据技术的飞速进步,国内对高性能计算的需求日益增长,促使科研机构与高新技术企业加大对AI芯片研发的投入&#xff0…

vscode编辑器警告, jsconfig.json 第一行红色波浪线

问题 配置文件jsconfig.json 第一行总是有个,红色下划线 项目是可以正常运行的,就是编辑器有个红色提示看着不顺眼。 开启检查 解决红色波浪线问题

算法体系-20 第二十节暴力递归到动态规划

前言 动态规划模型从尝试暴力递归到傻缓存到动态规划 四种模型和体系班两种模型一共六种模型 0.1 从左往右模型 0.2 范围讨论模型范围尝试模型 (这种模型特别在乎讨论开头如何如何 结尾如何如何) 玩家博弈问题,玩家玩纸牌只能那左或者右 0.3 …

MapperStruct拷贝数据的介绍和使用

1、前言 在java 编程中,对象直接拷贝是很常用的方法,最初我们常用spring提供的拷贝工具BeanUtils的copyProperties方法完成对象之间属性的拷贝。但是它有几个明显的如下缺点 1、属性类型不一致导致摸一个属性值拷贝失败 2、通一个字段使用基本类型和包…

2024年高考志愿填报,计算机相关专业还值得选择吗?

一、引言 在科技日新月异的今天,计算机专业无疑是推动社会进步和经济发展的重要力量。从最初的简单数据处理到如今的云计算、大数据、人工智能等前沿技术,计算机专业的发展速度之快、影响之广,已经深入到我们生活的方方面面。本文将详细分析…

Qt程序打包成单个exe文件

文章目录 0. 准备工作1. 使用 windeployqt 提取必要的动态链接库和资源文件1.1 操作步骤1.2 补充 2. 使用 Enigma Virtual Box将文件夹打包成单个exe2.1 操作步骤 0. 准备工作 Qt程序打包用到的工具有: windeployqt :安装Qt时自带Enigma Virtual Box 下…