[iOS] Avoid clang 3.5 optimization bug (import/export kmz). Need to review in future.

This commit is contained in:
vng 2014-10-07 14:06:33 +03:00 committed by Alex Zolotarev
parent f501445ddc
commit 64a6968973
4 changed files with 51 additions and 22 deletions

View file

@ -1,3 +1,4 @@
#pragma once
static const size_t READ_FILE_BUFFER_SIZE = 512 * 1024;
static const size_t ZIP_FILE_BUFFER_SIZE = 64 * 1024;

View file

@ -4,8 +4,12 @@
#include "../../coding/file_name_utils.hpp"
#include "../../coding/internal/file_data.hpp"
#include "../../coding/reader.hpp"
#include "../../coding/constants.hpp"
#include "../../base/logging.hpp"
#include "../../base/scope_guard.hpp"
#include "../../std/vector.hpp"
#include "../../std/ctime.hpp"
#include "../../std/algorithm.hpp"
@ -53,6 +57,14 @@ void CreateTMZip(tm_zip & res)
bool CreateZipFromPathDeflatedAndDefaultCompression(string const & filePath, string const & zipFilePath)
{
/// Prepare buffer at the very beginning to avoid clang 3.5, loop optimization.
/// @todo Need to check with the new XCode (and clang) update.
size_t const bufSize = ZIP_FILE_BUFFER_SIZE;
vector<char> buffer(bufSize);
// 2. Open zip file for writing.
MY_SCOPE_GUARD(outFileGuard, bind(&my::DeleteFileX, cref(zipFilePath)));
ZipHandle zip(zipFilePath);
if (!zip.Handle())
return false;
@ -60,34 +72,43 @@ bool CreateZipFromPathDeflatedAndDefaultCompression(string const & filePath, str
// Special syntax to initialize struct with zeroes
zip_fileinfo zipInfo = zip_fileinfo();
CreateTMZip(zipInfo.tmz_date);
string fileName = filePath;
my::GetNameFromFullPath(fileName);
if (!strings::IsASCIIString(fileName))
fileName = "MapsWithMe.kml";
fileName = "MapsMe.kml";
if (::zipOpenNewFileInZip(zip.Handle(), fileName.c_str(), &zipInfo,
if (zipOpenNewFileInZip(zip.Handle(), fileName.c_str(), &zipInfo,
NULL, 0, NULL, 0, "ZIP from MapsWithMe", Z_DEFLATED, Z_DEFAULT_COMPRESSION) < 0)
{
return false;
}
my::FileData f(filePath, my::FileData::OP_READ);
size_t const bufSize = READ_FILE_BUFFER_SIZE;
vector<char> buffer(bufSize);
size_t const fileSize = f.Size();
size_t currSize = 0;
while (currSize < fileSize)
// Write source file into zip file.
try
{
size_t const toRead = min(bufSize, fileSize - currSize);
f.Read(currSize, &buffer[0], toRead);
my::FileData file(filePath, my::FileData::OP_READ);
size_t const fileSize = file.Size();
if (ZIP_OK != ::zipWriteInFileInZip(zip.Handle(), &buffer[0], toRead))
return false;
size_t currSize = 0;
while (currSize < fileSize)
{
size_t const toRead = min(bufSize, fileSize - currSize);
file.Read(currSize, &buffer[0], toRead);
currSize += toRead;
if (ZIP_OK != zipWriteInFileInZip(zip.Handle(), &buffer[0], toRead))
return false;
currSize += toRead;
}
}
catch (Reader::Exception const & ex)
{
LOG(LERROR, ("Error reading file:", filePath, ex.Msg()));
return false;
}
// Success.
outFileGuard.release();
return true;
}

View file

@ -4,6 +4,7 @@
#include "../base/logging.hpp"
#include "../coding/file_writer.hpp"
#include "../coding/constants.hpp"
#include "../std/bind.hpp"
@ -75,6 +76,12 @@ bool ZipFileReader::IsZip(string const & zipContainer)
void ZipFileReader::UnzipFile(string const & zipContainer, string const & fileInZip,
string const & outFilePath, ProgressFn progressFn)
{
/// Prepare buffer at the very beginning to avoid clang 3.5, loop optimization.
/// @todo Need to check with the new XCode (and clang) update.
size_t const bufSize = ZIP_FILE_BUFFER_SIZE;
vector<char> buf(bufSize);
unzFile zip = unzOpen64(zipContainer.c_str());
if (!zip)
MYTHROW(OpenZipException, ("Can't get zip file handle", zipContainer));
@ -91,9 +98,6 @@ void ZipFileReader::UnzipFile(string const & zipContainer, string const & fileIn
if (UNZ_OK != unzGetCurrentFileInfo64(zip, &fileInfo, NULL, 0, NULL, 0, NULL, 0))
MYTHROW(LocateZipException, ("Can't get uncompressed file size inside zip", fileInZip));
size_t const BUF_SIZE = 1024 * 50;
vector<char> buf(BUF_SIZE);
// First outFile should be closed, then FileWriter::DeleteFileX is called,
// so make correct order of guards.
MY_SCOPE_GUARD(outFileGuard, bind(&FileWriter::DeleteFileX, cref(outFilePath)));
@ -102,7 +106,7 @@ void ZipFileReader::UnzipFile(string const & zipContainer, string const & fileIn
uint64_t pos = 0;
while (true)
{
int const readBytes = unzReadCurrentFile(zip, &buf[0], BUF_SIZE);
int const readBytes = unzReadCurrentFile(zip, &buf[0], bufSize);
if (readBytes > 0)
outFile.Write(&buf[0], static_cast<size_t>(readBytes));
else if (readBytes < 0)

View file

@ -261,17 +261,20 @@
BookmarkCategory const * cat = GetFramework().GetBmCategory(m_categoryIndex);
if (cat)
{
NSString * filePath = [NSString stringWithUTF8String:cat->GetFileName().c_str()];
NSMutableString * catName = [NSMutableString stringWithUTF8String:cat->GetName().c_str()];
if (![catName length])
[catName setString:@"maps.me"];
[catName setString:@"MapsMe"];
NSString * filePath = [NSString stringWithUTF8String:cat->GetFileName().c_str()];
NSMutableString * kmzFile = [NSMutableString stringWithString:filePath];
[kmzFile replaceCharactersInRange:NSMakeRange([filePath length] - 1, 1) withString:@"z"];
if (CreateZipFromPathDeflatedAndDefaultCompression([filePath UTF8String], [kmzFile UTF8String]))
[self sendBookmarksWithExtension:@".kmz" andType:@"application/vnd.google-earth.kmz" andFile:kmzFile andCategory:catName];
else
[self sendBookmarksWithExtension:@".kml" andType:@"application/vnd.google-earth.kml+xml" andFile:filePath andCategory:catName];
my::DeleteFileX([kmzFile UTF8String]);
(void)my::DeleteFileX([kmzFile UTF8String]);
}
}
}