|
@@ -21,13 +21,14 @@
|
|
|
# Date Author Notes
|
|
|
# 2017-10-04 Bernard The first version
|
|
|
# 2025-01-07 ZhaoCake components copy and gen doc
|
|
|
+# 2025-03-02 ZhaoCake Add MkDist_Strip
|
|
|
+
|
|
|
|
|
|
import os
|
|
|
import subprocess
|
|
|
import shutil
|
|
|
from shutil import ignore_patterns
|
|
|
from SCons.Script import *
|
|
|
-import time
|
|
|
|
|
|
def do_copy_file(src, dst):
|
|
|
# check source file
|
|
@@ -42,6 +43,7 @@ def do_copy_file(src, dst):
|
|
|
shutil.copy2(src, dst)
|
|
|
|
|
|
def do_copy_folder(src_dir, dst_dir, ignore=None):
|
|
|
+ import shutil
|
|
|
# check source directory
|
|
|
if not os.path.exists(src_dir):
|
|
|
return
|
|
@@ -94,7 +96,7 @@ def walk_kconfig(RTT_ROOT, source_list):
|
|
|
def bsp_copy_files(bsp_root, dist_dir):
|
|
|
# copy BSP files
|
|
|
do_copy_folder(os.path.join(bsp_root), dist_dir,
|
|
|
- ignore_patterns('build','__pycache__','dist', '*.pyc', '*.old', '*.map', 'rtthread.bin', '.sconsign.dblite', '*.elf', '*.axf', 'cconfig.h'))
|
|
|
+ ignore_patterns('build', '__pycache__', 'dist', '*.pyc', '*.old', '*.map', 'rtthread.bin', '.sconsign.dblite', '*.elf', '*.axf', 'cconfig.h'))
|
|
|
|
|
|
def bsp_update_sconstruct(dist_dir):
|
|
|
with open(os.path.join(dist_dir, 'SConstruct'), 'r') as f:
|
|
@@ -172,326 +174,6 @@ def zip_dist(dist_dir, dist_name):
|
|
|
|
|
|
zip.close()
|
|
|
|
|
|
-def parse_components_from_config(config_file):
|
|
|
- """Parse enabled components from .config file"""
|
|
|
- enabled_components = set()
|
|
|
-
|
|
|
- if not os.path.exists(config_file):
|
|
|
- print(f"Error: {config_file} does not exist")
|
|
|
- return enabled_components
|
|
|
-
|
|
|
- with open(config_file, 'r') as f:
|
|
|
- for line in f:
|
|
|
- line = line.strip()
|
|
|
- # Skip empty lines and comments
|
|
|
- if not line or line.startswith('#'):
|
|
|
- continue
|
|
|
-
|
|
|
- if line.startswith('CONFIG_'):
|
|
|
- if '=' in line:
|
|
|
- config = line.split('=')[0][7:] # Remove CONFIG_ prefix
|
|
|
- if config.startswith('RT_USING_'):
|
|
|
- component = config[9:].lower() # Remove RT_USING_ prefix
|
|
|
- enabled_components.add(component)
|
|
|
- return enabled_components
|
|
|
-
|
|
|
-def scan_components_dir(RTT_ROOT):
|
|
|
- """Scan component directory structure and generate component mapping"""
|
|
|
- components_map = {}
|
|
|
- components_root = os.path.join(RTT_ROOT, 'components')
|
|
|
-
|
|
|
- def parse_kconfig(kconfig_file):
|
|
|
- """Parse configuration options from Kconfig file"""
|
|
|
- components = set()
|
|
|
- try:
|
|
|
- with open(kconfig_file, 'r') as f:
|
|
|
- content = f.read()
|
|
|
- # Find configurations in the form of config RT_USING_XXX
|
|
|
- import re
|
|
|
- matches = re.finditer(r'config\s+RT_USING_(\w+)', content)
|
|
|
- for match in matches:
|
|
|
- component_name = match.group(1).lower()
|
|
|
- components.add(component_name)
|
|
|
- except Exception as e:
|
|
|
- print(f"Warning: Failed to parse {kconfig_file}: {str(e)}")
|
|
|
- return components
|
|
|
-
|
|
|
- def get_relative_path(full_path):
|
|
|
- """Get path relative to RTT_ROOT"""
|
|
|
- rel_path = os.path.relpath(os.path.dirname(full_path), RTT_ROOT)
|
|
|
- # Skip if path is directly under components directory
|
|
|
- if rel_path == 'components' or rel_path == os.path.join('components', ''):
|
|
|
- return None
|
|
|
- return rel_path
|
|
|
-
|
|
|
- # Scan all component directories
|
|
|
- for root, dirs, files in os.walk(components_root):
|
|
|
- if 'Kconfig' in files:
|
|
|
- kconfig_path = os.path.join(root, 'Kconfig')
|
|
|
- component_configs = parse_kconfig(kconfig_path)
|
|
|
- rel_path = get_relative_path(kconfig_path)
|
|
|
-
|
|
|
- # Only add component if it has a valid sub-path
|
|
|
- if rel_path:
|
|
|
- for comp_name in component_configs:
|
|
|
- components_map[comp_name] = rel_path
|
|
|
-
|
|
|
- return components_map
|
|
|
-
|
|
|
-def get_component_path(component_name, RTT_ROOT):
|
|
|
- """Get actual path of component"""
|
|
|
- # Get dynamic component mapping
|
|
|
- dynamic_map = scan_components_dir(RTT_ROOT)
|
|
|
- return dynamic_map.get(component_name)
|
|
|
-
|
|
|
-def generate_dist_doc(dist_dir, enabled_components, project_name, BSP_ROOT, RTT_ROOT):
|
|
|
- """Generate distribution package documentation"""
|
|
|
- doc_lines = [] # Store document content in a list
|
|
|
-
|
|
|
- # Basic information
|
|
|
- doc_lines.extend([
|
|
|
- "# RT-Thread Distribution Package\n",
|
|
|
- "\n## Basic Information\n\n",
|
|
|
- f"- Project Name: {project_name}\n",
|
|
|
- f"- Generation Time: {time.strftime('%Y-%m-%d %H:%M:%S')}\n",
|
|
|
- f"- BSP: {os.path.basename(BSP_ROOT)}\n",
|
|
|
- "\n## Components\n\n",
|
|
|
- "### Included Components:\n\n"
|
|
|
- ])
|
|
|
-
|
|
|
- # Add component information
|
|
|
- for comp in sorted(enabled_components):
|
|
|
- path = get_component_path(comp, RTT_ROOT)
|
|
|
- if path:
|
|
|
- doc_lines.append(f"- {comp}\n - Path: {path}\n")
|
|
|
-
|
|
|
- # Add configuration information
|
|
|
- doc_lines.extend(["\n## Configuration\n\n"])
|
|
|
- config_file = os.path.join(BSP_ROOT, '.config')
|
|
|
- if os.path.exists(config_file):
|
|
|
- doc_lines.extend([
|
|
|
- "### Main Configuration Items:\n\n```\n"
|
|
|
- ])
|
|
|
- with open(config_file, 'r') as f:
|
|
|
- for line in f:
|
|
|
- if line.startswith('CONFIG_'):
|
|
|
- doc_lines.append(line)
|
|
|
- doc_lines.append("```\n")
|
|
|
-
|
|
|
- # Add simplified directory structure
|
|
|
- doc_lines.extend(["\n## Directory Structure\n\n```\n"])
|
|
|
-
|
|
|
- # Show only top-level directories
|
|
|
- items = os.listdir(dist_dir)
|
|
|
- items.sort()
|
|
|
- for item in items:
|
|
|
- if item.startswith('.') or item == 'dist':
|
|
|
- continue
|
|
|
- path = os.path.join(dist_dir, item)
|
|
|
- if os.path.isdir(path):
|
|
|
- doc_lines.append(f"├── {item}/\n")
|
|
|
- else:
|
|
|
- doc_lines.append(f"├── {item}\n")
|
|
|
-
|
|
|
- doc_lines.append("```\n")
|
|
|
-
|
|
|
- # Add build instructions
|
|
|
- doc_lines.extend(["""
|
|
|
-## Build Instructions
|
|
|
-
|
|
|
-1. Requirements:
|
|
|
- - Python 3.x
|
|
|
- - SCons build tool
|
|
|
- - Appropriate cross-compiler toolchain
|
|
|
-
|
|
|
-2. Build Steps:
|
|
|
- ```bash
|
|
|
- scons
|
|
|
- ```
|
|
|
-
|
|
|
-3. Clean Build:
|
|
|
- ```bash
|
|
|
- scons -c
|
|
|
- ```
|
|
|
-
|
|
|
-## Notes
|
|
|
-
|
|
|
-1. Make sure the toolchain environment variables are properly set
|
|
|
-2. To modify configuration, use menuconfig:
|
|
|
- ```bash
|
|
|
- scons --menuconfig
|
|
|
- ```
|
|
|
-
|
|
|
-## License
|
|
|
-
|
|
|
-See `COPYING` file for details.
|
|
|
-"""])
|
|
|
-
|
|
|
- # Write documentation
|
|
|
- doc_file = os.path.join(dist_dir, 'dist_readme.md')
|
|
|
- with open(doc_file, 'w', encoding='utf-8') as f:
|
|
|
- f.writelines(doc_lines)
|
|
|
-
|
|
|
- print(f"=> Generated distribution documentation: {doc_file}")
|
|
|
-
|
|
|
-def is_text_file(filepath):
|
|
|
- """Check if a file is a text file"""
|
|
|
- text_extensions = {
|
|
|
- '.h', '.c', '.cpp', '.hpp', '.S', '.s', '.asm',
|
|
|
- '.txt', '.md', '.rst', '.ini', '.conf',
|
|
|
- 'Kconfig', 'SConscript', 'SConstruct',
|
|
|
- '.json', '.yml', '.yaml',
|
|
|
- '.cmake', 'CMakeLists.txt',
|
|
|
- '.py', '.sh', '.bat',
|
|
|
- 'README', 'LICENSE', 'Makefile'
|
|
|
- }
|
|
|
-
|
|
|
- # Check by extension
|
|
|
- ext = os.path.splitext(filepath)[1].lower()
|
|
|
- if ext in text_extensions or os.path.basename(filepath) in text_extensions:
|
|
|
- return True
|
|
|
-
|
|
|
- # Additional check for files without extension
|
|
|
- if '.' not in os.path.basename(filepath):
|
|
|
- try:
|
|
|
- with open(filepath, 'r', encoding='utf-8') as f:
|
|
|
- f.read(1024) # Try to read as text
|
|
|
- return True
|
|
|
- except:
|
|
|
- return False
|
|
|
-
|
|
|
- return False
|
|
|
-
|
|
|
-def copy_component_dependencies(src_path, dst_base, RTT_ROOT, copied_files=None):
|
|
|
- """Copy component dependencies (text files) from parent directories"""
|
|
|
- if copied_files is None:
|
|
|
- copied_files = set()
|
|
|
-
|
|
|
- # Get relative path from RTT_ROOT
|
|
|
- rel_path = os.path.relpath(src_path, RTT_ROOT)
|
|
|
- parent_path = os.path.dirname(rel_path)
|
|
|
-
|
|
|
- # Process all parent directories until RTT_ROOT
|
|
|
- while parent_path and parent_path != '.':
|
|
|
- src_dir = os.path.join(RTT_ROOT, parent_path)
|
|
|
-
|
|
|
- # Copy all text files in the directory (not recursively)
|
|
|
- for item in os.listdir(src_dir):
|
|
|
- src_file = os.path.join(src_dir, item)
|
|
|
- if os.path.isfile(src_file) and src_file not in copied_files:
|
|
|
- if is_text_file(src_file):
|
|
|
- dst_file = os.path.join(dst_base, parent_path, item)
|
|
|
- dst_dir = os.path.dirname(dst_file)
|
|
|
-
|
|
|
- if not os.path.exists(dst_dir):
|
|
|
- os.makedirs(dst_dir)
|
|
|
-
|
|
|
- do_copy_file(src_file, dst_file)
|
|
|
- copied_files.add(src_file)
|
|
|
- print(f' => copying {item} from {parent_path}')
|
|
|
-
|
|
|
- parent_path = os.path.dirname(parent_path)
|
|
|
-
|
|
|
- return copied_files
|
|
|
-
|
|
|
-def get_essential_paths():
|
|
|
- """Get essential paths that must be included"""
|
|
|
- return {
|
|
|
- 'components/libc/compilers', # Common compiler support
|
|
|
- 'components/drivers/include', # Driver headers
|
|
|
- 'components/drivers/core', # Driver core
|
|
|
- 'components/utilities', # Utility functions
|
|
|
- 'components/mm', # Memory management
|
|
|
- 'components/legacy/ipc', # IPC support, not always used, but have no config option for it
|
|
|
- }
|
|
|
-
|
|
|
-def copy_essential_paths(RTT_ROOT, rtt_dir_path, copied_files=None):
|
|
|
- """Copy essential paths and their build files"""
|
|
|
- if copied_files is None:
|
|
|
- copied_files = set()
|
|
|
-
|
|
|
- print('=> copying essential paths')
|
|
|
- for path in get_essential_paths():
|
|
|
- src_path = os.path.join(RTT_ROOT, path)
|
|
|
- if os.path.exists(src_path):
|
|
|
- dst_path = os.path.join(rtt_dir_path, path)
|
|
|
- print(f' => copying {path}')
|
|
|
- do_copy_folder(src_path, dst_path)
|
|
|
-
|
|
|
- # Copy build files for this path
|
|
|
- copied_files = copy_component_dependencies(src_path, rtt_dir_path, RTT_ROOT, copied_files)
|
|
|
-
|
|
|
- return copied_files
|
|
|
-
|
|
|
-def copy_components_kconfig(RTT_ROOT, rtt_dir_path):
|
|
|
- """Copy all Kconfig files under components directory"""
|
|
|
- components_dir = os.path.join(RTT_ROOT, 'components')
|
|
|
- print('=> copying components Kconfig files')
|
|
|
-
|
|
|
- # Walk through all directories under components
|
|
|
- for root, dirs, files in os.walk(components_dir):
|
|
|
- if 'Kconfig' in files:
|
|
|
- # Get relative path from components directory
|
|
|
- rel_path = os.path.relpath(root, RTT_ROOT)
|
|
|
- src_file = os.path.join(root, 'Kconfig')
|
|
|
- dst_file = os.path.join(rtt_dir_path, rel_path, 'Kconfig')
|
|
|
-
|
|
|
- # Create destination directory if not exists
|
|
|
- dst_dir = os.path.dirname(dst_file)
|
|
|
- if not os.path.exists(dst_dir):
|
|
|
- os.makedirs(dst_dir)
|
|
|
-
|
|
|
- do_copy_file(src_file, dst_file)
|
|
|
- print(f' => copying Kconfig from {rel_path}')
|
|
|
-
|
|
|
-def components_copy_files(RTT_ROOT, rtt_dir_path, config_file):
|
|
|
- """Copy components based on configuration"""
|
|
|
- print('=> components (selective copy)')
|
|
|
-
|
|
|
- # Copy all Kconfig files first
|
|
|
- copy_components_kconfig(RTT_ROOT, rtt_dir_path)
|
|
|
-
|
|
|
- # Track copied build files to avoid duplication
|
|
|
- copied_files = set()
|
|
|
-
|
|
|
- # Copy components/SConscript first
|
|
|
- components_sconscript = os.path.join(RTT_ROOT, 'components', 'SConscript')
|
|
|
- if os.path.exists(components_sconscript):
|
|
|
- dst_dir = os.path.join(rtt_dir_path, 'components')
|
|
|
- if not os.path.exists(dst_dir):
|
|
|
- os.makedirs(dst_dir)
|
|
|
- do_copy_file(components_sconscript, os.path.join(dst_dir, 'SConscript'))
|
|
|
- copied_files.add(components_sconscript)
|
|
|
-
|
|
|
- # Copy essential paths first
|
|
|
- copied_files = copy_essential_paths(RTT_ROOT, rtt_dir_path, copied_files)
|
|
|
-
|
|
|
- # Get enabled components
|
|
|
- enabled_components = parse_components_from_config(config_file)
|
|
|
- if not enabled_components:
|
|
|
- print("Warning: No components found in config file")
|
|
|
- return enabled_components
|
|
|
-
|
|
|
- # Copy each enabled component
|
|
|
- for comp_name in enabled_components:
|
|
|
- comp_path = get_component_path(comp_name, RTT_ROOT)
|
|
|
- if comp_path:
|
|
|
- src_path = os.path.join(RTT_ROOT, comp_path)
|
|
|
- dst_path = os.path.join(rtt_dir_path, comp_path)
|
|
|
-
|
|
|
- if os.path.exists(src_path):
|
|
|
- print(f' => copying {comp_name} from {comp_path}')
|
|
|
- do_copy_folder(src_path, dst_path)
|
|
|
-
|
|
|
- # Copy parent directory build files
|
|
|
- copied_files = copy_component_dependencies(src_path, rtt_dir_path, RTT_ROOT, copied_files)
|
|
|
- else:
|
|
|
- print(f"Warning: Component path not found: {src_path}")
|
|
|
- else:
|
|
|
- print(f"Note: Skipping system feature: {comp_name}")
|
|
|
-
|
|
|
- return enabled_components
|
|
|
-
|
|
|
def MkDist(program, BSP_ROOT, RTT_ROOT, Env, project_name, project_path):
|
|
|
print('make distribution....')
|
|
|
|
|
@@ -502,43 +184,42 @@ def MkDist(program, BSP_ROOT, RTT_ROOT, Env, project_name, project_path):
|
|
|
|
|
|
rtt_dir_path = os.path.join(dist_dir, 'rt-thread')
|
|
|
|
|
|
- # Copy BSP files
|
|
|
+ # copy BSP files
|
|
|
print('=> %s' % os.path.basename(BSP_ROOT))
|
|
|
bsp_copy_files(BSP_ROOT, dist_dir)
|
|
|
|
|
|
- # Do BSP special dist handle
|
|
|
+ # do bsp special dist handle
|
|
|
if 'dist_handle' in Env:
|
|
|
print("=> start dist handle")
|
|
|
dist_handle = Env['dist_handle']
|
|
|
dist_handle(BSP_ROOT, dist_dir)
|
|
|
|
|
|
- # Use new component copy function and get list of enabled components
|
|
|
- config_file = os.path.join(BSP_ROOT, '.config')
|
|
|
- enabled_components = components_copy_files(RTT_ROOT, rtt_dir_path, config_file)
|
|
|
-
|
|
|
- # Skip documentation directory
|
|
|
- # Skip examples
|
|
|
+ # copy tools directory
|
|
|
+ print('=> components')
|
|
|
+ do_copy_folder(os.path.join(RTT_ROOT, 'components'), os.path.join(rtt_dir_path, 'components'))
|
|
|
+
|
|
|
+ # skip documentation directory
|
|
|
+ # skip examples
|
|
|
|
|
|
- # Copy include directory
|
|
|
+ # copy include directory
|
|
|
print('=> include')
|
|
|
do_copy_folder(os.path.join(RTT_ROOT, 'include'), os.path.join(rtt_dir_path, 'include'))
|
|
|
|
|
|
- # Copy all libcpu/ARCH directory
|
|
|
+ # copy all libcpu/ARCH directory
|
|
|
print('=> libcpu')
|
|
|
import rtconfig
|
|
|
do_copy_folder(os.path.join(RTT_ROOT, 'libcpu', rtconfig.ARCH), os.path.join(rtt_dir_path, 'libcpu', rtconfig.ARCH))
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'libcpu', 'Kconfig'), os.path.join(rtt_dir_path, 'libcpu', 'Kconfig'))
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'libcpu', 'SConscript'), os.path.join(rtt_dir_path, 'libcpu', 'SConscript'))
|
|
|
|
|
|
- # Copy src directory
|
|
|
+ # copy src directory
|
|
|
print('=> src')
|
|
|
do_copy_folder(os.path.join(RTT_ROOT, 'src'), os.path.join(rtt_dir_path, 'src'))
|
|
|
|
|
|
- # Copy tools directory
|
|
|
+ # copy tools directory
|
|
|
print('=> tools')
|
|
|
do_copy_folder(os.path.join(RTT_ROOT, 'tools'), os.path.join(rtt_dir_path, 'tools'), ignore_patterns('*.pyc'))
|
|
|
|
|
|
- # Copy necessary files
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'Kconfig'), os.path.join(rtt_dir_path, 'Kconfig'))
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'AUTHORS'), os.path.join(rtt_dir_path, 'AUTHORS'))
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'COPYING'), os.path.join(rtt_dir_path, 'COPYING'))
|
|
@@ -546,14 +227,14 @@ def MkDist(program, BSP_ROOT, RTT_ROOT, Env, project_name, project_path):
|
|
|
do_copy_file(os.path.join(RTT_ROOT, 'README_zh.md'), os.path.join(rtt_dir_path, 'README_zh.md'))
|
|
|
|
|
|
print('Update configuration files...')
|
|
|
+ # change RTT_ROOT in SConstruct
|
|
|
bsp_update_sconstruct(dist_dir)
|
|
|
+ # change RTT_ROOT in Kconfig
|
|
|
bsp_update_kconfig(dist_dir)
|
|
|
bsp_update_kconfig_library(dist_dir)
|
|
|
+ # delete testcases in Kconfig
|
|
|
bsp_update_kconfig_testcases(dist_dir)
|
|
|
|
|
|
- # Generate documentation
|
|
|
- generate_dist_doc(dist_dir, enabled_components, project_name+'-dist', BSP_ROOT, RTT_ROOT)
|
|
|
-
|
|
|
target_project_type = GetOption('target')
|
|
|
if target_project_type:
|
|
|
child = subprocess.Popen('scons --target={} --project-name="{}"'.format(target_project_type, project_name), cwd=dist_dir, stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=True)
|
|
@@ -570,3 +251,90 @@ def MkDist(program, BSP_ROOT, RTT_ROOT, Env, project_name, project_path):
|
|
|
zip_dist(dist_dir, project_name)
|
|
|
|
|
|
print('dist project successfully!')
|
|
|
+
|
|
|
+def MkDist_Strip(program, BSP_ROOT, RTT_ROOT, env, project_name, project_path=None):
|
|
|
+ """Create a minimal distribution based on compile_commands.json but keeping all build system files.
|
|
|
+ First copies everything like MkDist, then only removes unused source files while keeping all headers.
|
|
|
+ """
|
|
|
+ print('Making minimal distribution for project...')
|
|
|
+
|
|
|
+ if project_path == None:
|
|
|
+ dist_dir = os.path.join(BSP_ROOT, 'dist', project_name)
|
|
|
+ else:
|
|
|
+ dist_dir = project_path
|
|
|
+
|
|
|
+ # First do a full distribution copy
|
|
|
+ MkDist(program, BSP_ROOT, RTT_ROOT, env, project_name, project_path)
|
|
|
+ print('\n=> Starting source files cleanup...')
|
|
|
+
|
|
|
+ # Get the minimal required source paths
|
|
|
+ import compile_commands
|
|
|
+ used_paths = compile_commands.get_minimal_dist_paths(
|
|
|
+ os.path.join(BSP_ROOT, 'compile_commands.json'),
|
|
|
+ RTT_ROOT
|
|
|
+ )
|
|
|
+
|
|
|
+ # Clean up RT-Thread directory except tools and build files
|
|
|
+ rt_thread_dir = os.path.join(dist_dir, 'rt-thread')
|
|
|
+ source_extensions = ('.c', '.cpp', '.cxx', '.cc', '.s', '.S')
|
|
|
+
|
|
|
+ removed_files = []
|
|
|
+ removed_dirs = []
|
|
|
+
|
|
|
+ for root, dirs, files in os.walk(rt_thread_dir, topdown=False):
|
|
|
+ rel_path = os.path.relpath(root, rt_thread_dir)
|
|
|
+
|
|
|
+ if rel_path.startswith('tools') or rel_path.startswith('include'):
|
|
|
+ continue
|
|
|
+
|
|
|
+ keep_files = {
|
|
|
+ 'SConscript',
|
|
|
+ 'Kconfig',
|
|
|
+ 'Sconscript',
|
|
|
+ '.config',
|
|
|
+ 'rtconfig.h'
|
|
|
+ }
|
|
|
+
|
|
|
+ for f in files:
|
|
|
+ if f in keep_files:
|
|
|
+ continue
|
|
|
+
|
|
|
+ if not f.endswith(source_extensions):
|
|
|
+ continue
|
|
|
+
|
|
|
+ file_path = os.path.join(root, f)
|
|
|
+ rel_file_path = os.path.relpath(file_path, rt_thread_dir)
|
|
|
+ dir_name = os.path.dirname(rel_file_path)
|
|
|
+
|
|
|
+ if dir_name not in used_paths and rel_file_path not in used_paths:
|
|
|
+ os.remove(file_path)
|
|
|
+ removed_files.append(rel_file_path)
|
|
|
+
|
|
|
+ # Remove empty directories
|
|
|
+ try:
|
|
|
+ if not os.listdir(root):
|
|
|
+ os.rmdir(root)
|
|
|
+ removed_dirs.append(rel_path)
|
|
|
+ except:
|
|
|
+ pass
|
|
|
+
|
|
|
+ # Output summary
|
|
|
+ if removed_files:
|
|
|
+ print(f"Removed {len(removed_files)} unused source files")
|
|
|
+ log_file = os.path.join(dist_dir, 'cleanup.log')
|
|
|
+ with open(log_file, 'w') as f:
|
|
|
+ f.write("Removed source files:\n")
|
|
|
+ f.write('\n'.join(removed_files))
|
|
|
+ if removed_dirs:
|
|
|
+ f.write("\n\nRemoved empty directories:\n")
|
|
|
+ f.write('\n'.join(removed_dirs))
|
|
|
+ print(f"Details have been written to {log_file}")
|
|
|
+ else:
|
|
|
+ print("No unused source files found")
|
|
|
+
|
|
|
+ # Make zip package like MkDist
|
|
|
+ if project_path is None:
|
|
|
+ zip_dist(dist_dir, project_name)
|
|
|
+ print(f"Distribution package created: {dist_dir}.zip")
|
|
|
+
|
|
|
+ print('=> Distribution stripped successfully')
|