mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 21:12:26 +01:00
f5d0c77062
modified: storage/connect/array.cpp modified: storage/connect/blkfil.cpp modified: storage/connect/block.h modified: storage/connect/catalog.h modified: storage/connect/colblk.cpp modified: storage/connect/colblk.h modified: storage/connect/connect.cc modified: storage/connect/filamap.cpp modified: storage/connect/filamdbf.cpp modified: storage/connect/filamfix.cpp modified: storage/connect/filamtxt.cpp modified: storage/connect/filamtxt.h modified: storage/connect/filamvct.cpp modified: storage/connect/filamzip.cpp modified: storage/connect/filter.h modified: storage/connect/ha_connect.c modified: storage/connect/jsonudf.cpp modified: storage/connect/mycat.h modified: storage/connect/myconn.cpp modified: storage/connect/plgdbutl.cpp modified: storage/connect/reldef.cpp modified: storage/connect/reldef.h modified: storage/connect/tabcol.cpp modified: storage/connect/tabdos.cpp modified: storage/connect/tabdos.h modified: storage/connect/tabfix.cpp modified: storage/connect/tabfmt.cpp modified: storage/connect/tabfmt.h modified: storage/connect/tabjson.cpp modified: storage/connect/tabjson.h modified: storage/connect/table.cpp modified: storage/connect/tabmul.cpp modified: storage/connect/tabmysql.cpp modified: storage/connect/tabmysql.h modified: storage/connect/taboccur.cpp modified: storage/connect/tabpivot.cpp modified: storage/connect/tabsys.cpp modified: storage/connect/tabtbl.cpp modified: storage/connect/tabtbl.h modified: storage/connect/tabutil.cpp modified: storage/connect/tabutil.h modified: storage/connect/tabvct.cpp modified: storage/connect/tabvir.cpp modified: storage/connect/tabvir.h modified: storage/connect/tabxcl.cpp modified: storage/connect/tabxcl.h modified: storage/connect/tabxml.cpp modified: storage/connect/tabxml.h modified: storage/connect/valblk.cpp modified: storage/connect/valblk.h modified: storage/connect/value.cpp modified: storage/connect/value.h modified: storage/connect/xindex.cpp modified: storage/connect/xindex.h modified: storage/connect/xobject.h modified: storage/connect/xtable.h
305 lines
11 KiB
C++
305 lines
11 KiB
C++
/************* tdbvir C++ Program Source Code File (.CPP) **************/
|
|
/* PROGRAM NAME: tdbvir.cpp Version 1.1 */
|
|
/* (C) Copyright to the author Olivier BERTRAND 2014 */
|
|
/* This program are the VIR classes DB execution routines. */
|
|
/***********************************************************************/
|
|
|
|
/***********************************************************************/
|
|
/* Include relevant sections of the MariaDB header file. */
|
|
/***********************************************************************/
|
|
#include <my_global.h>
|
|
|
|
/***********************************************************************/
|
|
/* Include application header files: */
|
|
/* global.h is header containing all global declarations. */
|
|
/* plgdbsem.h is header containing the DB application declarations. */
|
|
/* xtable.h is header containing the TDBASE declarations. */
|
|
/* tdbvir.h is header containing the VIR classes declarations. */
|
|
/***********************************************************************/
|
|
#include "global.h"
|
|
#include "plgdbsem.h"
|
|
#include "filter.h"
|
|
#include "xtable.h"
|
|
#include "reldef.h"
|
|
#include "colblk.h"
|
|
#include "mycat.h" // for FNC_COL
|
|
#include "tabvir.h"
|
|
#include "resource.h" // for IDS_COLUMNS
|
|
|
|
/***********************************************************************/
|
|
/* Return the unique column definition to MariaDB. */
|
|
/***********************************************************************/
|
|
PQRYRES VirColumns(PGLOBAL g, bool info)
|
|
{
|
|
int buftyp[] = {TYPE_STRING, TYPE_SHORT, TYPE_STRING,
|
|
TYPE_INT, TYPE_STRING, TYPE_STRING};
|
|
XFLD fldtyp[] = {FLD_NAME, FLD_TYPE, FLD_TYPENAME,
|
|
FLD_PREC, FLD_KEY, FLD_EXTRA};
|
|
unsigned int length[] = {8, 4, 16, 4, 16, 16};
|
|
int i, n, ncol = sizeof(buftyp) / sizeof(int);
|
|
PQRYRES qrp;
|
|
PCOLRES crp;
|
|
|
|
n = (info) ? 0 : 1;
|
|
|
|
/**********************************************************************/
|
|
/* Allocate the structures used to refer to the result set. */
|
|
/**********************************************************************/
|
|
if (!(qrp = PlgAllocResult(g, ncol, n, IDS_COLUMNS + 3,
|
|
buftyp, fldtyp, length, false, true)))
|
|
return NULL;
|
|
|
|
// Some columns must be renamed before info
|
|
for (i = 0, crp = qrp->Colresp; crp; crp = crp->Next)
|
|
switch (++i) {
|
|
case 5: crp->Name = "Key"; break;
|
|
case 6: crp->Name = "Extra"; break;
|
|
} // endswitch i
|
|
|
|
if (info)
|
|
return qrp;
|
|
|
|
/**********************************************************************/
|
|
/* Now get the results into blocks. */
|
|
/**********************************************************************/
|
|
// Set column name
|
|
crp = qrp->Colresp; // Column_Name
|
|
crp->Kdata->SetValue("n", 0);
|
|
|
|
// Set type, type name, precision
|
|
crp = crp->Next; // Data_Type
|
|
crp->Kdata->SetValue(TYPE_INT, 0);
|
|
|
|
crp = crp->Next; // Type_Name
|
|
crp->Kdata->SetValue(GetTypeName(TYPE_INT), 0);
|
|
|
|
crp = crp->Next; // Precision
|
|
crp->Kdata->SetValue(11, 0);
|
|
|
|
crp = crp->Next; // Key
|
|
crp->Kdata->SetValue("KEY", 0);
|
|
|
|
crp = crp->Next; // Extra
|
|
crp->Kdata->SetValue("SPECIAL=ROWID", 0);
|
|
|
|
qrp->Nblin = 1;
|
|
|
|
/**********************************************************************/
|
|
/* Return the result pointer for use by discovery routines. */
|
|
/**********************************************************************/
|
|
return qrp;
|
|
} // end of VirColumns
|
|
|
|
/* --------------------------- Class VIRDEF --------------------------- */
|
|
|
|
/***********************************************************************/
|
|
/* GetTable: makes a new Table Description Block. */
|
|
/***********************************************************************/
|
|
PTDB VIRDEF::GetTable(PGLOBAL g, MODE)
|
|
{
|
|
// Column blocks will be allocated only when needed.
|
|
if (Catfunc == FNC_COL)
|
|
return new(g) TDBVICL(this);
|
|
else
|
|
return new(g) TDBVIR(this);
|
|
|
|
} // end of GetTable
|
|
|
|
/* ------------------------ TDBVIR functions ------------------------- */
|
|
|
|
/***********************************************************************/
|
|
/* Implementation of the TDBVIR class. */
|
|
/***********************************************************************/
|
|
TDBVIR::TDBVIR(PVIRDEF tdp) : TDBASE(tdp)
|
|
{
|
|
Size = (tdp->GetElemt()) ? tdp->GetElemt() : 1;
|
|
N = -1;
|
|
} // end of TDBVIR constructor
|
|
|
|
/***********************************************************************/
|
|
/* Analyze the filter and reset the size limit accordingly. */
|
|
/* This is possible when a filter contains predicates implying the */
|
|
/* special column ROWID. Here we just test for when no more good */
|
|
/* records can be met in the remaining of the table. */
|
|
/***********************************************************************/
|
|
int TDBVIR::TestFilter(PFIL filp, bool nop)
|
|
{
|
|
int i, op = filp->GetOpc(), n = 0, type[2] = {0,0};
|
|
int l1 = 0, l2, limit = Size;
|
|
PXOB arg[2] = {NULL,NULL};
|
|
|
|
if (op == OP_GT || op == OP_GE || op == OP_LT || op == OP_LE) {
|
|
for (i = 0; i < 2; i++) {
|
|
arg[i] = filp->Arg(i);
|
|
|
|
switch (filp->GetArgType(i)) {
|
|
case TYPE_CONST:
|
|
if ((l1 = arg[i]->GetIntValue()) >= 0)
|
|
type[i] = 1;
|
|
|
|
break;
|
|
case TYPE_COLBLK:
|
|
if (((PCOL)arg[i])->GetTo_Tdb() == this &&
|
|
((PCOL)arg[i])->GetAmType() == TYPE_AM_ROWID)
|
|
type[i] = 2;
|
|
|
|
break;
|
|
default:
|
|
break;
|
|
} // endswitch ArgType
|
|
|
|
if (!type[i])
|
|
break;
|
|
|
|
n += type[i];
|
|
} // endfor i
|
|
|
|
if (n == 3) {
|
|
// If true it will be ok to delete the filter
|
|
BOOL ok = (filp == To_Filter);
|
|
|
|
if (type[0] == 1)
|
|
// Make it always a Column-op-Value
|
|
switch (op) {
|
|
case OP_GT: op = OP_LT; break;
|
|
case OP_GE: op = OP_LE; break;
|
|
case OP_LT: op = OP_GT; break;
|
|
case OP_LE: op = OP_GE; break;
|
|
} // endswitch op
|
|
|
|
if (!nop) switch (op) {
|
|
case OP_LT: l1--;
|
|
case OP_LE: limit = l1; break;
|
|
default: ok = false;
|
|
} // endswitch op
|
|
|
|
else switch (op) {
|
|
case OP_GE: l1--;
|
|
case OP_GT: limit = l1; break;
|
|
default: ok = false;
|
|
} // endswitch op
|
|
|
|
limit = MY_MIN(MY_MAX(0, limit), Size);
|
|
|
|
// Just one where clause such as Rowid < limit;
|
|
if (ok)
|
|
To_Filter = NULL;
|
|
|
|
} else
|
|
limit = Size;
|
|
|
|
} else if ((op == OP_AND && !nop) || (op == OP_OR && nop)) {
|
|
l1 = TestFilter((PFIL)filp->Arg(0), nop);
|
|
l2 = TestFilter((PFIL)filp->Arg(1), nop);
|
|
limit = MY_MIN(l1, l2);
|
|
} else if (op == OP_NOT)
|
|
limit = TestFilter((PFIL)filp->Arg(0), !nop);
|
|
|
|
return limit;
|
|
} // end of TestFilter
|
|
|
|
/***********************************************************************/
|
|
/* Allocate source column description block. */
|
|
/***********************************************************************/
|
|
PCOL TDBVIR::MakeCol(PGLOBAL g, PCOLDEF cdp, PCOL cprec, int n)
|
|
{
|
|
PCOL colp = NULL;
|
|
|
|
if (cdp->IsVirtual()) {
|
|
colp = new(g) VIRCOL(cdp, this, cprec, n);
|
|
} else strcpy(g->Message,
|
|
"Virtual tables accept only special or virtual columns");
|
|
|
|
return colp;
|
|
} // end of MakeCol
|
|
|
|
/***********************************************************************/
|
|
/* VIR Access Method opening routine. */
|
|
/***********************************************************************/
|
|
bool TDBVIR::OpenDB(PGLOBAL g)
|
|
{
|
|
if (Use == USE_OPEN) {
|
|
// Table already open
|
|
N = -1;
|
|
return false;
|
|
} // endif use
|
|
|
|
if (Mode != MODE_READ) {
|
|
strcpy(g->Message, "Virtual tables are read only");
|
|
return true;
|
|
} // endif Mode
|
|
|
|
/*********************************************************************/
|
|
/* Analyze the filter and refine Size accordingly. */
|
|
/*********************************************************************/
|
|
if (To_Filter)
|
|
Size = TestFilter(To_Filter, false);
|
|
|
|
return false;
|
|
} // end of OpenDB
|
|
|
|
/***********************************************************************/
|
|
/* Data Base read routine for the VIR access method. */
|
|
/***********************************************************************/
|
|
int TDBVIR::ReadDB(PGLOBAL)
|
|
{
|
|
return (++N >= Size) ? RC_EF : RC_OK;
|
|
} // end of ReadDB
|
|
|
|
/***********************************************************************/
|
|
/* WriteDB: Data Base write routine for the VIR access methods. */
|
|
/***********************************************************************/
|
|
int TDBVIR::WriteDB(PGLOBAL g)
|
|
{
|
|
sprintf(g->Message, MSG(VIR_READ_ONLY), To_Def->GetType());
|
|
return RC_FX;
|
|
} // end of WriteDB
|
|
|
|
/***********************************************************************/
|
|
/* Data Base delete line routine for the VIR access methods. */
|
|
/***********************************************************************/
|
|
int TDBVIR::DeleteDB(PGLOBAL g, int)
|
|
{
|
|
sprintf(g->Message, MSG(VIR_NO_DELETE), To_Def->GetType());
|
|
return RC_FX;
|
|
} // end of DeleteDB
|
|
|
|
/* ---------------------------- VIRCOL ------------------------------- */
|
|
|
|
/***********************************************************************/
|
|
/* VIRCOL public constructor. */
|
|
/***********************************************************************/
|
|
VIRCOL::VIRCOL(PCOLDEF cdp, PTDB tdbp, PCOL cprec, int i, PSZ)
|
|
: COLBLK(cdp, tdbp, i)
|
|
{
|
|
if (cprec) {
|
|
Next = cprec->GetNext();
|
|
cprec->SetNext(this);
|
|
} else {
|
|
Next = tdbp->GetColumns();
|
|
tdbp->SetColumns(this);
|
|
} // endif cprec
|
|
|
|
} // end of VIRCOL constructor
|
|
|
|
/***********************************************************************/
|
|
/* ReadColumn: */
|
|
/***********************************************************************/
|
|
void VIRCOL::ReadColumn(PGLOBAL g)
|
|
{
|
|
// This should never be called
|
|
sprintf(g->Message, "ReadColumn: Column %s is not virtual", Name);
|
|
longjmp(g->jumper[g->jump_level], TYPE_COLBLK);
|
|
} // end of ReadColumn
|
|
|
|
/* ---------------------------TDBVICL class -------------------------- */
|
|
|
|
/***********************************************************************/
|
|
/* GetResult: Get the list the VIRTUAL table columns. */
|
|
/***********************************************************************/
|
|
PQRYRES TDBVICL::GetResult(PGLOBAL g)
|
|
{
|
|
return VirColumns(g, false);
|
|
} // end of GetResult
|
|
|
|
/* ------------------------- End of Virtual -------------------------- */
|