/* uefitool.cpp Copyright (c) 2013, Nikolaj Schlej. All rights reserved. This program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License which accompanies this distribution. The full text of the license may be found at http://opensource.org/licenses/bsd-license.php THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. */ #include "uefitool.h" #include "ui_uefitool.h" UEFITool::UEFITool(QWidget *parent) : QMainWindow(parent), ui(new Ui::UEFITool) { ui->setupUi(this); ffsEngine = NULL; //Connect connect(ui->actionOpenImageFile, SIGNAL(triggered()), this, SLOT(openImageFile())); connect(ui->actionExtract, SIGNAL(triggered()), this, SLOT(extractAsIs())); connect(ui->actionExtractBody, SIGNAL(triggered()), this, SLOT(extractBody())); connect(ui->actionExtractUncompressed, SIGNAL(triggered()), this, SLOT(extractUncompressed())); connect(ui->actionInsertInto, SIGNAL(triggered()), this, SLOT(insertInto())); connect(ui->actionInsertBefore, SIGNAL(triggered()), this, SLOT(insertBefore())); connect(ui->actionInsertAfter, SIGNAL(triggered()), this, SLOT(insertAfter())); connect(ui->actionReplace, SIGNAL(triggered()), this, SLOT(replace())); connect(ui->actionRemove, SIGNAL(triggered()), this, SLOT(remove())); connect(ui->actionRebuild, SIGNAL(triggered()), this, SLOT(rebuild())); connect(ui->actionSaveImageFile, SIGNAL(triggered()), this, SLOT(saveImageFile())); connect(ui->actionChangeToNone, SIGNAL(triggered()), this, SLOT(changeToNone())); connect(ui->actionChangeToEfi11, SIGNAL(triggered()), this, SLOT(changeToEfi11())); connect(ui->actionChangeToTiano, SIGNAL(triggered()), this, SLOT(changeToTiano())); connect(ui->actionChangeToLzma, SIGNAL(triggered()), this, SLOT(changeToLzma())); connect(ui->actionAbout, SIGNAL(triggered()), this, SLOT(about())); connect(ui->actionAboutQt, SIGNAL(triggered()), this, SLOT(aboutQt())); connect(ui->actionQuit, SIGNAL(triggered()), this, SLOT(exit())); // Enable Drag-and-Drop actions this->setAcceptDrops(true); // Create menus createMenus(); // Initialize non-persistent data init(); } void UEFITool::createMenus() { // Capsule capsuleMenu.clear(); capsuleMenu.addAction(ui->actionExtract); capsuleMenu.addAction(ui->actionExtractBody); capsuleMenu.addSeparator(); capsuleMenu.addAction(ui->actionRebuild); // Image imageMenu.clear(); imageMenu.addAction(ui->actionExtract); imageMenu.addSeparator(); imageMenu.addAction(ui->actionRebuild); // Region regionMenu.clear(); regionMenu.addAction(ui->actionExtract); regionMenu.addSeparator(); regionMenu.addAction(ui->actionRebuild); // Padding paddingMenu.clear(); paddingMenu.addAction(ui->actionExtract); // Volume volumeMenu.clear(); volumeMenu.addAction(ui->actionExtract); volumeMenu.addAction(ui->actionExtractBody); volumeMenu.addSeparator(); volumeMenu.addAction(ui->actionRebuild); volumeMenu.addSeparator(); volumeMenu.addAction(ui->actionInsertInto); volumeMenu.addSeparator(); volumeMenu.addAction(ui->actionRemove); // File fileMenu.clear(); fileMenu.addAction(ui->actionExtract); fileMenu.addAction(ui->actionExtractBody); //fileMenu.addAction(ui->actionExtractUncompressed); fileMenu.addSeparator(); fileMenu.addAction(ui->actionRebuild); fileMenu.addSeparator(); fileMenu.addAction(ui->actionInsertInto); fileMenu.addAction(ui->actionInsertBefore); fileMenu.addAction(ui->actionInsertAfter); fileMenu.addSeparator(); fileMenu.addAction(ui->actionRemove); // Section sectionMenu.clear(); sectionMenu.addAction(ui->actionExtract); sectionMenu.addAction(ui->actionExtractBody); //sectionMenu.addAction(ui->actionExtractUncompressed); sectionMenu.addSeparator(); sectionMenu.addAction(ui->actionRebuild); sectionMenu.addSeparator(); sectionMenu.addAction(ui->actionInsertInto); sectionMenu.addAction(ui->actionInsertBefore); sectionMenu.addAction(ui->actionInsertAfter); sectionMenu.addSeparator(); sectionMenu.addAction(ui->actionRemove); sectionMenu.addSeparator(); sectionMenu.addMenu(ui->menuChangeCompressionTo); } UEFITool::~UEFITool() { delete ui; delete ffsEngine; } void UEFITool::init() { // Clear components ui->messageListWidget->clear(); ui->infoEdit->clear(); // Disable actions and menus ui->actionExtract->setDisabled(true); ui->actionExtractBody->setDisabled(true); ui->actionExtractUncompressed->setDisabled(true); ui->actionReplace->setDisabled(true); ui->actionRemove->setDisabled(true); ui->actionRebuild->setDisabled(true); ui->actionInsertInto->setDisabled(true); ui->actionInsertBefore->setDisabled(true); ui->actionInsertAfter->setDisabled(true); ui->actionSaveImageFile->setDisabled(true); ui->menuChangeCompressionTo->setDisabled(true); // Make new ffsEngine if (ffsEngine) delete ffsEngine; ffsEngine = new FfsEngine(this); ui->structureTreeView->setModel(ffsEngine->model()); // Connect connect(ui->structureTreeView, SIGNAL(collapsed(const QModelIndex &)), this, SLOT(resizeTreeViewColums(void))); connect(ui->structureTreeView, SIGNAL(expanded(const QModelIndex &)), this, SLOT(resizeTreeViewColums(void))); connect(ui->structureTreeView->selectionModel(), SIGNAL(currentChanged(const QModelIndex &, const QModelIndex &)), this, SLOT(populateUi(const QModelIndex &))); connect(ui->messageListWidget, SIGNAL(itemDoubleClicked(QListWidgetItem*)), this, SLOT(scrollTreeView(QListWidgetItem*))); resizeTreeViewColums(); } void UEFITool::populateUi(const QModelIndex ¤t) { if (!current.isValid()) return; TreeItem* item = static_cast(current.internalPointer()); UINT8 type = item->type(); UINT8 subtype = item->subtype(); UINT8 algorithm = item->compression(); ui->infoEdit->setPlainText(item->info()); ui->actionExtract->setDisabled(item->hasEmptyHeader() && item->hasEmptyBody() && item->hasEmptyTail()); ui->actionRebuild->setDisabled(item->hasEmptyHeader() && item->hasEmptyBody() && item->hasEmptyTail()); ui->actionExtractBody->setDisabled(item->hasEmptyHeader()); ui->actionRemove->setEnabled(type == TreeItem::Volume || type == TreeItem::File || type == TreeItem::Section); ui->actionInsertInto->setEnabled(type == TreeItem::Volume || (type == TreeItem::File && subtype != EFI_FV_FILETYPE_RAW && subtype != EFI_FV_FILETYPE_PAD) || (type == TreeItem::Section && (subtype == EFI_SECTION_COMPRESSION || subtype == EFI_SECTION_GUID_DEFINED || subtype == EFI_SECTION_DISPOSABLE))); ui->actionInsertBefore->setEnabled(type == TreeItem::File || type == TreeItem::Section); ui->actionInsertAfter->setEnabled(type == TreeItem::File || type == TreeItem::Section); ui->menuChangeCompressionTo->setEnabled(type == TreeItem::Section && subtype == EFI_SECTION_COMPRESSION && (algorithm == COMPRESSION_ALGORITHM_NONE || COMPRESSION_ALGORITHM_EFI11 || algorithm == COMPRESSION_ALGORITHM_TIANO || algorithm == COMPRESSION_ALGORITHM_LZMA)); } void UEFITool::rebuild() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->rebuild(index); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::remove() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->remove(index); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::insert(const UINT8 mode) { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; TreeItem* item = static_cast(index.internalPointer()); UINT8 type; UINT8 objectType; if (mode == INSERT_MODE_BEFORE || mode == INSERT_MODE_AFTER) type = item->parent()->type(); else type = item->type(); QString path; switch (type) { case TreeItem::Volume: path = QFileDialog::getOpenFileName(this, tr("Select FFS file to insert"),".","FFS files (*.ffs *.bin);;All files (*.*)"); objectType = TreeItem::File; break; case TreeItem::File: case TreeItem::Section: path = QFileDialog::getOpenFileName(this, tr("Select section file to insert"),".","Section files (*.sct *.bin);;All files (*.*)"); objectType = TreeItem::Section; break; default: return; } QFileInfo fileInfo = QFileInfo(path); if (!fileInfo.exists()) { ui->statusBar->showMessage(tr("Please select existing file")); return; } QFile inputFile; inputFile.setFileName(path); if (!inputFile.open(QFile::ReadOnly)) { ui->statusBar->showMessage(tr("Can't open file for reading")); return; } QByteArray buffer = inputFile.readAll(); inputFile.close(); UINT8 result = ffsEngine->insert(index, buffer, objectType, mode); if (result) ui->statusBar->showMessage(tr("File can't be inserted (%1)").arg(result)); else ui->actionSaveImageFile->setEnabled(true); } void UEFITool::insertInto() { insert(INSERT_MODE_PREPEND); } void UEFITool::insertBefore() { insert(INSERT_MODE_BEFORE); } void UEFITool::insertAfter() { insert(INSERT_MODE_AFTER); } void UEFITool::replace() { } void UEFITool::changeToEfi11() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->changeCompression(index, COMPRESSION_ALGORITHM_EFI11); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::changeToTiano() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->changeCompression(index, COMPRESSION_ALGORITHM_TIANO); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::changeToLzma() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->changeCompression(index, COMPRESSION_ALGORITHM_LZMA); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::changeToNone() { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; UINT8 result = ffsEngine->changeCompression(index, COMPRESSION_ALGORITHM_NONE); if (result == ERR_SUCCESS) ui->actionSaveImageFile->setEnabled(true); } void UEFITool::about() { QMessageBox::about(this, tr("About UEFITool"), tr( "Copyright (c) 2013, Nikolaj Schlej aka CodeRush.

" "The program is dedicated to RevoGirl. Rest in peace, young genius.

" "The program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License.
" "The full text of the license may be found at OpenSource.org.

" "THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN \"AS IS\" BASIS, " "WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, " "EITHER EXPRESS OR IMPLIED.")); } void UEFITool::aboutQt() { QMessageBox::aboutQt(this, tr("About Qt")); } void UEFITool::exit() { QCoreApplication::exit(0); } void UEFITool::saveImageFile() { QString path = QFileDialog::getSaveFileName(this, tr("Save BIOS image file"),".","BIOS image files (*.rom *.bin *.cap *.fd *.wph *.efi);;All files (*.*)"); QFile outputFile; outputFile.setFileName(path); if (!outputFile.open(QFile::WriteOnly)) { ui->statusBar->showMessage(tr("Can't open file for writing")); return; } QByteArray reconstructed; UINT8 result = ffsEngine->reconstructImage(reconstructed); showMessage(); if (result) { ui->statusBar->showMessage(tr("Reconstruction failed (%1)").arg(result)); return; } outputFile.resize(0); outputFile.write(reconstructed); outputFile.close(); ui->statusBar->showMessage(tr("Reconstructed image written")); } void UEFITool::resizeTreeViewColums() { int count = ffsEngine->model()->columnCount(); for(int i = 0; i < count; i++) ui->structureTreeView->resizeColumnToContents(i); } void UEFITool::openImageFile() { QString path = QFileDialog::getOpenFileName(this, tr("Open BIOS image file"),".","BIOS image files (*.rom *.bin *.cap *.bio *.fd *.wph *.efi);;All files (*.*)"); openImageFile(path); } void UEFITool::openImageFile(QString path) { QFileInfo fileInfo = QFileInfo(path); if (!fileInfo.exists()) { ui->statusBar->showMessage(tr("Please select existing file")); return; } QFile inputFile; inputFile.setFileName(path); if (!inputFile.open(QFile::ReadOnly)) { ui->statusBar->showMessage(tr("Can't open file for reading")); return; } QByteArray buffer = inputFile.readAll(); inputFile.close(); init(); UINT8 result = ffsEngine->parseInputFile(buffer); showMessage(); if (result) ui->statusBar->showMessage(tr("Opened file can't be parsed (%1)").arg(result)); else ui->statusBar->showMessage(tr("Opened: %1").arg(fileInfo.fileName())); resizeTreeViewColums(); } void UEFITool::extract(const UINT8 mode) { QModelIndex index = ui->structureTreeView->selectionModel()->currentIndex(); if (!index.isValid()) return; TreeItem* item = static_cast(index.internalPointer()); UINT8 type = item->type(); QString path; if(mode == EXTRACT_MODE_AS_IS) { switch (type) { case TreeItem::Capsule: path = QFileDialog::getSaveFileName(this, tr("Save capsule to binary file"),".","Capsule files (*.cap *.bin);;All files (*.*)"); break; case TreeItem::Image: path = QFileDialog::getSaveFileName(this, tr("Save image to binary file"),".","Image files (*.rom *.bin);;All files (*.*)"); break; case TreeItem::Region: path = QFileDialog::getSaveFileName(this, tr("Save region to binary file"),".","Region files (*.rgn *.bin);;All files (*.*)"); break; case TreeItem::Padding: path = QFileDialog::getSaveFileName(this, tr("Save padding to binary file"),".","Padding files (*.pad *.bin);;All files (*.*)"); break; case TreeItem::Volume: path = QFileDialog::getSaveFileName(this, tr("Save volume to binary file"),".","Volume files (*.vol *.bin);;All files (*.*)"); break; case TreeItem::File: path = QFileDialog::getSaveFileName(this, tr("Save FFS file to binary file"),".","FFS files (*.ffs *.bin);;All files (*.*)"); break; case TreeItem::Section: path = QFileDialog::getSaveFileName(this, tr("Save section file to binary file"),".","Section files (*.sct *.bin);;All files (*.*)"); break; default: return; } } else path = QFileDialog::getSaveFileName(this, tr("Save object to binary file"),".","Binary files (*.bin);;All files (*.*)"); QFile outputFile; outputFile.setFileName(path); if (!outputFile.open(QFile::WriteOnly)) { ui->statusBar->showMessage(tr("Can't open file for rewriting")); return; } QByteArray extracted; UINT8 result = ffsEngine->extract(index, extracted, mode); if (result) ui->statusBar->showMessage(tr("File can't be extracted (%1)").arg(result)); else { outputFile.resize(0); outputFile.write(extracted); outputFile.close(); } } void UEFITool::extractAsIs() { extract(EXTRACT_MODE_AS_IS); } void UEFITool::extractBody() { extract(EXTRACT_MODE_BODY_ONLY); } void UEFITool::extractUncompressed() { extract(EXTRACT_MODE_UNCOMPRESSED); } void UEFITool::dragEnterEvent(QDragEnterEvent* event) { if (event->mimeData()->hasFormat("text/uri-list")) event->acceptProposedAction(); } void UEFITool::dropEvent(QDropEvent* event) { QString path = event->mimeData()->urls().at(0).toLocalFile(); openImageFile(path); } void UEFITool::showMessage() { ui->messageListWidget->clear(); if (!ffsEngine) return; messageItems = ffsEngine->message(); for (int i = 0; i < messageItems.count(); i++) { ui->messageListWidget->addItem(new MessageListItem(messageItems.at(i))); } } void UEFITool::scrollTreeView(QListWidgetItem* item) { MessageListItem* messageItem = (MessageListItem*) item; QModelIndex index = messageItem->index(); if (index.isValid()) { ui->structureTreeView->scrollTo(index); ui->structureTreeView->selectionModel()->clearSelection(); ui->structureTreeView->selectionModel()->select(index, QItemSelectionModel::Select); } } void UEFITool::contextMenuEvent (QContextMenuEvent* event) { if(!ui->structureTreeView->underMouse()) return; QPoint pt = event->pos(); QModelIndex index = ui->structureTreeView->indexAt(ui->structureTreeView->viewport()->mapFrom(this, pt)); if(!index.isValid()) return; TreeItem* item = static_cast(index.internalPointer()); switch(item->type()) { case TreeItem::Capsule: capsuleMenu.exec(event->globalPos()); break; case TreeItem::Image: imageMenu.exec(event->globalPos()); break; case TreeItem::Region: regionMenu.exec(event->globalPos()); break; case TreeItem::Padding: paddingMenu.exec(event->globalPos()); break; case TreeItem::Volume: volumeMenu.exec(event->globalPos()); break; case TreeItem::File: fileMenu.exec(event->globalPos()); break; case TreeItem::Section: sectionMenu.exec(event->globalPos()); break; } }