vnodeArenaMAImpl.c 2.9 KB
Newer Older
H
more  
Hongze Cheng 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
/*
 * 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 "vnodeDef.h"

static SVArenaNode *vArenaNodeNew(uint64_t capacity);
static void         vArenaNodeFree(SVArenaNode *pNode);

SVMemAllocator *vmaCreate(uint64_t capacity, uint64_t ssize, uint64_t lsize) {
  SVMemAllocator *pVMA = (SVMemAllocator *)malloc(sizeof(*pVMA));
  if (pVMA == NULL) {
    return NULL;
  }

  pVMA->capacity = capacity;
  pVMA->ssize = ssize;
  pVMA->lsize = lsize;
  tlistInit(&(pVMA->nlist));

  SVArenaNode *pNode = vArenaNodeNew(capacity);
  if (pNode == NULL) {
H
more  
Hongze Cheng 已提交
34
    free(pVMA);
H
more  
Hongze Cheng 已提交
35 36 37 38 39 40 41 42 43
    return NULL;
  }

  tlistAppend(&(pVMA->nlist), pNode);

  return pVMA;
}

void vmaDestroy(SVMemAllocator *pVMA) {
H
more  
Hongze Cheng 已提交
44 45
  if (pVMA) {
    while (true) {
H
more  
Hongze Cheng 已提交
46
      SVArenaNode *pNode = tlistTail(&(pVMA->nlist));
H
more  
Hongze Cheng 已提交
47 48

      if (pNode) {
H
more  
Hongze Cheng 已提交
49
        tlistPop(&(pVMA->nlist), pNode);
H
more  
Hongze Cheng 已提交
50 51 52 53 54 55 56 57
        vArenaNodeFree(pNode);
      } else {
        break;
      }
    }

    free(pVMA);
  }
H
more  
Hongze Cheng 已提交
58 59 60
}

void vmaReset(SVMemAllocator *pVMA) {
H
more  
Hongze Cheng 已提交
61
  while (tlistNEles(&(pVMA->nlist)) > 1) {
H
more  
Hongze Cheng 已提交
62 63
    SVArenaNode *pNode = tlistTail(&(pVMA->nlist));
    tlistPop(&(pVMA->nlist), pNode);
H
more  
Hongze Cheng 已提交
64 65 66 67 68
    vArenaNodeFree(pNode);
  }

  SVArenaNode *pNode = tlistHead(&(pVMA->nlist));
  pNode->ptr = pNode->data;
H
more  
Hongze Cheng 已提交
69 70 71
}

void *vmaMalloc(SVMemAllocator *pVMA, uint64_t size) {
H
more  
Hongze Cheng 已提交
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
  SVArenaNode *pNode = tlistTail(&(pVMA->nlist));
  void *       ptr;

  if (pNode->size < POINTER_DISTANCE(pNode->ptr, pNode->data) + size) {
    uint64_t capacity = MAX(pVMA->ssize, size);
    pNode = vArenaNodeNew(capacity);
    if (pNode == NULL) {
      // TODO: handle error
      return NULL;
    }

    tlistAppend(&(pVMA->nlist), pNode);
  }

  ptr = pNode->ptr;
  pNode->ptr = POINTER_SHIFT(ptr, size);

  return ptr;
H
more  
Hongze Cheng 已提交
90 91 92 93 94 95
}

void vmaFree(SVMemAllocator *pVMA, void *ptr) {
  // TODO
}

H
more  
Hongze Cheng 已提交
96 97 98 99 100 101
bool vmaIsFull(SVMemAllocator *pVMA) {
  SVArenaNode *pNode = tlistTail(&(pVMA->nlist));

  return (tlistNEles(&(pVMA->nlist)) > 1) || (pNode->size < POINTER_DISTANCE(pNode->ptr, pNode->data) + pVMA->lsize);
}

H
more  
Hongze Cheng 已提交
102 103 104
/* ------------------------ STATIC METHODS ------------------------ */
static SVArenaNode *vArenaNodeNew(uint64_t capacity) {
  SVArenaNode *pNode = NULL;
H
more  
Hongze Cheng 已提交
105 106 107 108 109 110 111 112 113

  pNode = (SVArenaNode *)malloc(sizeof(*pNode) + capacity);
  if (pNode == NULL) {
    return NULL;
  }

  pNode->size = capacity;
  pNode->ptr = pNode->data;

H
more  
Hongze Cheng 已提交
114 115 116 117
  return pNode;
}

static void vArenaNodeFree(SVArenaNode *pNode) {
H
more  
Hongze Cheng 已提交
118 119 120
  if (pNode) {
    free(pNode);
  }
H
more  
Hongze Cheng 已提交
121
}