[Feature][T108][task-view-1706] enable fastboot && auto make mtdpartinfo from dtsi fiel by self-config

    Only Configure: No
    Affected branch: GSW_V1453
    Affected module: SDK
    Is it affected on IC: only ASR
    Self-test: yes
    Doc Update: no

Change-Id: If72389ab2adb601c19bfa30491c03424444d47f1
diff --git a/generate_partitions.py b/generate_partitions.py
new file mode 100755
index 0000000..f114e1b
--- /dev/null
+++ b/generate_partitions.py
@@ -0,0 +1,235 @@
+#!/usr/bin/env python3
+# ASR1806 Partition Configuration Generator (Final Correct Format)
+
+import re
+import json
+import sys
+import os
+import argparse
+from collections import OrderedDict
+from pathlib import Path
+
+def validate_file(path, description):
+    """Validate input file exists and is readable"""
+    path_obj = Path(path)
+    if not path_obj.exists():
+        raise FileNotFoundError(f"{description} not found: {path}")
+    if not path_obj.is_file():
+        raise IOError(f"{description} path is not a file: {path}")
+    if not os.access(str(path_obj), os.R_OK):
+        raise PermissionError(f"Cannot read {description}: {path}")
+    return path_obj
+
+def parse_dtsi_partitions(dtsi_file):
+    """Parse partition information from DTSI file"""
+    partitions = OrderedDict()
+    try:
+        with open(dtsi_file, 'r', encoding='utf-8') as f:
+            content = f.read()
+        
+        pattern = re.compile(
+            r'partition@([0-9A-Fa-f]+)\s*{\s*.*?label\s*=\s*"([^"]+)";\s*.*?reg\s*=\s*<([^>]+)>;\s*.*?};',
+            re.DOTALL
+        )
+        
+        for match in pattern.finditer(content):
+            try:
+                start_addr = int(match.group(1), 16)
+                label = match.group(2).strip()
+                reg_values = [int(x, 16) for x in match.group(3).split()]
+                
+                if not reg_values:
+                    print(f"Invalid reg value for partition {label}", file=sys.stderr)
+                    continue
+                    
+                size = reg_values[-1] if len(reg_values) > 1 else reg_values[0]
+                
+                partitions[label] = {
+                    'start_addr': f"0x{start_addr:08X}",
+                    'size': f"0x{size:08X}",
+                    'size_bytes': size
+                }
+            except ValueError as e:
+                print(f"Failed to parse partition data: {e}", file=sys.stderr)
+                continue
+        
+        if not partitions:
+            raise ValueError("No valid partitions found in DTSI file")
+            
+        print(f"Parsed {len(partitions)} partitions from {dtsi_file}")
+        return partitions
+    except Exception as e:
+        raise RuntimeError(f"DTSI parse failed: {str(e)}") from e
+
+def load_config(config_file):
+    """Load and validate partition configuration"""
+    try:
+        with open(config_file, 'r', encoding='utf-8') as f:
+            config = json.load(f, object_pairs_hook=OrderedDict)
+        
+        required = {
+            'mtdparts_order': list,
+            'partition_aliases': dict
+        }
+        for key, typ in required.items():
+            if key not in config:
+                raise KeyError(f"Missing required key: {key}")
+            if not isinstance(config[key], typ):
+                raise TypeError(f"{key} must be {typ.__name__}")
+        
+        print(f"Loaded config from {config_file}")
+        return config
+    except json.JSONDecodeError as e:
+        raise RuntimeError(f"Invalid JSON format: {str(e)}") from e
+    except Exception as e:
+        raise RuntimeError(f"Config load failed: {str(e)}") from e
+
+def generate_complete_config(partitions, config):
+    """Generate the complete configuration block with exact formatting"""
+    # 1. Generate partition defines
+    defines = []
+    for part_name, alias in config['partition_aliases'].items():
+        if part_name not in partitions:
+            print(f"Config references unknown partition: {part_name}", file=sys.stderr)
+            continue
+        defines.append(f'#define {alias} "{part_name}"')
+    defines_section = "\n".join(defines)
+
+    # 2. Generate MTDPARTS content with exact formatting
+    mtdparts = []
+    valid_parts = [p for p in config['mtdparts_order'] if p in partitions and p in config['partition_aliases']]
+    num_parts = len(valid_parts)
+    
+    for i, part_name in enumerate(valid_parts):
+        part = partitions[part_name]
+        alias = config['partition_aliases'][part_name]
+        
+        # 构建每个分区的定义,格式为 "0x00180000@0x02600000("UBOOT_MTD_PART_A"),"
+        mtd_entry = f'0x{part["size"][2:]}@0x{part["start_addr"][2:]}("{alias}")'
+        if i < num_parts - 1:
+            mtd_entry += ','  # 除最后一个分区外,每个分区后加逗号
+        
+        # 每个分区定义用双引号包裹,并添加行继续符和缩进
+        if i == 0:
+            # 第一个分区不需要额外缩进
+            mtdparts.append(f'"mtdparts=nand_mtd:{mtd_entry}"\\')
+        else:
+            # 后续分区需要 5 个空格缩进
+            mtdparts.append(f'     "{mtd_entry}"\\')
+    
+    # 最后一行需要单独处理,确保没有双引号和反斜杠
+    if mtdparts:
+        last_line = mtdparts.pop()  # 移除最后一行
+        # 构建不带双引号和反斜杠的最后一行
+        mtdparts.append(last_line.replace('\\"', '').replace('\\', ''))
+    
+    # 连接所有分区定义
+    mtdparts_joined = '\n'.join(mtdparts)
+    
+    # 构建最终的 MTDPARTS 部分
+    mtdparts_section = f"""#ifdef CONFIG_CMD_FASTBOOT
+#define CONFIG_MTDPARTS \\
+{mtdparts_joined}
+#endif /* CONFIG_CMD_FASTBOOT */"""
+
+    # 3. Combine into complete block
+    complete_content = f"""{defines_section}
+
+{mtdparts_section}"""
+    
+    return complete_content
+
+def update_header_file(header_file, complete_content):
+    """Replace ONLY the content between dynamic markers"""
+    try:
+        header_file.parent.mkdir(parents=True, exist_ok=True)
+        
+        if not header_file.exists():
+            raise RuntimeError(f"Header file does not exist: {header_file}")
+        
+        # Read existing content
+        with open(header_file, 'r', encoding='utf-8') as f:
+            content = f.read()
+        original_mode = header_file.stat().st_mode
+        
+        # Pattern to find the exact block to replace
+        pattern = re.compile(
+            r'(/\* add mtd parts dynamically \*/\n)(.*?)(\n#else)',
+            re.DOTALL
+        )
+        
+        match = pattern.search(content)
+        if not match:
+            raise RuntimeError("Could not find the replacement markers ('/* add mtd parts dynamically */' to '#else')")
+        
+        # Replace only the middle part
+        updated_content = (
+            content[:match.start(2)] + 
+            complete_content + 
+            content[match.end(2):]
+        )
+        
+        # Write updated content
+        with open(header_file, 'w', encoding='utf-8') as f:
+            f.write(updated_content)
+        header_file.chmod(original_mode)
+        
+        print(f"Successfully updated {header_file}")
+        return True
+        
+    except Exception as e:
+        raise RuntimeError(f"Header update failed: {str(e)}") from e
+
+def main():
+    parser = argparse.ArgumentParser(
+        description='Generate partition configuration between markers',
+        formatter_class=argparse.ArgumentDefaultsHelpFormatter)
+    
+    parser.add_argument('--dtsi', required=True,
+                       help='Input DTSI file with partition layout')
+    parser.add_argument('--config', required=True,
+                       help='JSON config file with partition aliases')
+    parser.add_argument('--header', required=True,
+                       help='Output header file path')
+    parser.add_argument('--verbose', action='store_true',
+                       help='Show detailed processing info')
+    
+    args = parser.parse_args()
+    
+    try:
+        dtsi_path = validate_file(args.dtsi, "DTSI file")
+        config_path = validate_file(args.config, "Config file")
+        header_path = Path(args.header).resolve()
+        
+        if args.verbose:
+            print(f"Processing:\n- DTSI: {dtsi_path}\n- Config: {config_path}\n- Output: {header_path}")
+        
+        partitions = parse_dtsi_partitions(dtsi_path)
+        config = load_config(config_path)
+        
+        # Verify all configured partitions exist
+        missing = [p for p in config['mtdparts_order'] if p not in partitions]
+        if missing:
+            print(f"Missing partitions in DTSI: {', '.join(missing)}", file=sys.stderr)
+        
+        complete_content = generate_complete_config(partitions, config)
+        
+        if args.verbose:
+            print("\nGenerated content to insert:")
+            print("-" * 40)
+            print(complete_content)
+            print("-" * 40)
+        
+        updated = update_header_file(header_path, complete_content)
+        
+        if updated:
+            print("Update completed successfully")
+        
+        return 0
+        
+    except Exception as e:
+        print(f"Error: {str(e)}", file=sys.stderr)
+        return 1
+
+if __name__ == "__main__":
+    sys.exit(main())
\ No newline at end of file