红联Linux门户
Linux帮助

mx4 ubuntu edtion救砖

发布时间:2016-08-09 13:32:07来源:linux网站作者:yTexas
记一次mx4 ubuntu edtion救砖
 
1.成砖原因:
作死在手机终端下执行rm -rf -i /删除根目录, 次要问题是recovery分区不能用fastboot刷写, 导致无法通过ubuntu-device-flash重新下载ubuntu touch
ubuntu-device-flash touch --channel=ubuntu-touch/stable/meizu.en --device=arale --bootstrap  
最后只好通过写入flyme救砖
 
2.救砖过程:
在meizu官网下载update.zip, 解压, 此过程需用到解压后目录下boot.img(内核), system.new.dat, system.transfer.list 这三个文件, 以及一个dat文件转img脚本 sdat2img
2.1.system.new.dat转system.img
2.1.1.准备dat文件转img的python脚本: sdat2img.py
将以下代码复制至你的.py脚本文件内, (非常感谢 xpirt, lux78, howellzhu的python脚本 XD)
#!/usr/bin/env python  
# -*- coding: utf-8 -*-  
#=========================================  
#          FILE: sdat2img.py  
#       AUTHORS: xpirt - luxi78 - howellzhu  
#=========================================  
import sys, os  
try:  
TRANSFER_LIST_FILE = str(sys.argv[1])  
NEW_DATA_FILE = str(sys.argv[2])  
OUTPUT_IMAGE_FILE = str(sys.argv[3])  
except IndexError:  
print ("\nsdat2img - usage is: \n\n   sdat2img <transfer_list> <system_new_file> <system_img>\n\n")  
print ("Visit xda thread for more information.\n")  
try:  
input = raw_input  
except NameError: pass  
input ("Press ENTER to exit...\n")  
sys.exit()  
BLOCK_SIZE = 4096  
def rangeset(src):  
src_set = src.split(',')  
num_set =  [int(item) for item in src_set]  
if len(num_set) != num_set[0]+1:  
print ('Error on parsing following data to rangeset:\n%s' % src)  
sys.exit(1)  
return tuple ([ (num_set[i], num_set[i+1]) for i in range(1, len(num_set), 2) ])  
def parse_transfer_list_file(path):  
trans_list = open(TRANSFER_LIST_FILE, 'r')  
version = int(trans_list.readline())    # 1st line = transfer list version  
new_blocks = int(trans_list.readline()) # 2nd line = total number of blocks  
# system.transfer.list:  
#   - version 1: android-5.0.0_r1  
#   - version 2: android-5.1.0_r1  
#   - version 3: android-6.0.0_r1  
# skip next 2 lines. we don't need this stuff now  
if version >= 2:  
trans_list.readline()               # 3rd line = stash entries needed simultaneously  
trans_list.readline()               # 4th line = number of blocks that will be stashed  
commands = []  
for line in trans_list:  
line = line.split(' ')              # 5th & next lines should be only commands  
cmd = line[0]  
if cmd in ['erase', 'new', 'zero']:  
commands.append([cmd, rangeset(line[1])])  
else:  
# skip lines starting with numbers, they're not commands anyway.  
if not cmd[0].isdigit():  
print ('No valid command: %s.' % cmd)  
trans_list.close()  
sys.exit(1)  
trans_list.close()  
return version, new_blocks, commands  
def init_output_file_size(output_file_obj, commands):  
all_block_sets = [i for command in commands for i in command[1]]  
max_block_num = max(pair[1] for pair in all_block_sets)  
output_file_obj.seek(max_block_num*BLOCK_SIZE - 1)  
output_file_obj.write('\0'.encode('utf-8'))  
output_file_obj.flush()  
def main(argv):  
version, new_blocks, commands = parse_transfer_list_file(TRANSFER_LIST_FILE)  
output_img = open(OUTPUT_IMAGE_FILE, 'wb')  
init_output_file_size(output_img, commands)  
new_data_file = open(NEW_DATA_FILE, 'rb')  
for command in commands:  
if command[0] == 'new':  
for block in command[1]:  
begin = block[0]  
end = block[1]
block_count = end - begin  
data = new_data_file.read(block_count*BLOCK_SIZE)  
print('Copying {} blocks into position {}...'.format(block_count, begin))  
output_img.seek(begin*BLOCK_SIZE)  
output_img.write(data)  
else:  
print('Skipping command %s' % command[0])  
output_img.close()  
new_data_file.close()  
print ('\nDone! Output image: %s' % os.path.realpath(output_img.name))  
if __name__ == "__main__":  
main(sys.argv)  
2.1.2.转system.new.dat 为 system.img:
建议将得到的python脚本文件xxx.py放在update.zip解压后的目录下, 如 xxxxx/update/. 执行以下命令生成system.img
./sdat2img.py system.transfer.list system.new.dat system.img
转为system.img后还有一件事情需要做, 先挂载system.img, 再删除挂载目录下的recovery-from-boot.p文件, 不删除的话, 刷写后会无法开机, 具体原因还不清楚 >.<.
可通过以下命令删除system.img中的recover-from-boot.p
mkidr system  
sudo mount --rw system system.img  
sudo rm system/recovery-from-boot.p  
sudo umount system  
rmdir system  
2.2.刷入boot.img与system.img
在mx4关机状态下长按电源键 + 音量(-)进入fastboot模式
打开终端, 进入update.zip解压后的目录, 输入
fastboot flash boot boot.img  //刷boot.img  
fastboot flash system system.img  //刷system.img  
fastboot reboot  //重启  
第一次开机会耗费较长时间, 请耐心等候, 至此, 救砖成功.
 
3.几点疑问:
3.1.最让我困惑的是, 我的这部mx4 ubuntu edtion在fastboot下无法刷写recovery, 问了其他网友说是没有这样的问题>.< . (如果recovery 可以在fastboot 下刷写的话, 就没上述的那么麻烦, 直接可以ubuntu-device-flash刷入)
3.2.刷入flyme的时候, 刚开始刷入boot.img system.img 后, 机器依然无法启动, 后来偶然看到某论坛说"删除'recovery-from-boot.p' ",试了下居然成功了实在不可思议, 但为什么删了这个文件就可以让手机启动了? 为什么有了这个文件手机就不能启动? 这个文件的作用?
3.3.刷入flyme, 其实是想再通过adb刷回ubuntu touch, 显然, 回不去了, recovery上锁了的样子, 之前也尝试过"xen0n"(blog.xen0n.name) 的 cm13, 但得到system.img刷入时总提示" get_partition_name() fail", 这又是为什么? Q.Q
3.4.更早以前, 按照https://wiki.ubuntu.com/Touch/Deploying的说明, 尝试制作system.img, 刷入失败(提示"Brick phone?" "Write data error"),
自己尝试制作system.img的步骤如下:
# 1. 创建system.img并挂载于system目录
fallocate -l 2G system.img
mke2fs -F system.img
mkdir system
sudo mount -o loop system.img system
# 此时system system.img在同一目录
# 2. 制作system.img
# 将ubuntu-device-flash下载的压缩包ubuntu-xxxx device-xxxx放在system.img所在目录 
cp ~/.cache/ubuntuimage/pool/ubuntu-* ./
cp ~/.cache/ubuntuimage/pool/device-* ./
# 再解压在system目录
sudo tar --numeric-owner -xaf ubuntu-xxxx.tar.xz system
# 将device-xxx解压, 把device-xxx/system/var/lib/lxc/android/system.img 复制到 ./system/var/lib/lxc/android/system.img
# 此步骤的作用是? 这里的system.img是android驱动? 求解答 >.<
# 3.压缩system.img
# 执行以下三行命令压缩 system.img
e2fsck -yf system.img
resize2fs -M system.img
e2fsck -yf system.img
#执行以下命令
dumpe2fs system.img | less
#在上行命令中找到Block Size 和 Block Count并相乘,将相乘结果作为参数SIZE执行下行脚本
truncate -s SIZE system.img
#至此完成
 
最后非常感谢大神们编写的各种工具, 脚本, 还有很多人的热心解答(sry我问的问题总是那么弱智), 虽然回不到uTouch, 至少手机活过来了, 所以建议不要rm -rf /
欢迎指出错别字等各种错误, 还有很多疑问希望能一起探讨解决.
 
本文永久更新地址:http://www.linuxdiyf.com/linux/23149.html