Auto detect and support both the v1 and v2 fstab formats Support putting TWRP style flags in a separate /etc/twrp.flags file twrp.flags format is the same as twrp.fstab (v1 with TWRP flags) Support using a wildcard in a block device and find all partitions: /usb-otg vfat /dev/block/sda* Support using sysfs entries (voldmanaged) and read uevents and scan for wildcard partitions from uevent data. (twvold?) May not be complete for some of the newer flags found in fstabs in newer build trees and there is a slim chance of a crash if the user removes a removable device while TWRP is performing actions. May need to add some kind of mutex to prevent the 2 threads from causing this crash. We need to start somewhere though and this change is pretty innocuous when not using a v2 fstab. Change-Id: I617d97c7db332cbe671a9d2b8ad98b3d9c4f03cc
219 lines
5.6 KiB
C++
219 lines
5.6 KiB
C++
/*
|
|
Copyright 2013 to 2017 TeamWin
|
|
This file is part of TWRP/TeamWin Recovery Project.
|
|
|
|
TWRP is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
TWRP is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with TWRP. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#include <fcntl.h>
|
|
#include <string>
|
|
#include <unistd.h>
|
|
#include "data.hpp"
|
|
#include "partitions.hpp"
|
|
#include "set_metadata.h"
|
|
#include "twrpDigestDriver.hpp"
|
|
#include "twrp-functions.hpp"
|
|
#include "twcommon.h"
|
|
#include "variables.h"
|
|
#include "gui/gui.hpp"
|
|
#include "twrpDigest/twrpDigest.hpp"
|
|
#include "twrpDigest/twrpMD5.hpp"
|
|
#include "twrpDigest/twrpSHA.hpp"
|
|
|
|
|
|
bool twrpDigestDriver::Check_Restore_File_Digest(const string& Filename) {
|
|
twrpDigest *digest;
|
|
string digestfile = Filename, file_name = Filename;
|
|
string digest_str;
|
|
bool use_sha2 = false;
|
|
|
|
#ifndef TW_NO_SHA2_LIBRARY
|
|
|
|
digestfile += ".sha2";
|
|
if (TWFunc::Path_Exists(digestfile)) {
|
|
digest = new twrpSHA256();
|
|
use_sha2 = true;
|
|
}
|
|
else {
|
|
digest = new twrpMD5();
|
|
digestfile = Filename + ".md5";
|
|
|
|
}
|
|
#else
|
|
digest = new twrpMD5();
|
|
digestfile = Filename + ".md5";
|
|
|
|
#endif
|
|
|
|
if (!TWFunc::Path_Exists(digestfile)) {
|
|
gui_msg(Msg(msg::kError, "no_digest_found=No digest file found for '{1}'. Please unselect Enable Digest verification to restore.")(Filename));
|
|
delete digest;
|
|
return false;
|
|
}
|
|
|
|
|
|
if (TWFunc::read_file(digestfile, digest_str) != 0) {
|
|
gui_msg("digest_error=Digest Error!");
|
|
delete digest;
|
|
return false;
|
|
}
|
|
|
|
if (!stream_file_to_digest(file_name, digest)) {
|
|
delete digest;
|
|
return false;
|
|
}
|
|
string digest_check = digest->return_digest_string();
|
|
if (digest_check == digest_str) {
|
|
if (use_sha2)
|
|
LOGINFO("SHA2 Digest: %s %s\n", digest_str.c_str(), TWFunc::Get_Filename(Filename).c_str());
|
|
else
|
|
LOGINFO("MD5 Digest: %s %s\n", digest_str.c_str(), TWFunc::Get_Filename(Filename).c_str());
|
|
delete digest;
|
|
return true;
|
|
}
|
|
|
|
gui_msg(Msg(msg::kError, "digest_fail_match=Digest failed to match on '{1}'.")(Filename));
|
|
delete digest;
|
|
return false;
|
|
|
|
}
|
|
|
|
bool twrpDigestDriver::Check_Digest(string Full_Filename) {
|
|
char split_filename[512];
|
|
int index = 0;
|
|
|
|
sync();
|
|
if (!TWFunc::Path_Exists(Full_Filename)) {
|
|
// This is a split archive, we presume
|
|
memset(split_filename, 0, sizeof(split_filename));
|
|
while (index < 1000) {
|
|
sprintf(split_filename, "%s%03i", Full_Filename.c_str(), index);
|
|
if (!TWFunc::Path_Exists(split_filename))
|
|
break;
|
|
LOGINFO("split_filename: %s\n", split_filename);
|
|
if (!Check_Restore_File_Digest(split_filename))
|
|
return false;
|
|
index++;
|
|
}
|
|
return true;
|
|
}
|
|
return Check_Restore_File_Digest(Full_Filename); // Single file archive
|
|
}
|
|
|
|
bool twrpDigestDriver::Write_Digest(string Full_Filename) {
|
|
twrpDigest *digest;
|
|
string digest_filename, digest_str;
|
|
int use_sha2;
|
|
|
|
#ifdef TW_NO_SHA2_LIBRARY
|
|
use_sha2 = 0;
|
|
#else
|
|
DataManager::GetValue(TW_USE_SHA2, use_sha2);
|
|
#endif
|
|
|
|
if (use_sha2) {
|
|
#ifndef TW_NO_SHA2_LIBRARY
|
|
digest = new twrpSHA256();
|
|
digest_filename = Full_Filename + ".sha2";
|
|
if (!stream_file_to_digest(Full_Filename, digest)) {
|
|
delete digest;
|
|
return false;
|
|
}
|
|
digest_str = digest->return_digest_string();
|
|
if (digest_str.empty()) {
|
|
delete digest;
|
|
return false;
|
|
}
|
|
LOGINFO("SHA2 Digest: %s %s\n", digest_str.c_str(), TWFunc::Get_Filename(Full_Filename).c_str());
|
|
#endif
|
|
}
|
|
else {
|
|
digest = new twrpMD5();
|
|
digest_filename = Full_Filename + ".md5";
|
|
if (!stream_file_to_digest(Full_Filename, digest)) {
|
|
delete digest;
|
|
return false;
|
|
}
|
|
digest_str = digest->return_digest_string();
|
|
if (digest_str.empty()) {
|
|
delete digest;
|
|
return false;
|
|
}
|
|
LOGINFO("MD5 Digest: %s %s\n", digest_str.c_str(), TWFunc::Get_Filename(Full_Filename).c_str());
|
|
}
|
|
|
|
digest_str = digest_str + " " + TWFunc::Get_Filename(Full_Filename) + "\n";
|
|
LOGINFO("digest_filename: %s\n", digest_filename.c_str());
|
|
|
|
if (TWFunc::write_to_file(digest_filename, digest_str) == 0) {
|
|
tw_set_default_metadata(digest_filename.c_str());
|
|
gui_msg("digest_created= * Digest Created.");
|
|
}
|
|
else {
|
|
gui_err("digest_error= * Digest Error!");
|
|
delete digest;
|
|
return false;
|
|
}
|
|
delete digest;
|
|
return true;
|
|
}
|
|
|
|
bool twrpDigestDriver::Make_Digest(string Full_Filename) {
|
|
string command, result;
|
|
|
|
TWFunc::GUI_Operation_Text(TW_GENERATE_DIGEST_TEXT, gui_parse_text("{@generating_digest1}"));
|
|
gui_msg("generating_digest2= * Generating digest...");
|
|
if (TWFunc::Path_Exists(Full_Filename)) {
|
|
if (!Write_Digest(Full_Filename))
|
|
return false;
|
|
} else {
|
|
char filename[512];
|
|
int index = 0;
|
|
sprintf(filename, "%s%03i", Full_Filename.c_str(), index);
|
|
while (index < 1000) {
|
|
string digest_src(filename);
|
|
if (TWFunc::Path_Exists(filename)) {
|
|
if (!Write_Digest(filename))
|
|
return false;
|
|
}
|
|
else
|
|
break;
|
|
index++;
|
|
sprintf(filename, "%s%03i", Full_Filename.c_str(), index);
|
|
}
|
|
if (index == 0) {
|
|
LOGERR("Backup file: '%s' not found!\n", filename);
|
|
return false;
|
|
}
|
|
gui_msg("digest_created= * Digest Created.");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool twrpDigestDriver::stream_file_to_digest(string filename, twrpDigest* digest) {
|
|
char buf[4096];
|
|
int bytes;
|
|
|
|
int fd = open(filename.c_str(), O_RDONLY);
|
|
if (fd < 0) {
|
|
return false;
|
|
}
|
|
while ((bytes = read(fd, &buf, sizeof(buf))) != 0) {
|
|
digest->update((unsigned char*)buf, bytes);
|
|
}
|
|
close(fd);
|
|
return true;
|
|
}
|