|
|
@ -6,17 +6,16 @@
|
|
|
|
#include "crwimage_int.hpp"
|
|
|
|
#include "crwimage_int.hpp"
|
|
|
|
#include "futils.hpp"
|
|
|
|
#include "futils.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
#include <iostream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <string>
|
|
|
|
#include <string>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void remove(Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
void remove(Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
void add(Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
void add(Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
void help();
|
|
|
|
void help();
|
|
|
|
void write(const std::string& filename, const Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
void write(const std::string& filename, const Exiv2::Internal::CiffHeader* pHead);
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char* const argv[])
|
|
|
|
int main(int argc, char* const argv[]) try {
|
|
|
|
try {
|
|
|
|
|
|
|
|
if (argc != 2) {
|
|
|
|
if (argc != 2) {
|
|
|
|
std::cout << "Usage: " << argv[0] << " file\n";
|
|
|
|
std::cout << "Usage: " << argv[0] << " file\n";
|
|
|
|
std::cout << "Edit the CIFF structure of a CRW file\n";
|
|
|
|
std::cout << "Edit the CIFF structure of a CRW file\n";
|
|
|
@ -25,14 +24,15 @@ try {
|
|
|
|
|
|
|
|
|
|
|
|
std::string filename(argv[1]);
|
|
|
|
std::string filename(argv[1]);
|
|
|
|
Exiv2::FileIo io(filename);
|
|
|
|
Exiv2::FileIo io(filename);
|
|
|
|
if(io.open() != 0) {
|
|
|
|
if (io.open() != 0) {
|
|
|
|
throw Exiv2::Error(Exiv2::kerDataSourceOpenFailed, io.path(), Exiv2::strError());
|
|
|
|
throw Exiv2::Error(Exiv2::kerDataSourceOpenFailed, io.path(), Exiv2::strError());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Exiv2::IoCloser closer(io);
|
|
|
|
Exiv2::IoCloser closer(io);
|
|
|
|
|
|
|
|
|
|
|
|
// Ensure that this is a CRW image
|
|
|
|
// Ensure that this is a CRW image
|
|
|
|
if (!Exiv2::isCrwType(io, false)) {
|
|
|
|
if (!Exiv2::isCrwType(io, false)) {
|
|
|
|
if (io.error() || io.eof()) throw Exiv2::Error(Exiv2::kerFailedToReadImageData);
|
|
|
|
if (io.error() || io.eof())
|
|
|
|
|
|
|
|
throw Exiv2::Error(Exiv2::kerFailedToReadImageData);
|
|
|
|
throw Exiv2::Error(Exiv2::kerNotACrwImage);
|
|
|
|
throw Exiv2::Error(Exiv2::kerNotACrwImage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -40,7 +40,8 @@ try {
|
|
|
|
long len = (long)io.size();
|
|
|
|
long len = (long)io.size();
|
|
|
|
Exiv2::DataBuf buf(len);
|
|
|
|
Exiv2::DataBuf buf(len);
|
|
|
|
io.read(buf.pData_, len);
|
|
|
|
io.read(buf.pData_, len);
|
|
|
|
if (io.error() || io.eof()) throw Exiv2::Error(Exiv2::kerFailedToReadImageData);
|
|
|
|
if (io.error() || io.eof())
|
|
|
|
|
|
|
|
throw Exiv2::Error(Exiv2::kerFailedToReadImageData);
|
|
|
|
|
|
|
|
|
|
|
|
// Parse the image, starting with a CIFF header component
|
|
|
|
// Parse the image, starting with a CIFF header component
|
|
|
|
Exiv2::Internal::CiffHeader::AutoPtr parseTree(new Exiv2::Internal::CiffHeader);
|
|
|
|
Exiv2::Internal::CiffHeader::AutoPtr parseTree(new Exiv2::Internal::CiffHeader);
|
|
|
@ -53,18 +54,29 @@ try {
|
|
|
|
std::cout << "command> ";
|
|
|
|
std::cout << "command> ";
|
|
|
|
std::cin >> cmd;
|
|
|
|
std::cin >> cmd;
|
|
|
|
switch (cmd) {
|
|
|
|
switch (cmd) {
|
|
|
|
case 'q': go = false; break;
|
|
|
|
case 'q':
|
|
|
|
case 'p': parseTree->print(std::cout); break;
|
|
|
|
go = false;
|
|
|
|
case 'a': add(parseTree.get()); break;
|
|
|
|
break;
|
|
|
|
case 'd': remove(parseTree.get()); break;
|
|
|
|
case 'p':
|
|
|
|
case 'w': write(filename, parseTree.get()); break;
|
|
|
|
parseTree->print(std::cout);
|
|
|
|
case 'h': help(); break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a':
|
|
|
|
|
|
|
|
add(parseTree.get());
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'd':
|
|
|
|
|
|
|
|
remove(parseTree.get());
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'w':
|
|
|
|
|
|
|
|
write(filename, parseTree.get());
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'h':
|
|
|
|
|
|
|
|
help();
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} catch (Exiv2::AnyError& e) {
|
|
|
|
catch (Exiv2::AnyError& e) {
|
|
|
|
|
|
|
|
std::cerr << e << "\n";
|
|
|
|
std::cerr << e << "\n";
|
|
|
|
return -1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -75,12 +87,13 @@ void write(const std::string& filename, const Exiv2::Internal::CiffHeader* pHead
|
|
|
|
pHead->write(blob);
|
|
|
|
pHead->write(blob);
|
|
|
|
|
|
|
|
|
|
|
|
Exiv2::FileIo io(filename);
|
|
|
|
Exiv2::FileIo io(filename);
|
|
|
|
if(io.open("wb") != 0) {
|
|
|
|
if (io.open("wb") != 0) {
|
|
|
|
throw Exiv2::Error(Exiv2::kerDataSourceOpenFailed, io.path(), Exiv2::strError());
|
|
|
|
throw Exiv2::Error(Exiv2::kerDataSourceOpenFailed, io.path(), Exiv2::strError());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Exiv2::IoCloser closer(io);
|
|
|
|
Exiv2::IoCloser closer(io);
|
|
|
|
long ret = io.write(&blob[0], (long) blob.size());
|
|
|
|
long ret = io.write(&blob[0], (long)blob.size());
|
|
|
|
if (static_cast<size_t>(ret) != blob.size()) throw Exiv2::Error(Exiv2::kerImageWriteFailed);
|
|
|
|
if (static_cast<size_t>(ret) != blob.size())
|
|
|
|
|
|
|
|
throw Exiv2::Error(Exiv2::kerImageWriteFailed);
|
|
|
|
io.close();
|
|
|
|
io.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -91,14 +104,12 @@ void remove(Exiv2::Internal::CiffHeader* pHead)
|
|
|
|
std::cin >> std::hex >> crwTag;
|
|
|
|
std::cin >> std::hex >> crwTag;
|
|
|
|
std::cout << "crwDir> 0x";
|
|
|
|
std::cout << "crwDir> 0x";
|
|
|
|
std::cin >> std::hex >> crwDir;
|
|
|
|
std::cin >> std::hex >> crwDir;
|
|
|
|
std::cout << "Deleting tag 0x" << std::hex << crwTag
|
|
|
|
std::cout << "Deleting tag 0x" << std::hex << crwTag << " in dir 0x" << crwDir << ", ok? ";
|
|
|
|
<< " in dir 0x" << crwDir << ", ok? ";
|
|
|
|
|
|
|
|
char cmd;
|
|
|
|
char cmd;
|
|
|
|
std::cin >> cmd;
|
|
|
|
std::cin >> cmd;
|
|
|
|
if (cmd != 'n' && cmd != 'N') {
|
|
|
|
if (cmd != 'n' && cmd != 'N') {
|
|
|
|
pHead->remove(crwTag, crwDir);
|
|
|
|
pHead->remove(crwTag, crwDir);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
else {
|
|
|
|
|
|
|
|
std::cout << "Canceled.\n";
|
|
|
|
std::cout << "Canceled.\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -113,16 +124,14 @@ void add(Exiv2::Internal::CiffHeader* pHead)
|
|
|
|
std::cin >> std::hex >> crwDir;
|
|
|
|
std::cin >> std::hex >> crwDir;
|
|
|
|
std::cout << "size> ";
|
|
|
|
std::cout << "size> ";
|
|
|
|
std::cin >> std::dec >> size;
|
|
|
|
std::cin >> std::dec >> size;
|
|
|
|
std::cout << "Adding tag 0x" << std::hex << crwTag
|
|
|
|
std::cout << "Adding tag 0x" << std::hex << crwTag << " in dir 0x" << crwDir << ", " << size << " bytes, ok? ";
|
|
|
|
<< " in dir 0x" << crwDir << ", " << size << " bytes, ok? ";
|
|
|
|
|
|
|
|
char cmd;
|
|
|
|
char cmd;
|
|
|
|
std::cin >> cmd;
|
|
|
|
std::cin >> cmd;
|
|
|
|
if (cmd != 'n' && cmd != 'N') {
|
|
|
|
if (cmd != 'n' && cmd != 'N') {
|
|
|
|
Exiv2::DataBuf buf(size);
|
|
|
|
Exiv2::DataBuf buf(size);
|
|
|
|
std::memset(buf.pData_, 0xaa, size);
|
|
|
|
std::memset(buf.pData_, 0xaa, size);
|
|
|
|
pHead->add(crwTag, crwDir, buf);
|
|
|
|
pHead->add(crwTag, crwDir, buf);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
else {
|
|
|
|
|
|
|
|
std::cout << "Canceled.\n";
|
|
|
|
std::cout << "Canceled.\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|