1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
|
/* sodata.cpp
* Copyright (C) 2003 Tommi Maekitalo
*
* This program 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 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* is provided AS IS, WITHOUT ANY WARRANTY; without even the implied
* warranty of MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, and
* NON-INFRINGEMENT. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
#include <tnt/sodata.h>
#include <tnt/component.h>
#include <cxxtools/thread.h>
#include <cxxtools/dlloader.h>
#include <cxxtools/log.h>
#include <zlib.h>
#include <stdlib.h>
#include <string.h>
log_define("tntnet.data")
namespace tnt
{
static cxxtools::Mutex mutex;
void Sodata::setLangSuffix()
{
const char* LANG = getenv("LANG");
std::string sosuffix = std::string(1, '.');
if (LANG)
sosuffix += LANG;
setSoSuffix(sosuffix);
}
void Sodata::addRef(const Compident& ci)
{
cxxtools::MutexLock lock(mutex);
if (refs++ == 0)
{
// read data from shared library
log_debug("load library " << ci.libname << sosuffix);
cxxtools::dl::Library so((ci.libname + sosuffix).c_str());
cxxtools::dl::Symbol datalen_sym = so.sym((ci.compname + "_datalen").c_str());
unsigned datalen = *reinterpret_cast<unsigned*>(datalen_sym.getSym());
try
{
// try to read compressed data
cxxtools::dl::Symbol zdata_sym = so.sym((ci.compname + "_zdata").c_str());
cxxtools::dl::Symbol zdatalen_sym = so.sym((ci.compname + "_zdatalen").c_str());
const char** zdata = reinterpret_cast<const char**>(zdata_sym.getSym());
unsigned zdatalen = *reinterpret_cast<unsigned*>(zdatalen_sym.getSym());
log_debug(zdatalen << " bytes compressed data, " << datalen
<< " bytes uncompressed");
// ... and uncompress it
data = new char[datalen];
uLong destlen = datalen;
int z_ret = uncompress((Bytef*)data, &destlen, (const Bytef*)*zdata, zdatalen);
if (z_ret != Z_OK)
{
throw std::runtime_error(std::string("error uncompressing data: ") +
(z_ret == Z_MEM_ERROR ? "Z_MEM_ERROR" :
z_ret == Z_BUF_ERROR ? "Z_BUF_ERROR" :
z_ret == Z_DATA_ERROR ? "Z_DATA_ERROR" : "unknown error"));
}
log_debug("uncompress ready");
}
catch(const cxxtools::dl::SymbolNotFound&)
{
// compressed data not found - try uncompressed
cxxtools::dl::Symbol data_sym = so.sym((ci.compname + "_data").c_str());
const char** srcdata = reinterpret_cast<const char**>(data_sym.getSym());
log_debug(datalen << " bytes data");
// ... and copy it
data = new char[datalen];
memcpy(data, *srcdata, datalen);
}
}
}
void Sodata::release()
{
cxxtools::MutexLock lock(mutex);
if (--refs <= 0)
{
log_debug("release");
delete[] data;
data = 0;
}
}
}
|