206 lines
6.7 KiB
C
206 lines
6.7 KiB
C
/*
|
|
* Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
|
|
*
|
|
* This program is free software: you can use, redistribute, and/or modify
|
|
* it under the terms of the GNU Affero General Public License, version 3
|
|
* or later ("AGPL"), as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "functionMgt.h"
|
|
|
|
#include "functionMgtInt.h"
|
|
#include "taos.h"
|
|
#include "taoserror.h"
|
|
#include "thash.h"
|
|
#include "builtins.h"
|
|
#include "catalog.h"
|
|
|
|
typedef struct SFuncMgtService {
|
|
SHashObj* pFuncNameHashTable;
|
|
SArray* pUdfTable; // SUdfInfo
|
|
} SFuncMgtService;
|
|
|
|
typedef struct SUdfInfo {
|
|
SDataType outputDt;
|
|
int8_t funcType;
|
|
} SUdfInfo;
|
|
|
|
static SFuncMgtService gFunMgtService;
|
|
static TdThreadOnce functionHashTableInit = PTHREAD_ONCE_INIT;
|
|
static int32_t initFunctionCode = 0;
|
|
|
|
static void doInitFunctionTable() {
|
|
gFunMgtService.pFuncNameHashTable = taosHashInit(funcMgtBuiltinsNum, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_NO_LOCK);
|
|
if (NULL == gFunMgtService.pFuncNameHashTable) {
|
|
initFunctionCode = TSDB_CODE_FAILED;
|
|
return;
|
|
}
|
|
|
|
for (int32_t i = 0; i < funcMgtBuiltinsNum; ++i) {
|
|
if (TSDB_CODE_SUCCESS != taosHashPut(gFunMgtService.pFuncNameHashTable, funcMgtBuiltins[i].name, strlen(funcMgtBuiltins[i].name), &i, sizeof(int32_t))) {
|
|
initFunctionCode = TSDB_CODE_FAILED;
|
|
return;
|
|
}
|
|
}
|
|
|
|
gFunMgtService.pUdfTable = NULL;
|
|
}
|
|
|
|
static int8_t getUdfType(int32_t funcId) {
|
|
SUdfInfo* pUdf = taosArrayGet(gFunMgtService.pUdfTable, funcId - FUNC_UDF_ID_START_OFFSET_VAL - 1);
|
|
return pUdf->funcType;
|
|
}
|
|
|
|
static bool isSpecificClassifyFunc(int32_t funcId, uint64_t classification) {
|
|
if (fmIsUserDefinedFunc(funcId)) {
|
|
return getUdfType(funcId);
|
|
}
|
|
if (funcId < 0 || funcId >= funcMgtBuiltinsNum) {
|
|
return false;
|
|
}
|
|
return FUNC_MGT_TEST_MASK(funcMgtBuiltins[funcId].classification, classification);
|
|
}
|
|
|
|
static int32_t getUdfId(SFmGetFuncInfoParam* pParam, const char* pFuncName) {
|
|
SFuncInfo* pInfo = NULL;
|
|
int32_t code = catalogGetUdfInfo(pParam->pCtg, pParam->pRpc, pParam->pMgmtEps, pFuncName, &pInfo);
|
|
if (TSDB_CODE_SUCCESS != code || NULL == pInfo) {
|
|
return -1;
|
|
}
|
|
if (NULL == gFunMgtService.pUdfTable) {
|
|
gFunMgtService.pUdfTable = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SUdfInfo));
|
|
}
|
|
SUdfInfo info = { .outputDt.type = pInfo->outputType, .outputDt.bytes = pInfo->outputLen, .funcType = pInfo->funcType };
|
|
taosArrayPush(gFunMgtService.pUdfTable, &info);
|
|
tFreeSFuncInfo(pInfo);
|
|
taosMemoryFree(pInfo);
|
|
return taosArrayGetSize(gFunMgtService.pUdfTable) + FUNC_UDF_ID_START_OFFSET_VAL;
|
|
}
|
|
|
|
static int32_t getFuncId(SFmGetFuncInfoParam* pParam, const char* pFuncName) {
|
|
void* pVal = taosHashGet(gFunMgtService.pFuncNameHashTable, pFuncName, strlen(pFuncName));
|
|
if (NULL == pVal) {
|
|
return getUdfId(pParam, pFuncName);
|
|
}
|
|
return *(int32_t*)pVal;
|
|
}
|
|
|
|
static int32_t getUdfResultType(SFunctionNode* pFunc) {
|
|
SUdfInfo* pUdf = taosArrayGet(gFunMgtService.pUdfTable, pFunc->funcId - FUNC_UDF_ID_START_OFFSET_VAL - 1);
|
|
pFunc->node.resType = pUdf->outputDt;
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t fmFuncMgtInit() {
|
|
taosThreadOnce(&functionHashTableInit, doInitFunctionTable);
|
|
return initFunctionCode;
|
|
}
|
|
|
|
int32_t fmGetFuncInfo(SFmGetFuncInfoParam* pParam, const char* pFuncName, int32_t* pFuncId, int32_t* pFuncType) {
|
|
*pFuncId = getFuncId(pParam, pFuncName);
|
|
if (*pFuncId < 0) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
if (fmIsUserDefinedFunc(*pFuncId)) {
|
|
*pFuncType = FUNCTION_TYPE_UDF;
|
|
} else {
|
|
*pFuncType = funcMgtBuiltins[*pFuncId].type;
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t fmGetFuncResultType(SFunctionNode* pFunc, char* pErrBuf, int32_t len) {
|
|
if (fmIsUserDefinedFunc(pFunc->funcId)) {
|
|
return getUdfResultType(pFunc);
|
|
}
|
|
|
|
if (pFunc->funcId < 0 || pFunc->funcId >= funcMgtBuiltinsNum) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
return funcMgtBuiltins[pFunc->funcId].translateFunc(pFunc, pErrBuf, len);
|
|
}
|
|
|
|
EFuncDataRequired fmFuncDataRequired(SFunctionNode* pFunc, STimeWindow* pTimeWindow) {
|
|
if (fmIsUserDefinedFunc(pFunc->funcId) || pFunc->funcId < 0 || pFunc->funcId >= funcMgtBuiltinsNum) {
|
|
return FUNC_DATA_REQUIRED_DATA_LOAD;
|
|
}
|
|
if (NULL == funcMgtBuiltins[pFunc->funcId].dataRequiredFunc) {
|
|
return FUNC_DATA_REQUIRED_DATA_LOAD;
|
|
}
|
|
return funcMgtBuiltins[pFunc->funcId].dataRequiredFunc(pFunc, pTimeWindow);
|
|
}
|
|
|
|
int32_t fmGetFuncExecFuncs(int32_t funcId, SFuncExecFuncs* pFpSet) {
|
|
if (fmIsUserDefinedFunc(funcId) || funcId < 0 || funcId >= funcMgtBuiltinsNum) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
pFpSet->getEnv = funcMgtBuiltins[funcId].getEnvFunc;
|
|
pFpSet->init = funcMgtBuiltins[funcId].initFunc;
|
|
pFpSet->process = funcMgtBuiltins[funcId].processFunc;
|
|
pFpSet->finalize = funcMgtBuiltins[funcId].finalizeFunc;
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t fmGetScalarFuncExecFuncs(int32_t funcId, SScalarFuncExecFuncs* pFpSet) {
|
|
if (fmIsUserDefinedFunc(funcId) || funcId < 0 || funcId >= funcMgtBuiltinsNum) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
pFpSet->process = funcMgtBuiltins[funcId].sprocessFunc;
|
|
pFpSet->getEnv = funcMgtBuiltins[funcId].getEnvFunc;
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
bool fmIsAggFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_AGG_FUNC);
|
|
}
|
|
|
|
bool fmIsScalarFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_SCALAR_FUNC);
|
|
}
|
|
|
|
bool fmIsPseudoColumnFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_PSEUDO_COLUMN_FUNC);
|
|
}
|
|
|
|
bool fmIsWindowPseudoColumnFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_WINDOW_PC_FUNC);
|
|
}
|
|
|
|
bool fmIsWindowClauseFunc(int32_t funcId) {
|
|
return fmIsAggFunc(funcId) || fmIsWindowPseudoColumnFunc(funcId);
|
|
}
|
|
|
|
bool fmIsNonstandardSQLFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_NONSTANDARD_SQL_FUNC);
|
|
}
|
|
|
|
bool fmIsSpecialDataRequiredFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_SPECIAL_DATA_REQUIRED);
|
|
}
|
|
|
|
bool fmIsDynamicScanOptimizedFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_DYNAMIC_SCAN_OPTIMIZED);
|
|
}
|
|
|
|
bool fmIsMultiResFunc(int32_t funcId) {
|
|
return isSpecificClassifyFunc(funcId, FUNC_MGT_MULTI_RES_FUNC);
|
|
}
|
|
|
|
bool fmIsUserDefinedFunc(int32_t funcId) {
|
|
return funcId > FUNC_UDF_ID_START_OFFSET_VAL;
|
|
}
|
|
|
|
void fmFuncMgtDestroy() {
|
|
void* m = gFunMgtService.pFuncNameHashTable;
|
|
if (m != NULL && atomic_val_compare_exchange_ptr((void**)&gFunMgtService.pFuncNameHashTable, m, 0) == m) {
|
|
taosHashCleanup(m);
|
|
}
|
|
}
|