mariadb/storage/ibmdb2i/db2i_ileBridge.cc
Davi Arnaut f56dd32bf7 Bug#34043: Server loops excessively in _checkchunk() when safemalloc is enabled
Essentially, the problem is that safemalloc is excruciatingly
slow as it checks all allocated blocks for overrun at each
memory management primitive, yielding a almost exponential
slowdown for the memory management functions (malloc, realloc,
free). The overrun check basically consists of verifying some
bytes of a block for certain magic keys, which catches some
simple forms of overrun. Another minor problem is violation
of aliasing rules and that its own internal list of blocks
is prone to corruption.

Another issue with safemalloc is rather the maintenance cost
as the tool has a significant impact on the server code.
Given the magnitude of memory debuggers available nowadays,
especially those that are provided with the platform malloc
implementation, maintenance of a in-house and largely obsolete
memory debugger becomes a burden that is not worth the effort
due to its slowness and lack of support for detecting more
common forms of heap corruption.

Since there are third-party tools that can provide the same
functionality at a lower or comparable performance cost, the
solution is to simply remove safemalloc. Third-party tools
can provide the same functionality at a lower or comparable
performance cost. 

The removal of safemalloc also allows a simplification of the
malloc wrappers, removing quite a bit of kludge: redefinition
of my_malloc, my_free and the removal of the unused second
argument of my_free. Since free() always check whether the
supplied pointer is null, redudant checks are also removed.

Also, this patch adds unit testing for my_malloc and moves
my_realloc implementation into the same file as the other
memory allocation primitives.

client/mysqldump.c:
  Pass my_free directly as its signature is compatible with the
  callback type -- which wasn't the case for free_table_ent.
2010-07-08 18:20:08 -03:00

1342 lines
36 KiB
C++

/*
Licensed Materials - Property of IBM
DB2 Storage Engine Enablement
Copyright IBM Corporation 2007,2008
All rights reserved
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
(a) Redistributions of source code must retain this list of conditions, the
copyright notice in section {d} below, and the disclaimer following this
list of conditions.
(b) Redistributions in binary form must reproduce this list of conditions, the
copyright notice in section (d) below, and the disclaimer following this
list of conditions, in the documentation and/or other materials provided
with the distribution.
(c) The name of IBM may not be used to endorse or promote products derived from
this software without specific prior written permission.
(d) The text of the required copyright notice is:
Licensed Materials - Property of IBM
DB2 Storage Engine Enablement
Copyright IBM Corporation 2007,2008
All rights reserved
THIS SOFTWARE IS PROVIDED BY IBM CORPORATION "AS IS" AND ANY EXPRESS OR IMPLIED
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
SHALL IBM CORPORATION BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
OF SUCH DAMAGE.
*/
#include "db2i_ileBridge.h"
#include "my_dbug.h"
#include "db2i_global.h"
#include "db2i_charsetSupport.h"
#include "db2i_errors.h"
// static class member data
ILEpointer* db2i_ileBridge::functionSymbols;
db2i_ileBridge* db2i_ileBridge::globalBridge;
#ifndef DBUG_OFF
uint32 db2i_ileBridge::registeredPtrs;
#endif
pthread_key(IleParms*, THR_ILEPARMS);
static void ileParmsDtor(void* parmsToFree)
{
if (parmsToFree)
{
free_aligned(parmsToFree);
DBUG_PRINT("db2i_ileBridge", ("Freeing space for parms"));
}
}
/**
Convert a timestamp in ILE time format into a unix time_t
*/
static inline time_t convertILEtime(const ILE_time_t& input)
{
tm temp;
temp.tm_sec = input.Second;
temp.tm_min = input.Minute;
temp.tm_hour = input.Hour;
temp.tm_mday = input.Day;
temp.tm_mon = input.Month-1;
temp.tm_year = input.Year - 1900;
temp.tm_isdst = -1;
return mktime(&temp);
}
/**
Allocate and intialize a new bridge structure
*/
db2i_ileBridge* db2i_ileBridge::createNewBridge(CONNECTION_HANDLE connID)
{
DBUG_PRINT("db2i_ileBridge::createNewBridge",("Building new bridge..."));
db2i_ileBridge* newBridge = (db2i_ileBridge*)my_malloc(sizeof(db2i_ileBridge), MYF(MY_WME));
if (unlikely(newBridge == NULL))
return NULL;
newBridge->stmtTxActive = false;
newBridge->connErrText = NULL;
newBridge->pendingLockedHandles.head = NULL;
newBridge->cachedConnectionID = connID;
return newBridge;
}
void db2i_ileBridge::destroyBridge(db2i_ileBridge* bridge)
{
bridge->freeErrorStorage();
my_free(bridge);
}
void db2i_ileBridge::destroyBridgeForThread(const THD* thd)
{
void* thdData = *thd_ha_data(thd, ibmdb2i_hton);
if (thdData != NULL)
{
destroyBridge((db2i_ileBridge*)thdData);
}
}
void db2i_ileBridge::registerPtr(const void* ptr, ILEMemHandle* receiver)
{
static const arg_type_t ileSignature[] = { ARG_MEMPTR, ARG_END };
if (unlikely(ptr == NULL))
{
*receiver = 0;
return;
}
struct ArgList
{
ILEarglist_base base;
ILEpointer ptr;
} *arguments;
char argBuf[sizeof(ArgList)+15];
arguments = (ArgList*)roundToQuadWordBdy(argBuf);
arguments->ptr.s.addr = (address64_t)(ptr);
_ILECALL(&functionSymbols[funcRegisterSpace],
&arguments->base,
ileSignature,
RESULT_INT64);
#ifndef DBUG_OFF
uint32 truncHandle = arguments->base.result.r_uint64;
DBUG_PRINT("db2i_ileBridge::registerPtr",("Register 0x%p with handle %d", ptr, truncHandle));
getBridgeForThread()->registeredPtrs++;
#endif
*receiver = arguments->base.result.r_uint64;
return;
}
void db2i_ileBridge::unregisterPtr(ILEMemHandle handle)
{
static const arg_type_t ileSignature[] = { ARG_UINT64, ARG_END };
if (unlikely(handle == NULL))
return;
struct ArgList
{
ILEarglist_base base;
uint64 handle;
} *arguments;
char argBuf[sizeof(ArgList)+15];
arguments = (ArgList*)roundToQuadWordBdy(argBuf);
arguments->handle = (uint64)(handle);
_ILECALL(&functionSymbols[funcUnregisterSpace],
&arguments->base,
ileSignature,
RESULT_VOID);
#ifndef DBUG_OFF
DBUG_PRINT("db2i_ileBridge::unregisterPtr",("Unregister handle %d", (uint32)handle));
getBridgeForThread()->registeredPtrs--;
#endif
}
/**
Initialize the bridge component
@details Resolves srvpgm and function names of the APIs. If this fails,
the approrpiate operating system support (PTFs) is probably not installed.
WARNING:
Must be called before any other functions in this class are used!!!!
Can only be called by a single thread!
*/
int db2i_ileBridge::setup()
{
static const char funcNames[db2i_ileBridge::funcListEnd][32] =
{
{"QmyRegisterParameterSpaces"},
{"QmyRegisterSpace"},
{"QmyUnregisterSpace"},
{"QmyProcessRequest"}
};
DBUG_ENTER("db2i_ileBridge::setup");
int actmark = _ILELOAD("QSYS/QMYSE", ILELOAD_LIBOBJ);
if ( actmark == -1 )
{
DBUG_PRINT("db2i_ileBridge::setup", ("srvpgm activation failed"));
DBUG_RETURN(1);
}
functionSymbols = (ILEpointer*)malloc_aligned(sizeof(ILEpointer) * db2i_ileBridge::funcListEnd);
for (int i = 0; i < db2i_ileBridge::funcListEnd; i++)
{
if (_ILESYM(&functionSymbols[i], actmark, funcNames[i]) == -1)
{
DBUG_PRINT("db2i_ileBridge::setup",
("resolve of %s failed", funcNames[i]));
DBUG_RETURN(errno);
}
}
pthread_key_create(&THR_ILEPARMS, &ileParmsDtor);
#ifndef DBUG_OFF
registeredPtrs = 0;
#endif
globalBridge = createNewBridge(0);
DBUG_RETURN(0);
}
/**
Cleanup any resources before shutting down plug-in
*/
void db2i_ileBridge::takedown()
{
if (globalBridge)
destroyBridge(globalBridge);
free_aligned(functionSymbols);
}
/**
Call off to QmyProcessRequest to perform the API that the caller prepared
*/
inline int32 db2i_ileBridge::doIt()
{
static const arg_type_t ileSignature[] = {ARG_END};
struct ArgList
{
ILEarglist_base base;
} *arguments;
char argBuf[sizeof(ArgList)+15];
arguments = (ArgList*)roundToQuadWordBdy(argBuf);
_ILECALL(&functionSymbols[funcProcessRequest],
&arguments->base,
ileSignature,
RESULT_INT32);
return translateErrorCode(arguments->base.result.s_int32.r_int32);
}
/**
Call off to QmyProcessRequest to perform the API that the caller prepared and
log any errors that may occur.
*/
inline int32 db2i_ileBridge::doItWithLog()
{
int32 rc = doIt();
if (unlikely(rc))
{
// Only report errors that we weren't expecting
if (rc != tacitErrors[0] &&
rc != tacitErrors[1] &&
rc != QMY_ERR_END_OF_BLOCK)
reportSystemAPIError(rc, (Qmy_Error_output_t*)parms()->outParms);
}
memset(tacitErrors, 0, sizeof(tacitErrors));
return rc;
}
/**
Interface to QMY_ALLOCATE_SHARE API
See QMY_ALLOCATE_SHARE documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::allocateFileDefn(ILEMemHandle definitionSpace,
ILEMemHandle handleSpace,
uint16 fileCount,
const char* schemaName,
uint16 schemaNameLength,
ILEMemHandle formatSpace,
uint32 formatSpaceLen)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MAOS0100 *input = (Qmy_MAOS0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_ALLOCATE_SHARE;
input->ShrDefSpcHnd = definitionSpace;
input->ShrHndSpcHnd = handleSpace;
input->ShrDefCnt = fileCount;
input->FmtSpcHnd = formatSpace;
input->FmtSpcLen = formatSpaceLen;
if (schemaNameLength > sizeof(input->SchNam))
{
// This should never happen!
DBUG_ASSERT(0);
return HA_ERR_GENERIC;
}
memcpy(input->SchNam, schemaName, schemaNameLength);
input->SchNamLen = schemaNameLength;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_ALLOCATE_INSTANCE API
See QMY_ALLOCATE_INSTANCE documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::allocateFileInstance(FILE_HANDLE defnHandle,
ILEMemHandle inuseSpace,
FILE_HANDLE* instance)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MAOI0100 *input = (Qmy_MAOI0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_ALLOCATE_INSTANCE;
input->ShrHnd = defnHandle;
input->CnnHnd = cachedConnectionID;
input->UseSpcHnd = inuseSpace;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MAOI0100_output* output = (Qmy_MAOI0100_output*)parmBlock->outParms;
DBUG_ASSERT(instance);
*instance = output->ObjHnd;
}
return rc;
}
/**
Interface to QMY_DEALLOCATE_OBJECT API
See QMY_DEALLOCATE_OBJECT documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::deallocateFile(FILE_HANDLE rfileHandle,
bool postDropTable)
{
IleParms* parmBlock = parms();
Qmy_MDLC0100 *input = (Qmy_MDLC0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DEALLOCATE_OBJECT;
input->ObjHnd = rfileHandle;
input->ObjDrp[0] = (postDropTable ? QMY_YES : QMY_NO);
DBUG_PRINT("db2i_ileBridge::deallocateFile", ("Deallocating %d", (uint32)rfileHandle));
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_OBJECT_INITIALIZATION API
See QMY_OBJECT_INITIALIZATION documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::initFileForIO(FILE_HANDLE rfileHandle,
char accessIntent,
char commitLevel,
uint16* inRecSize,
uint16* inRecNullOffset,
uint16* outRecSize,
uint16* outRecNullOffset)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MOIX0100 *input = (Qmy_MOIX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_OBJECT_INITIALIZATION;
input->CmtLvl[0] = commitLevel;
input->Intent[0] = accessIntent;
input->ObjHnd = rfileHandle;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MOIX0100_output* output = (Qmy_MOIX0100_output*)parmBlock->outParms;
*inRecSize = output->InNxtRowOff;
*inRecNullOffset = output->InNullMapOff;
*outRecSize = output->OutNxtRowOff;
*outRecNullOffset = output->OutNullMapOff;
}
return rc;
}
/**
Interface to QMY_READ_ROWS API for reading a row with a specific RRN.
See QMY_READ_ROWS documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::readByRRN(FILE_HANDLE rfileHandle,
ILEMemHandle buf,
uint32 inRRN,
char accessIntent,
char commitLevel)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MRDX0100 *input = (Qmy_MRDX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_READ_ROWS;
input->CmtLvl[0] = commitLevel;
input->ObjHnd = rfileHandle;
input->Intent[0] = accessIntent;
input->OutSpcHnd = (uint64)buf;
input->RelRowNbr = inRRN;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
if (rc == QMY_ERR_END_OF_BLOCK)
{
rc = 0;
DBUG_PRINT("db2i_ileBridge::readByRRN", ("End of block signalled"));
}
return rc;
}
/**
Interface to QMY_WRITE_ROWS API.
See QMY_WRITE_ROWS documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::writeRows(FILE_HANDLE rfileHandle,
ILEMemHandle buf,
char commitLevel,
int64* outIdVal,
bool* outIdGen,
uint32* dupKeyRRN,
char** dupKeyName,
uint32* dupKeyNameLen,
uint32* outIdIncrement)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MWRT0100 *input = (Qmy_MWRT0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_WRITE_ROWS;
input->CmtLvl[0] = commitLevel;
input->ObjHnd = rfileHandle;
input->InSpcHnd = (uint64_t) buf;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
Qmy_MWRT0100_output_t* output = (Qmy_MWRT0100_output_t*)parmBlock->outParms;
if (likely(rc == 0 || rc == HA_ERR_FOUND_DUPP_KEY))
{
DBUG_ASSERT(dupKeyRRN && dupKeyName && dupKeyNameLen && outIdGen && outIdIncrement && outIdVal);
*dupKeyRRN = output->DupRRN;
*dupKeyName = (char*)parmBlock->outParms + output->DupObjNamOff;
*dupKeyNameLen = output->DupObjNamLen;
*outIdGen = (output->NewIdGen[0] == QMY_YES ? TRUE : FALSE);
if (*outIdGen == TRUE)
{
*outIdIncrement = output->IdIncrement;
*outIdVal = output->NewIdVal;
}
}
return rc;
}
/**
Interface to QMY_EXECUTE_IMMEDIATE API.
See QMY_EXECUTE_IMMEDIATE documentation for more information about
parameters and return codes.
*/
uint32 db2i_ileBridge::execSQL(const char* statement,
uint32 statementCount,
uint8 commitLevel,
bool autoCreateSchema,
bool dropSchema,
bool noCommit,
FILE_HANDLE fileHandle)
{
IleParms* parmBlock = parms();
Qmy_MSEI0100 *input = (Qmy_MSEI0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_EXECUTE_IMMEDIATE;
registerPtr(statement, &input->StmtsSpcHnd);
input->NbrStmts = statementCount;
*(uint16*)(&input->StmtCCSID) = 850;
input->AutoCrtSchema[0] = (autoCreateSchema == TRUE ? QMY_YES : QMY_NO);
input->DropSchema[0] = (dropSchema == TRUE ? QMY_YES : QMY_NO);
input->CmtLvl[0] = commitLevel;
if ((commitLevel == QMY_NONE && statementCount == 1) || noCommit)
{
input->CmtBefore[0] = QMY_NO;
input->CmtAfter[0] = QMY_NO;
}
else
{
input->CmtBefore[0] = QMY_YES;
input->CmtAfter[0] = QMY_YES;
}
input->CnnHnd = current_thd->thread_id;
input->ObjHnd = fileHandle;
int32 rc = doItWithLog();
unregisterPtr(input->StmtsSpcHnd);
return rc;
}
/**
Interface to QMY_PREPARE_OPEN_CURSOR API.
See QMY_PREPARE_OPEN_CURSOR documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::prepOpen(const char* statement,
FILE_HANDLE* rfileHandle,
uint32* recLength)
{
IleParms* parmBlock = parms();
Qmy_MSPO0100 *input = (Qmy_MSPO0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_PREPARE_OPEN_CURSOR;
registerPtr(statement, &input->StmtsSpcHnd );
*(uint16*)(&input->StmtCCSID) = 850;
input->CnnHnd = current_thd->thread_id;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MSPO0100_output* output = (Qmy_MSPO0100_output*)parmBlock->outParms;
*rfileHandle = output->ObjHnd;
*recLength = max(output->InNxtRowOff, output->OutNxtRowOff);
}
unregisterPtr(input->StmtsSpcHnd);
return rc;
}
/**
Interface to QMY_DELETE_ROW API.
See QMY_DELETE_ROW documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::deleteRow(FILE_HANDLE rfileHandle,
uint32 rrn)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MDLT0100 *input = (Qmy_MDLT0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DELETE_ROW;
input->ObjHnd = rfileHandle;
input->RelRowNbr = rrn;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_UPDATE_ROW API.
See QMY_UPDATE_ROW documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::updateRow(FILE_HANDLE rfileHandle,
uint32 rrn,
ILEMemHandle buf,
uint32* dupKeyRRN,
char** dupKeyName,
uint32* dupKeyNameLen)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MUPD0100 *input = (Qmy_MUPD0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_UPDATE_ROW;
input->ObjHnd = rfileHandle;
input->InSpcHnd = (uint64)buf;
input->RelRowNbr = rrn;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
if (rc == HA_ERR_FOUND_DUPP_KEY)
{
Qmy_MUPD0100_output* output = (Qmy_MUPD0100_output*)parmBlock->outParms;
DBUG_ASSERT(dupKeyRRN && dupKeyName && dupKeyNameLen);
*dupKeyRRN = output->DupRRN;
*dupKeyName = (char*)parmBlock->outParms + output->DupObjNamOff;
*dupKeyNameLen = output->DupObjNamLen;
}
return rc;
}
/**
Interface to QMY_DESCRIBE_RANGE API.
See QMY_DESCRIBE_RANGE documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::recordsInRange(FILE_HANDLE defnHandle,
ILEMemHandle inSpc,
uint32 inKeyCnt,
uint32 inLiteralCnt,
uint32 inBoundsOff,
uint32 inLitDefOff,
uint32 inLiteralsOff,
uint32 inCutoff,
uint32 inSpcLen,
uint16 inEndByte,
uint64* outRecCnt,
uint16* outRtnCode)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MDRG0100 *input = (Qmy_MDRG0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DESCRIBE_RANGE;
input->ShrHnd = defnHandle;
input->SpcHnd = (uint64)inSpc;
input->KeyCnt = inKeyCnt;
input->LiteralCnt = inLiteralCnt;
input->BoundsOff = inBoundsOff;
input->LitDefOff = inLitDefOff;
input->LiteralsOff = inLiteralsOff;
input->Cutoff = inCutoff;
input->SpcLen = inSpcLen;
input->EndByte = inEndByte;
input->CnnHnd = cachedConnectionID;
int rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MDRG0100_output* output = (Qmy_MDRG0100_output*)parmBlock->outParms;
DBUG_ASSERT(outRecCnt && outRtnCode);
*outRecCnt = output->RecCnt;
*outRtnCode = output->RtnCode;
}
return rc;
}
/**
Interface to QMY_RELEASE_ROW API.
See QMY_RELEASE_ROW documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::rrlslck(FILE_HANDLE rfileHandle, char accessIntent)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MRRX0100 *input = (Qmy_MRRX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_RELEASE_ROW;
input->ObjHnd = rfileHandle;
input->CnnHnd = cachedConnectionID;
input->Intent[0] = accessIntent;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_LOCK_OBJECT API.
See QMY_LOCK_OBJECT documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::lockObj(FILE_HANDLE defnHandle,
uint64 lockVal,
char lockAction,
char lockType,
char lockTimeout)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MOLX0100 *input = (Qmy_MOLX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_LOCK_OBJECT;
input->ShrHnd = defnHandle;
input->LckTimeoutVal = lockVal;
input->Action[0] = lockAction;
input->LckTyp[0] = lockType;
input->LckTimeout[0] = lockTimeout;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_DESCRIBE_CONSTRAINTS API.
See QMY_DESCRIBE_CONSTRAINTS documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::constraints(FILE_HANDLE defnHandle,
ILEMemHandle inSpc,
uint32 inSpcLen,
uint32* outLen,
uint32* outCnt)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MDCT0100 *input = (Qmy_MDCT0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DESCRIBE_CONSTRAINTS;
input->ShrHnd = defnHandle;
input->CstSpcHnd = (uint64)inSpc;
input->CstSpcLen = inSpcLen;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MDCT0100_output* output = (Qmy_MDCT0100_output*)parmBlock->outParms;
DBUG_ASSERT(outLen && outCnt);
*outLen = output->NeededLen;
*outCnt = output->CstCnt;
}
return rc;
}
/**
Interface to QMY_REORGANIZE_TABLE API.
See QMY_REORGANIZE_TABLE documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::optimizeTable(FILE_HANDLE defnHandle)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MRGX0100 *input = (Qmy_MRGX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_REORGANIZE_TABLE;
input->ShrHnd = defnHandle;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_PROCESS_COMMITMENT_CONTROL API.
See QMY_PROCESS_COMMITMENT_CONTROL documentation for more information about
parameters and return codes.
*/
int32 db2i_ileBridge::commitmentControl(uint8 function)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MCCX0100 *input = (Qmy_MCCX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_PROCESS_COMMITMENT_CONTROL;
input->Function[0] = function;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_PROCESS_SAVEPOINT API.
See QMY_PROCESS_SAVEPOINT documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::savepoint(uint8 function,
const char* savepointName)
{
DBUG_ASSERT(cachedStateIsCoherent());
DBUG_PRINT("db2i_ileBridge::savepoint",("%d %s", (uint32)function, savepointName));
IleParms* parmBlock = parms();
Qmy_MSPX0100 *input = (Qmy_MSPX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
char* savPtNam = (char*)(input+1);
input->Format = QMY_PROCESS_SAVEPOINT;
if (strlen(savepointName) > MAX_DB2_SAVEPOINTNAME_LENGTH)
{
DBUG_ASSERT(0);
return HA_ERR_GENERIC;
}
strcpy(savPtNam, savepointName);
input->Function[0] = function;
input->SavPtNamOff = savPtNam - (char*)(input);
input->SavPtNamLen = strlen(savepointName);
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
static ILEMemHandle traceSpcHandle;
/**
Do initialization for the QMY_* APIs.
@parm aspName The name of the relational database to use for all
connections.
@return 0 if successful; error otherwise
*/
int32 db2i_ileBridge::initILE(const char* aspName,
uint16* traceCtlPtr)
{
// We forego the typical thread-based parms space because MySQL doesn't
// allow us to clean it up before checking for memory leaks. As a result
// we get a complaint about leaked memory on server shutdown.
int32 rc;
char inParms[db2i_ileBridge_MAX_INPARM_SIZE];
char outParms[db2i_ileBridge_MAX_OUTPARM_SIZE];
if (rc = registerParmSpace(inParms, outParms))
{
reportSystemAPIError(rc, NULL);
return rc;
}
registerPtr(traceCtlPtr, &traceSpcHandle);
struct ParmBlock
{
Qmy_MINI0100 parms;
} *parmBlock = (ParmBlock*)inParms;
memset(inParms, 0, sizeof(ParmBlock));
parmBlock->parms.Format = QMY_INITIALIZATION;
char paddedName[18];
if (strlen(aspName) > sizeof(paddedName))
{
getErrTxt(DB2I_ERR_BAD_RDB_NAME);
return DB2I_ERR_BAD_RDB_NAME;
}
memset(paddedName, ' ', sizeof(paddedName));
memcpy(paddedName, aspName, strlen(aspName));
convToEbcdic(paddedName, parmBlock->parms.RDBName, strlen(paddedName));
parmBlock->parms.RDBNamLen = strlen(paddedName);
parmBlock->parms.TrcSpcHnd = traceSpcHandle;
rc = doIt();
if (rc)
{
reportSystemAPIError(rc, (Qmy_Error_output_t*)outParms);
}
return rc;
}
/**
Signal to the QMY_ APIs to perform any cleanup they need to do.
*/
int32 db2i_ileBridge::exitILE()
{
IleParms* parmBlock = parms();
Qmy_MCLN0100 *input = (Qmy_MCLN0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_CLEANUP;
int32 rc = doIt();
if (rc)
{
reportSystemAPIError(rc, (Qmy_Error_output_t*)parmBlock->outParms);
}
unregisterPtr(traceSpcHandle);
DBUG_PRINT("db2i_ileBridge::exitILE", ("Registered ptrs remaining: %d", registeredPtrs));
#ifndef DBUG_OFF
if (registeredPtrs != 0)
printf("Oh no! IBMDB2I left some pointers registered. Count was %d.\n", registeredPtrs);
#endif
// This is needed to prevent SAFE_MALLOC from complaining at process termination.
my_pthread_setspecific_ptr(THR_ILEPARMS, NULL);
free_aligned(parmBlock);
return rc;
}
/**
Designate the specified addresses as parameter passing buffers.
@parm in Input to the API will go here; format is defined by the individual API
@parm out Output from the API will be; format is defined by the individual API
@return 0 if success; error otherwise
*/
int db2i_ileBridge::registerParmSpace(char* in, char* out)
{
static const arg_type_t ileSignature[] = { ARG_MEMPTR, ARG_MEMPTR, ARG_END };
struct ArgList
{
ILEarglist_base base;
ILEpointer input;
ILEpointer output;
} *arguments;
char argBuf[sizeof(ArgList)+15];
arguments = (ArgList*)roundToQuadWordBdy(argBuf);
arguments->input.s.addr = (address64_t)(in);
arguments->output.s.addr = (address64_t)(out);
_ILECALL(&functionSymbols[funcRegisterParameterSpaces],
&arguments->base,
ileSignature,
RESULT_INT32);
return arguments->base.result.s_int32.r_int32;
}
/**
Interface to QMY_OBJECT_OVERRIDE API.
See QMY_OBJECT_OVERRIDE documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::objectOverride(FILE_HANDLE rfileHandle,
ILEMemHandle buf,
uint32 recordWidth)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MOOX0100 *input = (Qmy_MOOX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_OBJECT_OVERRIDE;
input->ObjHnd = rfileHandle;
input->OutSpcHnd = (uint64)buf;
input->NxtRowOff = recordWidth;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_DESCRIBE_OBJECT API for obtaining table stats.
See QMY_DESCRIBE_OBJECT documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::retrieveTableInfo(FILE_HANDLE defnHandle,
uint16 dataRequested,
ha_statistics& stats,
ILEMemHandle inSpc)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MDSO0100 *input = (Qmy_MDSO0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DESCRIBE_OBJECT;
input->ShrHnd = defnHandle;
input->CnnHnd = cachedConnectionID;
if (dataRequested & objLength)
input->RtnObjLen[0] = QMY_YES;
if (dataRequested & rowCount)
input->RtnRowCnt[0] = QMY_YES;
if (dataRequested & deletedRowCount)
input->RtnDltRowCnt[0] = QMY_YES;
if (dataRequested & rowsPerKey)
{
input->RowKeyHnd = (uint64)inSpc;
input->RtnRowKey[0] = QMY_YES;
}
if (dataRequested & meanRowLen)
input->RtnMeanRowLen[0] = QMY_YES;
if (dataRequested & lastModTime)
input->RtnModTim[0] = QMY_YES;
if (dataRequested & createTime)
input->RtnCrtTim[0] = QMY_YES;
if (dataRequested & ioCount)
input->RtnEstIoCnt[0] = QMY_YES;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MDSO0100_output* output = (Qmy_MDSO0100_output*)parmBlock->outParms;
if (dataRequested & objLength)
stats.data_file_length = output->ObjLen;
if (dataRequested & rowCount)
stats.records= output->RowCnt;
if (dataRequested & deletedRowCount)
stats.deleted = output->DltRowCnt;
if (dataRequested & meanRowLen)
stats.mean_rec_length = output->MeanRowLen;
if (dataRequested & lastModTime)
stats.update_time = convertILEtime(output->ModTim);
if (dataRequested & createTime)
stats.create_time = convertILEtime(output->CrtTim);
if (dataRequested & ioCount)
stats.data_file_length = output->EstIoCnt;
}
return rc;
}
/**
Interface to QMY_DESCRIBE_OBJECT API for finding index size.
See QMY_DESCRIBE_OBJECT documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::retrieveIndexInfo(FILE_HANDLE defnHandle,
uint64* outPageCnt)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MDSO0100 *input = (Qmy_MDSO0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_DESCRIBE_OBJECT;
input->ShrHnd = defnHandle;
input->CnnHnd = cachedConnectionID;
input->RtnPageCnt[0] = QMY_YES;
int32 rc = doItWithLog();
if (likely(rc == 0))
{
Qmy_MDSO0100_output* output = (Qmy_MDSO0100_output*)parmBlock->outParms;
*outPageCnt = output->PageCnt;
}
return rc;
}
/**
Interface to QMY_CLOSE_CONNECTION API
See QMY_CLOSE_CONNECTION documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::closeConnection(CONNECTION_HANDLE conn)
{
IleParms* parmBlock = parms();
Qmy_MCCN0100 *input = (Qmy_MCCN0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_CLOSE_CONNECTION;
input->CnnHnd = conn;
int32 rc = doItWithLog();
return rc;
}
/**
Interface to QMY_INTERRUPT API
See QMY_INTERRUPT documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::readInterrupt(FILE_HANDLE fileHandle)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MINT0100 *input = (Qmy_MINT0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_INTERRUPT;
input->CnnHnd = cachedConnectionID;
input->ObjHnd = fileHandle;
int32 rc = doItWithLog();
if (rc == QMY_ERR_END_OF_BLOCK)
{
rc = 0;
DBUG_PRINT("db2i_ileBridge::readInterrupt", ("End of block signalled"));
}
return rc;
}
/**
Interface to QMY_READ_ROWS API
See QMY_READ_ROWS documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::read(FILE_HANDLE rfileHandle,
ILEMemHandle buf,
char accessIntent,
char commitLevel,
char orientation,
bool asyncRead,
ILEMemHandle rrn,
ILEMemHandle key,
uint32 keylen,
uint16 keyParts,
int pipeFD)
{
DBUG_ASSERT(cachedStateIsCoherent());
IleParms* parmBlock = parms();
Qmy_MRDX0100 *input = (Qmy_MRDX0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_READ_ROWS;
input->CmtLvl[0] = commitLevel;
input->ObjHnd = rfileHandle;
input->Intent[0] = accessIntent;
input->OutSpcHnd = (uint64)buf;
input->OutRRNSpcHnd = (uint64)rrn;
input->RtnData[0] = QMY_RETURN_DATA;
if (key)
{
input->KeySpcHnd = (uint64)key;
input->KeyColsLen = keylen;
input->KeyColsNbr = keyParts;
}
input->Async[0] = (asyncRead ? QMY_YES : QMY_NO);
input->PipeDesc = pipeFD;
input->Orientation[0] = orientation;
input->CnnHnd = cachedConnectionID;
int32 rc = doItWithLog();
// QMY_ERR_END_OF_BLOCK is informational only, so we ignore it.
if (rc == QMY_ERR_END_OF_BLOCK)
{
rc = 0;
DBUG_PRINT("db2i_ileBridge::read", ("End of block signalled"));
}
return rc;
}
/**
Interface to QMY_QUIESCE_OBJECT API
See QMY_QUIESCE_OBJECT documentation for more information about parameters and
return codes.
*/
int32 db2i_ileBridge::quiesceFileInstance(FILE_HANDLE rfileHandle)
{
IleParms* parmBlock = parms();
Qmy_MQSC0100 *input = (Qmy_MQSC0100*)&(parmBlock->inParms);
memset(input, 0, sizeof(*input));
input->Format = QMY_QUIESCE_OBJECT;
input->ObjHnd = rfileHandle;
int32 rc = doItWithLog();
#ifndef DBUG_OFF
if (unlikely(rc))
{
DBUG_ASSERT(0);
}
#endif
return rc;
}
void db2i_ileBridge::PreservedHandleList::add(const char* newname, FILE_HANDLE newhandle, IBMDB2I_SHARE* share)
{
NameHandlePair *newPair = (NameHandlePair*)my_malloc(sizeof(NameHandlePair), MYF(MY_WME));
newPair->next = head;
head = newPair;
strcpy(newPair->name, newname);
newPair->handle = newhandle;
newPair->share = share;
DBUG_PRINT("db2i_ileBridge", ("Added handle %d for %s", uint32(newhandle), newname));
}
FILE_HANDLE db2i_ileBridge::PreservedHandleList::findAndRemove(const char* fileName, IBMDB2I_SHARE** share)
{
NameHandlePair* current = head;
NameHandlePair* prev = NULL;
while (current)
{
NameHandlePair* next = current->next;
if (strcmp(fileName, current->name) == 0)
{
FILE_HANDLE tmp = current->handle;
*share = current->share;
if (prev)
prev->next = next;
if (current == head)
head = next;
my_free(current);
DBUG_PRINT("db2i_ileBridge", ("Found handle %d for %s", uint32(tmp), fileName));
return tmp;
}
prev = current;
current = next;
}
return 0;
}
IleParms* db2i_ileBridge::initParmsForThread()
{
IleParms* p = (IleParms*)malloc_aligned(sizeof(IleParms));
DBUG_ASSERT((uint64)(&(p->outParms))% 16 == 0); // Guarantee that outParms are aligned correctly
if (likely(p))
{
int32 rc = registerParmSpace((p->inParms), (p->outParms));
if (likely(rc == 0))
{
my_pthread_setspecific_ptr(THR_ILEPARMS, p);
DBUG_PRINT("db2i_ileBridge", ("Inited space for parms"));
return p;
}
else
reportSystemAPIError(rc, NULL);
}
return NULL;
}