/* * Copyright (c) 2019 TAOS Data, Inc. * * 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 . */ #include "sma.h" // ================================================================================================= static int32_t tdFetchQTaskInfoFiles(SSma *pSma, int64_t version, SArray **output); static int32_t tdQTaskInfCmprFn1(const void *p1, const void *p2); static int32_t tdQTaskInfCmprFn2(const void *p1, const void *p2); /** * @brief Open RSma FS from qTaskInfo files * * @param pSma * @param version * @return int32_t */ int32_t tdRSmaFSOpen(SSma *pSma, int64_t version) { SVnode *pVnode = pSma->pVnode; int64_t commitID = pVnode->state.commitID; SSmaEnv *pEnv = SMA_RSMA_ENV(pSma); SRSmaStat *pStat = NULL; SArray *output = NULL; terrno = TSDB_CODE_SUCCESS; if (!pEnv) { return TSDB_CODE_SUCCESS; } if (tdFetchQTaskInfoFiles(pSma, version, &output) < 0) { goto _end; } pStat = (SRSmaStat *)SMA_ENV_STAT(pEnv); for (int32_t i = 0; i < taosArrayGetSize(output); ++i) { int32_t vid = 0; int64_t version = -1; sscanf((const char *)taosArrayGetP(output, i), "v%dqinf.v%" PRIi64, &vid, &version); SQTaskFile qTaskFile = {.version = version, .nRef = 1}; if ((terrno = tdRSmaFSUpsertQTaskFile(RSMA_FS(pStat), &qTaskFile)) < 0) { goto _end; } smaInfo("vgId:%d, open fs, version:%" PRIi64 ", ref:%" PRIi64, TD_VID(pVnode), qTaskFile.version, qTaskFile.nRef); } _end: for (int32_t i = 0; i < taosArrayGetSize(output); ++i) { void *ptr = taosArrayGetP(output, i); taosMemoryFreeClear(ptr); } taosArrayDestroy(output); if (terrno != TSDB_CODE_SUCCESS) { smaError("vgId:%d, open rsma fs failed since %s", TD_VID(pVnode), terrstr()); return TSDB_CODE_FAILED; } return TSDB_CODE_SUCCESS; } void tdRSmaFSClose(SRSmaFS *fs) { taosArrayDestroy(fs->aQTaskInf); } static int32_t tdQTaskInfCmprFn1(const void *p1, const void *p2) { if (*(int64_t *)p1 < ((SQTaskFile *)p2)->version) { return -1; } else if (*(int64_t *)p1 > ((SQTaskFile *)p2)->version) { return 1; } return 0; } int32_t tdRSmaFSRef(SSma *pSma, SRSmaStat *pStat, int64_t version) { SArray *aQTaskInf = RSMA_FS(pStat)->aQTaskInf; SQTaskFile *pTaskF = NULL; int32_t oldVal = 0; taosRLockLatch(RSMA_FS_LOCK(pStat)); if ((pTaskF = taosArraySearch(aQTaskInf, &version, tdQTaskInfCmprFn1, TD_EQ))) { oldVal = atomic_fetch_add_32(&pTaskF->nRef, 1); ASSERT(oldVal > 0); } taosRUnLockLatch(RSMA_FS_LOCK(pStat)); return oldVal; } int64_t tdRSmaFSMaxVer(SSma *pSma, SRSmaStat *pStat) { SArray *aQTaskInf = RSMA_FS(pStat)->aQTaskInf; int64_t version = -1; taosRLockLatch(RSMA_FS_LOCK(pStat)); if (taosArrayGetSize(aQTaskInf) > 0) { version = ((SQTaskFile *)taosArrayGetLast(aQTaskInf))->version; } taosRUnLockLatch(RSMA_FS_LOCK(pStat)); return version; } void tdRSmaFSUnRef(SSma *pSma, SRSmaStat *pStat, int64_t version) { SVnode *pVnode = pSma->pVnode; SArray *aQTaskInf = RSMA_FS(pStat)->aQTaskInf; char qTaskFullName[TSDB_FILENAME_LEN]; SQTaskFile *pTaskF = NULL; int32_t idx = -1; taosWLockLatch(RSMA_FS_LOCK(pStat)); if ((idx = taosArraySearchIdx(aQTaskInf, &version, tdQTaskInfCmprFn1, TD_EQ)) >= 0) { ASSERT(idx < taosArrayGetSize(aQTaskInf)); pTaskF = taosArrayGet(aQTaskInf, idx); if (atomic_sub_fetch_32(&pTaskF->nRef, 1) <= 0) { tdRSmaQTaskInfoGetFullName(TD_VID(pVnode), pTaskF->version, tfsGetPrimaryPath(pVnode->pTfs), qTaskFullName); if (taosRemoveFile(qTaskFullName) < 0) { smaWarn("vgId:%d, failed to remove %s since %s", TD_VID(pVnode), qTaskFullName, tstrerror(TAOS_SYSTEM_ERROR(errno))); } else { smaDebug("vgId:%d, success to remove %s", TD_VID(pVnode), qTaskFullName); } taosArrayRemove(aQTaskInf, idx); } } taosWUnLockLatch(RSMA_FS_LOCK(pStat)); } /** * @brief Fetch qtaskfiles LE than version * * @param pSma * @param version * @param output * @return int32_t */ static int32_t tdFetchQTaskInfoFiles(SSma *pSma, int64_t version, SArray **output) { SVnode *pVnode = pSma->pVnode; TdDirPtr pDir = NULL; TdDirEntryPtr pDirEntry = NULL; char dir[TSDB_FILENAME_LEN]; const char *pattern = "v[0-9]+qinf\\.v([0-9]+)?$"; regex_t regex; int code = 0; terrno = TSDB_CODE_SUCCESS; tdGetVndDirName(TD_VID(pVnode), tfsGetPrimaryPath(pVnode->pTfs), VNODE_RSMA_DIR, true, dir); if (!taosCheckExistFile(dir)) { smaDebug("vgId:%d, fetch qtask files, no need as dir %s not exist", TD_VID(pVnode), dir); return TSDB_CODE_SUCCESS; } // Resource allocation and init if ((code = regcomp(®ex, pattern, REG_EXTENDED)) != 0) { terrno = TSDB_CODE_RSMA_REGEX_MATCH; char errbuf[128]; regerror(code, ®ex, errbuf, sizeof(errbuf)); smaWarn("vgId:%d, fetch qtask files, regcomp for %s failed since %s", TD_VID(pVnode), dir, errbuf); return TSDB_CODE_FAILED; } if (!(pDir = taosOpenDir(dir))) { regfree(®ex); terrno = TAOS_SYSTEM_ERROR(errno); smaError("vgId:%d, fetch qtask files, open dir %s failed since %s", TD_VID(pVnode), dir, terrstr()); return TSDB_CODE_FAILED; } int32_t dirLen = strlen(dir); char *dirEnd = POINTER_SHIFT(dir, dirLen); regmatch_t regMatch[2]; while ((pDirEntry = taosReadDir(pDir))) { char *entryName = taosGetDirEntryName(pDirEntry); if (!entryName) { continue; } code = regexec(®ex, entryName, 2, regMatch, 0); if (code == 0) { // match smaInfo("vgId:%d, fetch qtask files, max ver:%" PRIi64 ", %s found", TD_VID(pVnode), version, entryName); int64_t ver = -1; sscanf((const char *)POINTER_SHIFT(entryName, regMatch[1].rm_so), "%" PRIi64, &ver); if ((ver <= version) && (ver > -1)) { if (!(*output)) { if (!(*output = taosArrayInit(1, POINTER_BYTES))) { terrno = TSDB_CODE_OUT_OF_MEMORY; goto _end; } } char *entryDup = strdup(entryName); if (!entryDup) { terrno = TSDB_CODE_OUT_OF_MEMORY; goto _end; } if (!taosArrayPush(*output, &entryDup)) { terrno = TSDB_CODE_OUT_OF_MEMORY; goto _end; } } else { } } else if (code == REG_NOMATCH) { // not match smaTrace("vgId:%d, fetch qtask files, not match %s", TD_VID(pVnode), entryName); continue; } else { // has other error char errbuf[128]; regerror(code, ®ex, errbuf, sizeof(errbuf)); smaWarn("vgId:%d, fetch qtask files, regexec failed since %s", TD_VID(pVnode), errbuf); terrno = TSDB_CODE_RSMA_REGEX_MATCH; goto _end; } } _end: taosCloseDir(&pDir); regfree(®ex); return terrno == 0 ? TSDB_CODE_SUCCESS : TSDB_CODE_FAILED; } static int32_t tdQTaskFileCmprFn2(const void *p1, const void *p2) { if (((SQTaskFile *)p1)->version < ((SQTaskFile *)p2)->version) { return -1; } else if (((SQTaskFile *)p1)->version > ((SQTaskFile *)p2)->version) { return 1; } return 0; } int32_t tdRSmaFSUpsertQTaskFile(SRSmaFS *pFS, SQTaskFile *qTaskFile) { int32_t code = 0; int32_t idx = taosArraySearchIdx(pFS->aQTaskInf, qTaskFile, tdQTaskFileCmprFn2, TD_GE); if (idx < 0) { idx = taosArrayGetSize(pFS->aQTaskInf); } else { SQTaskFile *pTaskF = (SQTaskFile *)taosArrayGet(pFS->aQTaskInf, idx); int32_t c = tdQTaskFileCmprFn2(pTaskF, qTaskFile); if (c == 0) { pTaskF->nRef = qTaskFile->nRef; pTaskF->version = qTaskFile->version; pTaskF->size = qTaskFile->size; goto _exit; } } if (taosArrayInsert(pFS->aQTaskInf, idx, qTaskFile) == NULL) { code = TSDB_CODE_OUT_OF_MEMORY; goto _exit; } _exit: return code; }