mirror of
https://git.sr.ht/~grimler/Heimdall
synced 2025-03-01 03:01:16 +00:00
198 lines
5.8 KiB
C++
198 lines
5.8 KiB
C++
/* Copyright (c) 2010-2017 Benjamin Dobell, Glass Echidna
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.*/
|
|
|
|
// C Standard Library
|
|
#include <stdio.h>
|
|
|
|
// Heimdall
|
|
#include "Arguments.h"
|
|
#include "BridgeManager.h"
|
|
#include "Heimdall.h"
|
|
#include "Interface.h"
|
|
#include "PrintPitAction.h"
|
|
|
|
using namespace std;
|
|
using namespace libpit;
|
|
using namespace Heimdall;
|
|
|
|
const char *PrintPitAction::usage = "Action: print-pit\n\
|
|
Arguments: [--file <filename>] [--verbose] [--no-reboot] [--stdout-errors]\n\
|
|
[--usb-log-level <none/error/warning/debug>]\n\
|
|
Description: Prints the contents of a PIT file in a human readable format. If\n\
|
|
a filename is not provided then Heimdall retrieves the PIT file from the \n\
|
|
connected device.\n\
|
|
Note: --no-reboot causes the device to remain in download mode after the action\n\
|
|
is completed. If you wish to perform another action whilst remaining in\n\
|
|
download mode, then the following action must specify the --resume flag.\n";
|
|
|
|
int PrintPitAction::Execute(int argc, char **argv)
|
|
{
|
|
// Handle arguments
|
|
|
|
map<string, ArgumentType> argumentTypes;
|
|
argumentTypes["file"] = kArgumentTypeString;
|
|
argumentTypes["no-reboot"] = kArgumentTypeFlag;
|
|
argumentTypes["resume"] = kArgumentTypeFlag;
|
|
argumentTypes["verbose"] = kArgumentTypeFlag;
|
|
argumentTypes["stdout-errors"] = kArgumentTypeFlag;
|
|
argumentTypes["usb-log-level"] = kArgumentTypeString;
|
|
|
|
Arguments arguments(argumentTypes);
|
|
|
|
if (!arguments.ParseArguments(argc, argv, 2))
|
|
{
|
|
Interface::Print(PrintPitAction::usage);
|
|
return (0);
|
|
}
|
|
|
|
const StringArgument *fileArgument = static_cast<const StringArgument *>(arguments.GetArgument("file"));
|
|
|
|
bool reboot = arguments.GetArgument("no-reboot") == nullptr;
|
|
bool resume = arguments.GetArgument("resume") != nullptr;
|
|
bool verbose = arguments.GetArgument("verbose") != nullptr;
|
|
|
|
if (arguments.GetArgument("stdout-errors") != nullptr)
|
|
Interface::SetStdoutErrors(true);
|
|
|
|
const StringArgument *usbLogLevelArgument = static_cast<const StringArgument *>(arguments.GetArgument("usb-log-level"));
|
|
|
|
BridgeManager::UsbLogLevel usbLogLevel = BridgeManager::UsbLogLevel::Default;
|
|
|
|
if (usbLogLevelArgument)
|
|
{
|
|
const string& usbLogLevelString = usbLogLevelArgument->GetValue();
|
|
|
|
if (usbLogLevelString.compare("none") == 0 || usbLogLevelString.compare("NONE") == 0)
|
|
{
|
|
usbLogLevel = BridgeManager::UsbLogLevel::None;
|
|
}
|
|
else if (usbLogLevelString.compare("error") == 0 || usbLogLevelString.compare("ERROR") == 0)
|
|
{
|
|
usbLogLevel = BridgeManager::UsbLogLevel::Error;
|
|
}
|
|
else if (usbLogLevelString.compare("warning") == 0 || usbLogLevelString.compare("WARNING") == 0)
|
|
{
|
|
usbLogLevel = BridgeManager::UsbLogLevel::Warning;
|
|
}
|
|
else if (usbLogLevelString.compare("info") == 0 || usbLogLevelString.compare("INFO") == 0)
|
|
{
|
|
usbLogLevel = BridgeManager::UsbLogLevel::Info;
|
|
}
|
|
else if (usbLogLevelString.compare("debug") == 0 || usbLogLevelString.compare("DEBUG") == 0)
|
|
{
|
|
usbLogLevel = BridgeManager::UsbLogLevel::Debug;
|
|
}
|
|
else
|
|
{
|
|
Interface::Print("Unknown USB log level: %s\n\n", usbLogLevelString.c_str());
|
|
Interface::Print(PrintPitAction::usage);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
// Open file (if specified).
|
|
|
|
FILE *localPitFile = nullptr;
|
|
|
|
if (fileArgument)
|
|
{
|
|
const char *filename = fileArgument->GetValue().c_str();
|
|
|
|
localPitFile = FileOpen(filename, "rb");
|
|
|
|
if (!localPitFile)
|
|
{
|
|
Interface::PrintError("Failed to open file \"%s\"\n", filename);
|
|
return (1);
|
|
}
|
|
}
|
|
|
|
// Info
|
|
|
|
Interface::PrintReleaseInfo();
|
|
Sleep(1000);
|
|
|
|
if (localPitFile)
|
|
{
|
|
// Print PIT from file; there's no need for a BridgeManager.
|
|
|
|
FileSeek(localPitFile, 0, SEEK_END);
|
|
unsigned int localPitFileSize = (unsigned int)FileTell(localPitFile);
|
|
FileRewind(localPitFile);
|
|
|
|
// Load the local pit file into memory.
|
|
unsigned char *pitFileBuffer = new unsigned char[localPitFileSize];
|
|
(void)fread(pitFileBuffer, 1, localPitFileSize, localPitFile);
|
|
FileClose(localPitFile);
|
|
|
|
PitData *pitData = new PitData();
|
|
pitData->Unpack(pitFileBuffer);
|
|
|
|
delete [] pitFileBuffer;
|
|
|
|
Interface::PrintPit(pitData);
|
|
delete pitData;
|
|
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
// Print PIT from a device.
|
|
|
|
BridgeManager *bridgeManager = new BridgeManager(verbose);
|
|
bridgeManager->SetUsbLogLevel(usbLogLevel);
|
|
|
|
if (bridgeManager->Initialise(resume) != BridgeManager::kInitialiseSucceeded || !bridgeManager->BeginSession())
|
|
{
|
|
delete bridgeManager;
|
|
return (1);
|
|
}
|
|
|
|
unsigned char *devicePit;
|
|
bool success = bridgeManager->DownloadPitFile(&devicePit) != 0;
|
|
|
|
if (success)
|
|
{
|
|
PitData *pitData = new PitData();
|
|
|
|
if (pitData->Unpack(devicePit))
|
|
{
|
|
Interface::PrintPit(pitData);
|
|
}
|
|
else
|
|
{
|
|
Interface::PrintError("Failed to unpack device's PIT file!\n");
|
|
success = false;
|
|
}
|
|
|
|
delete pitData;
|
|
}
|
|
|
|
delete [] devicePit;
|
|
|
|
if (!bridgeManager->EndSession(reboot))
|
|
success = false;
|
|
|
|
delete bridgeManager;
|
|
|
|
return (success ? 0 : 1);
|
|
}
|
|
}
|