/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "Blob.hxx" #include "Connection.hxx" #include "Util.hxx" #include #include #include #include using namespace ::connectivity::firebird; using namespace ::cppu; using namespace ::osl; using namespace ::com::sun::star; using namespace ::com::sun::star::io; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::uno; Blob::Blob(isc_db_handle* pDatabaseHandle, isc_tr_handle* pTransactionHandle, ISC_QUAD& aBlobID): Blob_BASE(m_aMutex), m_pDatabaseHandle(pDatabaseHandle), m_pTransactionHandle(pTransactionHandle), m_blobID(aBlobID), m_blobHandle(0), m_bBlobOpened(false), m_nBlobLength(0), m_nBlobPosition(0) { } void Blob::ensureBlobIsOpened() throw(SQLException) { MutexGuard aGuard(m_aMutex); if (m_bBlobOpened) return; ISC_STATUS aErr; aErr = isc_open_blob2(m_statusVector, m_pDatabaseHandle, m_pTransactionHandle, &m_blobHandle, &m_blobID, 0, nullptr); if (aErr) evaluateStatusVector(m_statusVector, "isc_open_blob2", *this); m_bBlobOpened = true; m_nBlobPosition = 0; char aBlobItems[] = { isc_info_blob_total_length }; char aResultBuffer[20]; isc_blob_info(m_statusVector, &m_blobHandle, sizeof(aBlobItems), aBlobItems, sizeof(aResultBuffer), aResultBuffer); if (aErr) evaluateStatusVector(m_statusVector, "isc_blob_info", *this); if (*aResultBuffer == isc_info_blob_total_length) { short aResultLength = (short) isc_vax_integer(aResultBuffer+1, 2); m_nBlobLength = isc_vax_integer(aResultBuffer+3, aResultLength); } else { assert(false); } } void Blob::closeBlob() throw (SQLException) { MutexGuard aGuard(m_aMutex); if (m_bBlobOpened) { ISC_STATUS aErr; aErr = isc_close_blob(m_statusVector, &m_blobHandle); if (aErr) evaluateStatusVector(m_statusVector, "isc_close_blob", *this); m_bBlobOpened = false; m_blobHandle = 0; } } void SAL_CALL Blob::disposing() { try { closeBlob(); } catch (SQLException &e) { // we cannot throw any exceptions here... SAL_WARN("connectivity.firebird", "isc_close_blob failed\n" << e.Message); assert(false); } Blob_BASE::disposing(); } sal_Int64 SAL_CALL Blob::length() throw(SQLException, RuntimeException, std::exception) { MutexGuard aGuard(m_aMutex); checkDisposed(Blob_BASE::rBHelper.bDisposed); ensureBlobIsOpened(); return m_nBlobLength; } uno::Sequence< sal_Int8 > SAL_CALL Blob::getBytes(sal_Int64 nPosition, sal_Int32 nBytes) throw(SQLException, RuntimeException, std::exception) { MutexGuard aGuard(m_aMutex); checkDisposed(Blob_BASE::rBHelper.bDisposed); ensureBlobIsOpened(); if (nPosition > m_nBlobLength) throw lang::IllegalArgumentException("nPosition out of range", *this, 0); // We only have to read as many bytes as are available, i.e. nPosition+nBytes // can legally be greater than the total length, hence we don't bother to check. if (nPosition > m_nBlobPosition) { // Resets to the beginning (we can't seek these blobs) closeBlob(); ensureBlobIsOpened(); } skipBytes(nPosition - m_nBlobPosition); // Don't bother preallocating: readBytes does the appropriate calculations // and reallocates for us. uno::Sequence< sal_Int8 > aBytes; readBytes(aBytes, nBytes); return aBytes; } uno::Reference< XInputStream > SAL_CALL Blob::getBinaryStream() throw(SQLException, RuntimeException, std::exception) { return this; } sal_Int64 SAL_CALL Blob::position(const uno::Sequence< sal_Int8 >& /*rPattern*/, sal_Int64 /*nStart*/) throw(SQLException, RuntimeException, std::exception) { ::dbtools::throwFeatureNotImplementedSQLException("Blob::position", *this); return 0; } sal_Int64 SAL_CALL Blob::positionOfBlob(const uno::Reference< XBlob >& /*rPattern*/, sal_Int64 /*aStart*/) throw(SQLException, RuntimeException, std::exception) { ::dbtools::throwFeatureNotImplementedSQLException("Blob::positionOfBlob", *this); return 0; } // ---- XInputStream ---------------------------------------------------------- sal_Int32 SAL_CALL Blob::readBytes(uno::Sequence< sal_Int8 >& rDataOut, sal_Int32 nBytes) throw (NotConnectedException, BufferSizeExceededException, IOException, RuntimeException, std::exception) { MutexGuard aGuard(m_aMutex); try { checkDisposed(Blob_BASE::rBHelper.bDisposed); ensureBlobIsOpened(); } catch (const NotConnectedException&) { throw; } catch (const BufferSizeExceededException&) { throw; } catch (const IOException&) { throw; } catch (const RuntimeException&) { throw; } catch (const Exception& e) { css::uno::Any a(cppu::getCaughtException()); throw css::lang::WrappedTargetRuntimeException( "wrapped Exception " + e.Message, css::uno::Reference(), a); } // Ensure we have enough space for the amount of data we can actually read. const sal_Int64 nBytesAvailable = m_nBlobLength - m_nBlobPosition; const sal_Int32 nBytesToRead = nBytes < nBytesAvailable ? nBytes : nBytesAvailable; if (rDataOut.getLength() < nBytesToRead) rDataOut.realloc(nBytesToRead); sal_Int32 nTotalBytesRead = 0; ISC_STATUS aErr; while (nTotalBytesRead < nBytesToRead) { sal_uInt16 nBytesRead = 0; sal_uInt64 nDataRemaining = nBytesToRead - nTotalBytesRead; sal_uInt16 nReadSize = (nDataRemaining > SAL_MAX_UINT16) ? SAL_MAX_UINT16 : nDataRemaining; aErr = isc_get_segment(m_statusVector, &m_blobHandle, &nBytesRead, nReadSize, reinterpret_cast(rDataOut.getArray()) + nTotalBytesRead); if (aErr && IndicatesError(m_statusVector)) { OUString sError(StatusVectorToString(m_statusVector, "isc_get_segment")); throw IOException(sError, *this); } nTotalBytesRead += nBytesRead; m_nBlobPosition += nBytesRead; } return nTotalBytesRead; } sal_Int32 SAL_CALL Blob::readSomeBytes(uno::Sequence< sal_Int8 >& rDataOut, sal_Int32 nMaximumBytes) throw (NotConnectedException, BufferSizeExceededException, IOException, RuntimeException, std::exception) { // We don't have any way of verifying how many bytes are immediately available, // hence we just pass through direct to readBytes // (Spec: "reads the available number of bytes, at maximum nMaxBytesToRead.") return readBytes(rDataOut, nMaximumBytes); } void SAL_CALL Blob::skipBytes(sal_Int32 nBytesToSkip) throw (NotConnectedException, BufferSizeExceededException, IOException, RuntimeException, std::exception) { // There is no way of directly skipping, hence we have to pretend to skip // by reading & discarding the data. uno::Sequence< sal_Int8 > aBytes; readBytes(aBytes, nBytesToSkip); } sal_Int32 SAL_CALL Blob::available() throw (NotConnectedException, IOException, RuntimeException, std::exception) { MutexGuard aGuard(m_aMutex); try { checkDisposed(Blob_BASE::rBHelper.bDisposed); ensureBlobIsOpened(); } catch (const NotConnectedException&) { throw; } catch (const IOException&) { throw; } catch (const RuntimeException&) { throw; } catch (const Exception& e) { css::uno::Any a(cppu::getCaughtException()); throw css::lang::WrappedTargetRuntimeException( "wrapped Exception " + e.Message, css::uno::Reference(), a); } return m_nBlobLength - m_nBlobPosition; } void SAL_CALL Blob::closeInput() throw(NotConnectedException, IOException, RuntimeException, std::exception) { try { closeBlob(); } catch (const NotConnectedException&) { throw; } catch (const IOException&) { throw; } catch (const RuntimeException&) { throw; } catch (const Exception& e) { css::uno::Any a(cppu::getCaughtException()); throw css::lang::WrappedTargetRuntimeException( "wrapped Exception " + e.Message, css::uno::Reference(), a); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */