123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404 |
- /*
- * Copyright (c) 1988-1996 Sam Leffler
- * Copyright (c) 1991-1996 Silicon Graphics, Inc.
- *
- * Permission to use, copy, modify, distribute, and sell this software and
- * its documentation for any purpose is hereby granted without fee, provided
- * that (i) the above copyright notices and this permission notice appear in
- * all copies of the software and related documentation, and (ii) the names of
- * Sam Leffler and Silicon Graphics may not be used in any advertising or
- * publicity relating to the software without the specific, prior written
- * permission of Sam Leffler and Silicon Graphics.
- *
- * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
- * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
- *
- * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
- * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
- * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
- * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
- * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
- * OF THIS SOFTWARE.
- */
- /*
- * TIFF Library UNIX-specific Routines.
- */
- #include "tiffiop.h"
- #include <iostream>
- using namespace std;
- /*
- ISO C++ uses a 'std::streamsize' type to define counts. This makes
- it similar to, (but perhaps not the same as) size_t.
- The std::ios::pos_type is used to represent stream positions as used
- by tellg(), tellp(), seekg(), and seekp(). This makes it similar to
- (but perhaps not the same as) 'off_t'. The std::ios::streampos type
- is used for character streams, but is documented to not be an
- integral type anymore, so it should *not* be assigned to an integral
- type.
- The std::ios::off_type is used to specify relative offsets needed by
- the variants of seekg() and seekp() which accept a relative offset
- argument.
- Useful prototype knowledge:
- Obtain read position
- ios::pos_type basic_istream::tellg()
- Set read position
- basic_istream& basic_istream::seekg(ios::pos_type)
- basic_istream& basic_istream::seekg(ios::off_type, ios_base::seekdir)
- Read data
- basic_istream& istream::read(char *str, streamsize count)
- Number of characters read in last unformatted read
- streamsize istream::gcount();
- Obtain write position
- ios::pos_type basic_ostream::tellp()
- Set write position
- basic_ostream& basic_ostream::seekp(ios::pos_type)
- basic_ostream& basic_ostream::seekp(ios::off_type, ios_base::seekdir)
- Write data
- basic_ostream& ostream::write(const char *str, streamsize count)
- */
- struct tiffis_data;
- struct tiffos_data;
- extern "C"
- {
- static tmsize_t _tiffosReadProc(thandle_t, void *, tmsize_t);
- static tmsize_t _tiffisReadProc(thandle_t fd, void *buf, tmsize_t size);
- static tmsize_t _tiffosWriteProc(thandle_t fd, void *buf, tmsize_t size);
- static tmsize_t _tiffisWriteProc(thandle_t, void *, tmsize_t);
- static uint64_t _tiffosSeekProc(thandle_t fd, uint64_t off, int whence);
- static uint64_t _tiffisSeekProc(thandle_t fd, uint64_t off, int whence);
- static uint64_t _tiffosSizeProc(thandle_t fd);
- static uint64_t _tiffisSizeProc(thandle_t fd);
- static int _tiffosCloseProc(thandle_t fd);
- static int _tiffisCloseProc(thandle_t fd);
- static int _tiffDummyMapProc(thandle_t, void **base, toff_t *size);
- static void _tiffDummyUnmapProc(thandle_t, void *base, toff_t size);
- static TIFF *_tiffStreamOpen(const char *name, const char *mode, void *fd);
- struct tiffis_data
- {
- istream *stream;
- ios::pos_type start_pos;
- };
- struct tiffos_data
- {
- ostream *stream;
- ios::pos_type start_pos;
- };
- static tmsize_t _tiffosReadProc(thandle_t, void *, tmsize_t) { return 0; }
- static tmsize_t _tiffisReadProc(thandle_t fd, void *buf, tmsize_t size)
- {
- tiffis_data *data = reinterpret_cast<tiffis_data *>(fd);
- // Verify that type does not overflow.
- streamsize request_size = size;
- if (static_cast<tmsize_t>(request_size) != size)
- return static_cast<tmsize_t>(-1);
- data->stream->read((char *)buf, request_size);
- return static_cast<tmsize_t>(data->stream->gcount());
- }
- static tmsize_t _tiffosWriteProc(thandle_t fd, void *buf, tmsize_t size)
- {
- tiffos_data *data = reinterpret_cast<tiffos_data *>(fd);
- ostream *os = data->stream;
- ios::pos_type pos = os->tellp();
- // Verify that type does not overflow.
- streamsize request_size = size;
- if (static_cast<tmsize_t>(request_size) != size)
- return static_cast<tmsize_t>(-1);
- os->write(reinterpret_cast<const char *>(buf), request_size);
- return static_cast<tmsize_t>(os->tellp() - pos);
- }
- static tmsize_t _tiffisWriteProc(thandle_t, void *, tmsize_t) { return 0; }
- static uint64_t _tiffosSeekProc(thandle_t fd, uint64_t off, int whence)
- {
- tiffos_data *data = reinterpret_cast<tiffos_data *>(fd);
- ostream *os = data->stream;
- // if the stream has already failed, don't do anything
- if (os->fail())
- return static_cast<uint64_t>(-1);
- switch (whence)
- {
- case SEEK_SET:
- {
- // Compute 64-bit offset
- uint64_t new_offset =
- static_cast<uint64_t>(data->start_pos) + off;
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(new_offset);
- if (static_cast<uint64_t>(offset) != new_offset)
- return static_cast<uint64_t>(-1);
- os->seekp(offset, ios::beg);
- break;
- }
- case SEEK_CUR:
- {
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(off);
- if (static_cast<uint64_t>(offset) != off)
- return static_cast<uint64_t>(-1);
- os->seekp(offset, ios::cur);
- break;
- }
- case SEEK_END:
- {
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(off);
- if (static_cast<uint64_t>(offset) != off)
- return static_cast<uint64_t>(-1);
- os->seekp(offset, ios::end);
- break;
- }
- }
- // Attempt to workaround problems with seeking past the end of the
- // stream. ofstream doesn't have a problem with this but
- // ostrstream/ostringstream does. In that situation, add intermediate
- // '\0' characters.
- if (os->fail())
- {
- ios::iostate old_state;
- ios::pos_type origin;
- old_state = os->rdstate();
- // reset the fail bit or else tellp() won't work below
- os->clear(os->rdstate() & ~ios::failbit);
- switch (whence)
- {
- case SEEK_SET:
- default:
- origin = data->start_pos;
- break;
- case SEEK_CUR:
- origin = os->tellp();
- break;
- case SEEK_END:
- os->seekp(0, ios::end);
- origin = os->tellp();
- break;
- }
- // restore original stream state
- os->clear(old_state);
- // only do something if desired seek position is valid
- if ((static_cast<uint64_t>(origin) + off) >
- static_cast<uint64_t>(data->start_pos))
- {
- uint64_t num_fill;
- // clear the fail bit
- os->clear(os->rdstate() & ~ios::failbit);
- // extend the stream to the expected size
- os->seekp(0, ios::end);
- num_fill = (static_cast<uint64_t>(origin)) + off - os->tellp();
- for (uint64_t i = 0; i < num_fill; i++)
- os->put('\0');
- // retry the seek
- os->seekp(static_cast<ios::off_type>(
- static_cast<uint64_t>(origin) + off),
- ios::beg);
- }
- }
- return static_cast<uint64_t>(os->tellp());
- }
- static uint64_t _tiffisSeekProc(thandle_t fd, uint64_t off, int whence)
- {
- tiffis_data *data = reinterpret_cast<tiffis_data *>(fd);
- switch (whence)
- {
- case SEEK_SET:
- {
- // Compute 64-bit offset
- uint64_t new_offset =
- static_cast<uint64_t>(data->start_pos) + off;
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(new_offset);
- if (static_cast<uint64_t>(offset) != new_offset)
- return static_cast<uint64_t>(-1);
- data->stream->seekg(offset, ios::beg);
- break;
- }
- case SEEK_CUR:
- {
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(off);
- if (static_cast<uint64_t>(offset) != off)
- return static_cast<uint64_t>(-1);
- data->stream->seekg(offset, ios::cur);
- break;
- }
- case SEEK_END:
- {
- // Verify that value does not overflow
- ios::off_type offset = static_cast<ios::off_type>(off);
- if (static_cast<uint64_t>(offset) != off)
- return static_cast<uint64_t>(-1);
- data->stream->seekg(offset, ios::end);
- break;
- }
- }
- return (uint64_t)(data->stream->tellg() - data->start_pos);
- }
- static uint64_t _tiffosSizeProc(thandle_t fd)
- {
- tiffos_data *data = reinterpret_cast<tiffos_data *>(fd);
- ostream *os = data->stream;
- ios::pos_type pos = os->tellp();
- ios::pos_type len;
- os->seekp(0, ios::end);
- len = os->tellp();
- os->seekp(pos);
- return (uint64_t)len;
- }
- static uint64_t _tiffisSizeProc(thandle_t fd)
- {
- tiffis_data *data = reinterpret_cast<tiffis_data *>(fd);
- ios::pos_type pos = data->stream->tellg();
- ios::pos_type len;
- data->stream->seekg(0, ios::end);
- len = data->stream->tellg();
- data->stream->seekg(pos);
- return (uint64_t)len;
- }
- static int _tiffosCloseProc(thandle_t fd)
- {
- // Our stream was not allocated by us, so it shouldn't be closed by us.
- delete reinterpret_cast<tiffos_data *>(fd);
- return 0;
- }
- static int _tiffisCloseProc(thandle_t fd)
- {
- // Our stream was not allocated by us, so it shouldn't be closed by us.
- delete reinterpret_cast<tiffis_data *>(fd);
- return 0;
- }
- static int _tiffDummyMapProc(thandle_t, void **base, toff_t *size)
- {
- (void)base;
- (void)size;
- return (0);
- }
- static void _tiffDummyUnmapProc(thandle_t, void *base, toff_t size)
- {
- (void)base;
- (void)size;
- }
- /*
- * Open a TIFF file descriptor for read/writing.
- */
- static TIFF *_tiffStreamOpen(const char *name, const char *mode, void *fd)
- {
- TIFF *tif;
- if (strchr(mode, 'w'))
- {
- tiffos_data *data = new tiffos_data;
- data->stream = reinterpret_cast<ostream *>(fd);
- data->start_pos = data->stream->tellp();
- // Open for writing.
- tif = TIFFClientOpen(
- name, mode, reinterpret_cast<thandle_t>(data), _tiffosReadProc,
- _tiffosWriteProc, _tiffosSeekProc, _tiffosCloseProc,
- _tiffosSizeProc, _tiffDummyMapProc, _tiffDummyUnmapProc);
- if (!tif)
- {
- delete data;
- }
- }
- else
- {
- tiffis_data *data = new tiffis_data;
- data->stream = reinterpret_cast<istream *>(fd);
- data->start_pos = data->stream->tellg();
- // Open for reading.
- tif = TIFFClientOpen(
- name, mode, reinterpret_cast<thandle_t>(data), _tiffisReadProc,
- _tiffisWriteProc, _tiffisSeekProc, _tiffisCloseProc,
- _tiffisSizeProc, _tiffDummyMapProc, _tiffDummyUnmapProc);
- if (!tif)
- {
- delete data;
- }
- }
- return (tif);
- }
- } /* extern "C" */
- TIFF *TIFFStreamOpen(const char *name, ostream *os)
- {
- // If os is either a ostrstream or ostringstream, and has no data
- // written to it yet, then tellp() will return -1 which will break us.
- // We workaround this by writing out a dummy character and
- // then seek back to the beginning.
- if (!os->fail() && static_cast<int>(os->tellp()) < 0)
- {
- *os << '\0';
- os->seekp(0);
- }
- // NB: We don't support mapped files with streams so add 'm'
- return _tiffStreamOpen(name, "wm", os);
- }
- TIFF *TIFFStreamOpen(const char *name, istream *is)
- {
- // NB: We don't support mapped files with streams so add 'm'
- return _tiffStreamOpen(name, "rm", is);
- }
|