LCOV - code coverage report
Current view: top level - monetdb5/modules/kernel - bat5.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 193 637 30.3 %
Date: 2025-03-24 23:16:36 Functions: 23 45 51.1 %

          Line data    Source code
       1             : /*
       2             :  * SPDX-License-Identifier: MPL-2.0
       3             :  *
       4             :  * This Source Code Form is subject to the terms of the Mozilla Public
       5             :  * License, v. 2.0.  If a copy of the MPL was not distributed with this
       6             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       7             :  *
       8             :  * Copyright 2024, 2025 MonetDB Foundation;
       9             :  * Copyright August 2008 - 2023 MonetDB B.V.;
      10             :  * Copyright 1997 - July 2008 CWI.
      11             :  */
      12             : 
      13             : /*
      14             :  * Peter Boncz, M.L. Kersten
      15             :  * Binary Association Tables
      16             :  * This module contains the commands and patterns to manage Binary
      17             :  * Association Tables (BATs). The relational operations you can execute
      18             :  * on BATs have the form of a neat algebra, described in algebra.c
      19             :  *
      20             :  * But a database system needs more that just this algebra, since often it
      21             :  * is crucial to do table-updates (this would not be permitted in a strict
      22             :  * algebra).
      23             :  *
      24             :  * All commands needed for BAT updates, property management, basic I/O,
      25             :  * persistence, and storage options can be found in this module.
      26             :  *
      27             :  * All parameters to the modules are passed by reference.
      28             :  * In particular, this means that string values are passed to the module
      29             :  * layer as (str *)
      30             :  * and we have to de-reference them before entering the gdk library.
      31             :  * (Actual a design error in gdk to differentiate passing int/str)
      32             :  * This calls for knowledge on the underlying BAT types`s
      33             :  */
      34             : 
      35             : #include "monetdb_config.h"
      36             : #include "bat5.h"
      37             : #include "mal_exception.h"
      38             : 
      39             : /*
      40             :  * The remainder contains the wrapper code over the mserver version 4
      41             :  * InformationFunctions
      42             :  * In most cases we pass a BAT identifier, which should be unified
      43             :  * with a BAT descriptor. Upon failure we can simply abort the function.
      44             :  *
      45             :  * The logical head type :oid is mapped to a TYPE_void
      46             :  * with sequenceBase. It represents the old fashioned :vid
      47             :  */
      48             : 
      49             : 
      50             : #define derefStr(b, v)                                                  \
      51             :         do {                                                                            \
      52             :                 int _tpe= ATOMstorage((b)->ttype);           \
      53             :                 if (_tpe >= TYPE_str) {                                      \
      54             :                         if ((v) == 0 || *(str*) (v) == 0)       \
      55             :                                 (v) = (str) str_nil;                    \
      56             :                         else                                                            \
      57             :                                 (v) = *(str *) (v);                             \
      58             :                 }                                                                               \
      59             :         } while (0)
      60             : 
      61             : str
      62      193845 : BKCnewBAT(bat *res, const int *tt, const BUN *cap, role_t role)
      63             : {
      64      193845 :         BAT *bn;
      65             : 
      66      193845 :         bn = COLnew(0, *tt, *cap, role);
      67      199693 :         if (bn == NULL)
      68           0 :                 throw(MAL, "bat.new", GDK_EXCEPTION);
      69      199693 :         *res = bn->batCacheid;
      70      199693 :         BBPretain(bn->batCacheid);
      71      199067 :         BBPunfix(bn->batCacheid);
      72      199067 :         return MAL_SUCCEED;
      73             : }
      74             : 
      75             : static str
      76           0 : BKCdensebat(bat *ret, const lng *size)
      77             : {
      78           0 :         BAT *bn;
      79           0 :         lng sz = *size;
      80             : 
      81           0 :         if (sz < 0)
      82             :                 sz = 0;
      83             :         if (sz > (lng) BUN_MAX)
      84             :                 sz = (lng) BUN_MAX;
      85           0 :         bn = BATdense(0, 0, (BUN) sz);
      86           0 :         if (bn == NULL)
      87           0 :                 throw(MAL, "bat.densebat", GDK_EXCEPTION);
      88           0 :         *ret = bn->batCacheid;
      89           0 :         BBPkeepref(bn);
      90           0 :         return MAL_SUCCEED;
      91             : }
      92             : 
      93             : str
      94      121594 : BKCmirror(bat *ret, const bat *bid)
      95             : {
      96      121594 :         BAT *b, *bn;
      97             : 
      98      121594 :         *ret = 0;
      99      121594 :         if (!(b = BBPquickdesc(*bid)))
     100           0 :                 throw(MAL, "bat.mirror", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     101      121752 :         if (!(bn = BATdense(b->hseqbase, b->hseqbase, BATcount(b))))
     102           0 :                 throw(MAL, "bat.mirror", GDK_EXCEPTION);
     103      121586 :         *ret = bn->batCacheid;
     104      121586 :         BBPkeepref(bn);
     105      121586 :         return MAL_SUCCEED;
     106             : }
     107             : 
     108             : static str
     109           4 : BKCdelete(bat *r, const bat *bid, const oid *h)
     110             : {
     111           4 :         BAT *b;
     112             : 
     113           4 :         if ((b = BATdescriptor(*bid)) == NULL)
     114           0 :                 throw(MAL, "bat.delete", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     115           4 :         if (BUNdelete(b, *h) != GDK_SUCCEED) {
     116           2 :                 BBPunfix(b->batCacheid);
     117           2 :                 throw(MAL, "bat.delete", GDK_EXCEPTION);
     118             :         }
     119           2 :         *r = b->batCacheid;
     120           2 :         BBPretain(b->batCacheid);
     121           2 :         BBPunfix(b->batCacheid);
     122           2 :         return MAL_SUCCEED;
     123             : }
     124             : 
     125             : static str
     126           4 : BKCdelete_multi(bat *r, const bat *bid, const bat *sid)
     127             : {
     128           4 :         BAT *b, *s;
     129           4 :         gdk_return ret;
     130             : 
     131           4 :         if ((b = BATdescriptor(*bid)) == NULL)
     132           0 :                 throw(MAL, "bat.delete", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     133           4 :         if ((s = BATdescriptor(*sid)) == NULL) {
     134           0 :                 BBPunfix(b->batCacheid);
     135           0 :                 throw(MAL, "bat.delete", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     136             :         }
     137           4 :         ret = BATdel(b, s);
     138           4 :         BBPunfix(s->batCacheid);
     139           4 :         if (ret != GDK_SUCCEED) {
     140           0 :                 BBPunfix(b->batCacheid);
     141           0 :                 throw(MAL, "bat.delete", GDK_EXCEPTION);
     142             :         }
     143           4 :         *r = b->batCacheid;
     144           4 :         BBPretain(b->batCacheid);
     145           4 :         BBPunfix(b->batCacheid);
     146           4 :         return MAL_SUCCEED;
     147             : }
     148             : 
     149             : static str
     150           7 : BKCdelete_all(bat *r, const bat *bid)
     151             : {
     152           7 :         BAT *b;
     153             : 
     154           7 :         if ((b = BATdescriptor(*bid)) == NULL)
     155           0 :                 throw(MAL, "bat.delete", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     156           7 :         if (BATclear(b, false) != GDK_SUCCEED) {
     157           0 :                 BBPunfix(b->batCacheid);
     158           0 :                 throw(MAL, "bat.delete", GDK_EXCEPTION);
     159             :         }
     160           7 :         *r = b->batCacheid;
     161           7 :         BBPretain(b->batCacheid);
     162           7 :         BBPunfix(b->batCacheid);
     163           7 :         return MAL_SUCCEED;
     164             : }
     165             : 
     166             : static str
     167      104065 : BKCappend_cand_force_wrap(bat *r, const bat *bid, const bat *uid,
     168             :                                                   const bat *sid, const bit *force)
     169             : {
     170      104065 :         BAT *b, *u, *s = NULL;
     171      104065 :         gdk_return ret;
     172             : 
     173      104065 :         if ((b = BATdescriptor(*bid)) == NULL)
     174           0 :                 throw(MAL, "bat.append", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     175      104529 :         if (isVIEW(b)) {
     176        9503 :                 BAT *bn = COLcopy(b, b->ttype, true, TRANSIENT);
     177        9423 :                 MT_lock_set(&b->theaplock);
     178        9473 :                 restrict_t mode = b->batRestricted;
     179        9473 :                 MT_lock_unset(&b->theaplock);
     180        9504 :                 BBPunfix(b->batCacheid);
     181        9494 :                 if (bn == NULL || (b = BATsetaccess(bn, mode)) == NULL)
     182           0 :                         throw(MAL, "bat.append", GDK_EXCEPTION);
     183             :         }
     184      104530 :         if ((u = BATdescriptor(*uid)) == NULL) {
     185           0 :                 BBPunfix(b->batCacheid);
     186           0 :                 throw(MAL, "bat.append", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     187             :         }
     188      104455 :         if (mask_cand(u)) {
     189           0 :                 BAT *ou = u;
     190           0 :                 u = BATunmask(u);
     191           0 :                 BBPunfix(ou->batCacheid);
     192           0 :                 if (!u) {
     193           0 :                         BBPunfix(b->batCacheid);
     194           0 :                         throw(MAL, "bat.append", GDK_EXCEPTION);
     195             :                 }
     196             :         }
     197      104455 :         if (sid && !is_bat_nil(*sid) && (s = BATdescriptor(*sid)) == NULL) {
     198           0 :                 BBPunfix(b->batCacheid);
     199           0 :                 BBPunfix(u->batCacheid);
     200           0 :                 throw(MAL, "bat.append", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     201             :         }
     202      104676 :         ret = BATappend(b, u, s, force ? *force : false);
     203      103966 :         BBPunfix(u->batCacheid);
     204      104396 :         BBPreclaim(s);
     205      104289 :         if (ret != GDK_SUCCEED) {
     206           0 :                 BBPunfix(b->batCacheid);
     207           0 :                 throw(MAL, "bat.append", GDK_EXCEPTION);
     208             :         }
     209      104289 :         *r = b->batCacheid;
     210      104289 :         BBPretain(b->batCacheid);
     211      104353 :         BBPunfix(b->batCacheid);
     212      104353 :         return MAL_SUCCEED;
     213             : }
     214             : 
     215             : static str
     216           0 : BKCappend_cand_wrap(bat *r, const bat *bid, const bat *uid, const bat *sid)
     217             : {
     218           0 :         return BKCappend_cand_force_wrap(r, bid, uid, sid, NULL);
     219             : }
     220             : 
     221             : static str
     222         221 : BKCappend_wrap(bat *r, const bat *bid, const bat *uid)
     223             : {
     224         221 :         return BKCappend_cand_force_wrap(r, bid, uid, NULL, NULL);
     225             : }
     226             : 
     227             : static str
     228      103944 : BKCappend_force_wrap(bat *r, const bat *bid, const bat *uid, const bit *force)
     229             : {
     230      103944 :         return BKCappend_cand_force_wrap(r, bid, uid, NULL, force);
     231             : }
     232             : 
     233             : static str
     234    14304064 : BKCappend_val_force_wrap(bat *r, const bat *bid, const void *u,
     235             :                                                  const bit *force)
     236             : {
     237    14304064 :         BAT *b;
     238             : 
     239    14304064 :         if ((b = BATdescriptor(*bid)) == NULL)
     240           0 :                 throw(MAL, "bat.append", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     241    14304064 :         if (isVIEW(b)) {
     242          12 :                 BAT *bn = COLcopy(b, b->ttype, true, TRANSIENT);
     243          12 :                 MT_lock_set(&b->theaplock);
     244          12 :                 restrict_t mode = b->batRestricted;
     245          12 :                 MT_lock_unset(&b->theaplock);
     246          12 :                 BBPunfix(b->batCacheid);
     247          12 :                 if (bn == NULL || (b = BATsetaccess(bn, mode)) == NULL)
     248           0 :                         throw(MAL, "bat.append", GDK_EXCEPTION);
     249             :         }
     250    14304064 :         derefStr(b, u);
     251    28607773 :         if (BUNappend(b, u, force ? *force : false) != GDK_SUCCEED) {
     252           0 :                 BBPunfix(b->batCacheid);
     253           0 :                 throw(MAL, "bat.append", GDK_EXCEPTION);
     254             :         }
     255    14304064 :         *r = b->batCacheid;
     256    14304064 :         BBPretain(b->batCacheid);
     257    14304064 :         BBPunfix(b->batCacheid);
     258    14304064 :         return MAL_SUCCEED;
     259             : }
     260             : 
     261             : static str
     262    14303709 : BKCappend_val_wrap(bat *r, const bat *bid, const void *u)
     263             : {
     264    14303709 :         return BKCappend_val_force_wrap(r, bid, u, NULL);
     265             : }
     266             : 
     267             : static str
     268         103 : BKCbun_inplace(bat *r, const bat *bid, const oid *id, const void *t)
     269             : {
     270         103 :         BAT *b;
     271             : 
     272         103 :         if ((b = BATdescriptor(*bid)) == NULL)
     273           0 :                 throw(MAL, "bat.inplace", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     274         103 :         derefStr(b, t);
     275         103 :         if (void_inplace(b, *id, t, false) != GDK_SUCCEED) {
     276           0 :                 BBPunfix(b->batCacheid);
     277           0 :                 throw(MAL, "bat.inplace", GDK_EXCEPTION);
     278             :         }
     279         103 :         *r = b->batCacheid;
     280         103 :         BBPretain(b->batCacheid);
     281         103 :         BBPunfix(b->batCacheid);
     282         103 :         return MAL_SUCCEED;
     283             : }
     284             : 
     285             : static str
     286           0 : BKCbun_inplace_force(bat *r, const bat *bid, const oid *id, const void *t,
     287             :                                          const bit *force)
     288             : {
     289           0 :         BAT *b;
     290             : 
     291           0 :         if ((b = BATdescriptor(*bid)) == NULL)
     292           0 :                 throw(MAL, "bat.inplace", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     293           0 :         derefStr(b, t);
     294           0 :         if (void_inplace(b, *id, t, *force) != GDK_SUCCEED) {
     295           0 :                 BBPunfix(b->batCacheid);
     296           0 :                 throw(MAL, "bat.inplace", GDK_EXCEPTION);
     297             :         }
     298           0 :         *r = b->batCacheid;
     299           0 :         BBPretain(b->batCacheid);
     300           0 :         BBPunfix(b->batCacheid);
     301           0 :         return MAL_SUCCEED;
     302             : }
     303             : 
     304             : 
     305             : static str
     306      209469 : BKCbat_inplace_force(bat *r, const bat *bid, const bat *rid, const bat *uid,
     307             :                                          const bit *force)
     308             : {
     309      209469 :         BAT *b, *p, *u;
     310             : 
     311      209469 :         if ((b = BATdescriptor(*bid)) == NULL)
     312           0 :                 throw(MAL, "bat.inplace", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     313      210096 :         if ((p = BATdescriptor(*rid)) == NULL) {
     314           0 :                 BBPunfix(b->batCacheid);
     315           0 :                 throw(MAL, "bat.inplace", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     316             :         }
     317      210050 :         if ((u = BATdescriptor(*uid)) == NULL) {
     318           0 :                 BBPunfix(b->batCacheid);
     319           0 :                 BBPunfix(p->batCacheid);
     320           0 :                 throw(MAL, "bat.inplace", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     321             :         }
     322      210106 :         if (BATreplace(b, p, u, *force) != GDK_SUCCEED) {
     323           0 :                 BBPunfix(b->batCacheid);
     324           0 :                 BBPunfix(p->batCacheid);
     325           0 :                 BBPunfix(u->batCacheid);
     326           0 :                 throw(MAL, "bat.inplace", GDK_EXCEPTION);
     327             :         }
     328      209308 :         *r = b->batCacheid;
     329      209308 :         BBPretain(b->batCacheid);
     330      210008 :         BBPunfix(b->batCacheid);
     331      209923 :         BBPunfix(p->batCacheid);
     332      210017 :         BBPunfix(u->batCacheid);
     333      210017 :         return MAL_SUCCEED;
     334             : }
     335             : 
     336             : static str
     337           2 : BKCbat_inplace(bat *r, const bat *bid, const bat *rid, const bat *uid)
     338             : {
     339           2 :         bit F = FALSE;
     340             : 
     341           2 :         return BKCbat_inplace_force(r, bid, rid, uid, &F);
     342             : }
     343             : 
     344             : /*end of SQL enhancement */
     345             : 
     346             : static str
     347           0 : BKCgetCapacity(lng *res, const bat *bid)
     348             : {
     349           0 :         *res = lng_nil;
     350           0 :         BAT *b = BBPquickdesc(*bid);
     351             : 
     352           0 :         if (b == NULL)
     353           0 :                 throw(MAL, "bat.getCapacity", ILLEGAL_ARGUMENT);
     354           0 :         *res = (lng) BATcapacity(b);
     355           0 :         return MAL_SUCCEED;
     356             : }
     357             : 
     358             : static str
     359           0 : BKCgetColumnType(str *res, const bat *bid)
     360             : {
     361           0 :         const char *ret = str_nil;
     362           0 :         BAT *b = BBPquickdesc(*bid);
     363             : 
     364           0 :         if (b == NULL)
     365           0 :                 throw(MAL, "bat.getColumnType", ILLEGAL_ARGUMENT);
     366           0 :         ret = *bid < 0 ? ATOMname(TYPE_void) : ATOMname(b->ttype);
     367           0 :         *res = GDKstrdup(ret);
     368           0 :         if (*res == NULL)
     369           0 :                 throw(MAL, "bat.getColumnType", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     370             :         return MAL_SUCCEED;
     371             : }
     372             : 
     373             : static str
     374           0 : BKCisSorted(bit *res, const bat *bid)
     375             : {
     376           0 :         BAT *b;
     377             : 
     378           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     379           0 :                 throw(MAL, "bat.isSorted", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     380             :         }
     381           0 :         *res = BATordered(b);
     382           0 :         BBPunfix(b->batCacheid);
     383           0 :         return MAL_SUCCEED;
     384             : }
     385             : 
     386             : static str
     387           0 : BKCisSortedReverse(bit *res, const bat *bid)
     388             : {
     389           0 :         BAT *b;
     390             : 
     391           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     392           0 :                 throw(MAL, "bat.isSorted", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     393             :         }
     394           0 :         *res = BATordered_rev(b);
     395           0 :         BBPunfix(b->batCacheid);
     396           0 :         return MAL_SUCCEED;
     397             : }
     398             : 
     399             : /*
     400             :  * We must take care of the special case of a nil column (TYPE_void,seqbase=nil)
     401             :  * such nil columns never set tkey
     402             :  * a nil column of a BAT with <= 1 entries does not contain doubles => return TRUE.
     403             :  */
     404             : 
     405             : static str
     406           0 : BKCgetKey(bit *ret, const bat *bid)
     407             : {
     408           0 :         BAT *b;
     409             : 
     410           0 :         if ((b = BATdescriptor(*bid)) == NULL)
     411           0 :                 throw(MAL, "bat.setPersistence",
     412             :                           SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     413           0 :         MT_lock_set(&b->theaplock);
     414           0 :         *ret = b->tkey;
     415           0 :         MT_lock_unset(&b->theaplock);
     416           0 :         BBPunfix(b->batCacheid);
     417           0 :         return MAL_SUCCEED;
     418             : }
     419             : 
     420             : static str
     421           2 : BKCpersists(void *r, const bat *bid, const bit *flg)
     422             : {
     423           2 :         BAT *b;
     424             : 
     425           2 :         (void) r;
     426           2 :         if ((b = BATdescriptor(*bid)) == NULL) {
     427           0 :                 throw(MAL, "bat.setPersistence",
     428             :                           SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     429             :         }
     430           2 :         if (BATmode(b, (*flg != TRUE)) != GDK_SUCCEED) {
     431           0 :                 BBPunfix(b->batCacheid);
     432           0 :                 throw(MAL, "bat.setPersistence", ILLEGAL_ARGUMENT);
     433             :         }
     434           2 :         BBPunfix(b->batCacheid);
     435           2 :         return MAL_SUCCEED;
     436             : }
     437             : 
     438             : static str
     439           1 : BKCsetPersistent(void *r, const bat *bid)
     440             : {
     441           1 :         bit flag = TRUE;
     442           1 :         return BKCpersists(r, bid, &flag);
     443             : }
     444             : 
     445             : static str
     446           2 : BKCisPersistent(bit *res, const bat *bid)
     447             : {
     448           2 :         BAT *b;
     449             : 
     450           2 :         if ((b = BATdescriptor(*bid)) == NULL) {
     451           0 :                 throw(MAL, "bat.setPersistence",
     452             :                           SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     453             :         }
     454           2 :         MT_lock_set(&b->theaplock);
     455           2 :         *res = !b->batTransient;
     456           2 :         MT_lock_unset(&b->theaplock);
     457           2 :         BBPunfix(b->batCacheid);
     458           2 :         return MAL_SUCCEED;
     459             : }
     460             : 
     461             : static str
     462           1 : BKCsetTransient(void *r, const bat *bid)
     463             : {
     464           1 :         bit flag = FALSE;
     465           1 :         return BKCpersists(r, bid, &flag);
     466             : }
     467             : 
     468             : static str
     469           2 : BKCisTransient(bit *res, const bat *bid)
     470             : {
     471           2 :         BAT *b;
     472             : 
     473           2 :         if ((b = BATdescriptor(*bid)) == NULL) {
     474           0 :                 throw(MAL, "bat.setTransient", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     475             :         }
     476           2 :         MT_lock_set(&b->theaplock);
     477           2 :         *res = b->batTransient;
     478           2 :         MT_lock_unset(&b->theaplock);
     479           2 :         BBPunfix(b->batCacheid);
     480           2 :         return MAL_SUCCEED;
     481             : }
     482             : 
     483             : static str
     484           3 : BKCsetAccess(bat *res, const bat *bid, const char *const *param)
     485             : {
     486           3 :         BAT *b;
     487           3 :         restrict_t m;
     488             : 
     489           3 :         if ((b = BATdescriptor(*bid)) == NULL)
     490           0 :                 throw(MAL, "bat.setAccess", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     491           3 :         switch (*param[0]) {
     492             :         case 'r':
     493             :                 m = BAT_READ;
     494             :                 break;
     495           0 :         case 'a':
     496           0 :                 m = BAT_APPEND;
     497           0 :                 break;
     498           0 :         case 'w':
     499           0 :                 m = BAT_WRITE;
     500           0 :                 break;
     501           0 :         default:
     502           0 :                 *res = 0;
     503           0 :                 BBPunfix(b->batCacheid);
     504           0 :                 throw(MAL, "bat.setAccess",
     505             :                           ILLEGAL_ARGUMENT " Got %c" " expected 'r','a', or 'w'",
     506           0 :                           *param[0]);
     507             :         }
     508           3 :         if ((b = BATsetaccess(b, m)) == NULL)
     509           0 :                 throw(MAL, "bat.setAccess", OPERATION_FAILED);
     510           3 :         *res = b->batCacheid;
     511           3 :         BBPretain(b->batCacheid);
     512           3 :         BBPunfix(b->batCacheid);
     513           3 :         return MAL_SUCCEED;
     514             : }
     515             : 
     516             : static str
     517           0 : BKCgetAccess(str *res, const bat *bid)
     518             : {
     519           0 :         BAT *b;
     520             : 
     521           0 :         if ((b = BATdescriptor(*bid)) == NULL)
     522           0 :                 throw(MAL, "bat.getAccess", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     523           0 :         switch (BATgetaccess(b)) {
     524           0 :         case BAT_READ:
     525           0 :                 *res = GDKstrdup("read");
     526           0 :                 break;
     527           0 :         case BAT_APPEND:
     528           0 :                 *res = GDKstrdup("append");
     529           0 :                 break;
     530           0 :         case BAT_WRITE:
     531           0 :                 *res = GDKstrdup("write");
     532           0 :                 break;
     533             :         default:
     534           0 :                 MT_UNREACHABLE();
     535             :         }
     536           0 :         BBPunfix(b->batCacheid);
     537           0 :         if (*res == NULL)
     538           0 :                 throw(MAL, "bat.getAccess", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     539             :         return MAL_SUCCEED;
     540             : }
     541             : 
     542             : /*
     543             :  * Property management
     544             :  * All property operators should ensure exclusive access to the BAT
     545             :  * descriptor.
     546             :  * Where necessary use the primary view to access the properties
     547             :  */
     548             : static inline char *
     549           0 : pre(const char *s1, const char *s2, char *buf)
     550             : {
     551           0 :         snprintf(buf, 64, "%s%s", s1, s2);
     552           0 :         return buf;
     553             : }
     554             : 
     555             : static inline char *
     556           0 : local_itoa(ssize_t i, char *buf)
     557             : {
     558           0 :         snprintf(buf, 32, "%zd", i);
     559           0 :         return buf;
     560             : }
     561             : 
     562             : static inline char *
     563           0 : local_utoa(size_t i, char *buf)
     564             : {
     565           0 :         snprintf(buf, 32, "%zu", i);
     566           0 :         return buf;
     567             : }
     568             : 
     569             : static inline char *
     570           0 : oidtostr(oid i, char *p, size_t len)
     571             : {
     572           0 :         if (OIDtoStr(&p, &len, &i, false) < 0)
     573             :                 return NULL;
     574           0 :         return p;
     575             : }
     576             : 
     577             : static gdk_return
     578           0 : infoHeap(BAT *bk, BAT *bv, Heap *hp, const char *nme)
     579             : {
     580           0 :         char buf[1024], *p = buf;
     581             : 
     582           0 :         if (!hp)
     583             :                 return GDK_SUCCEED;
     584           0 :         while (*nme)
     585           0 :                 *p++ = *nme++;
     586           0 :         strcpy(p, "free");
     587           0 :         if (BUNappend(bk, buf, false) != GDK_SUCCEED ||
     588           0 :                 BUNappend(bv, local_utoa(hp->free, buf), false) != GDK_SUCCEED)
     589           0 :                 return GDK_FAIL;
     590           0 :         strcpy(p, "size");
     591           0 :         if (BUNappend(bk, buf, false) != GDK_SUCCEED ||
     592           0 :                 BUNappend(bv, local_utoa(hp->size, buf), false) != GDK_SUCCEED)
     593           0 :                 return GDK_FAIL;
     594           0 :         strcpy(p, "storage");
     595           0 :         if (BUNappend(bk, buf, false) != GDK_SUCCEED ||
     596           0 :                 BUNappend(bv, (hp->base == NULL || hp->base == (char *) 1) ? "absent" : (hp->storage == STORE_MMAP) ? (hp-> filename [0] ? "memory mapped" : "anonymous vm") : (hp->storage == STORE_PRIV) ? "private map" : "malloced", false) != GDK_SUCCEED)
     597           0 :                 return GDK_FAIL;
     598           0 :         strcpy(p, "newstorage");
     599           0 :         if (BUNappend(bk, buf, false) != GDK_SUCCEED ||
     600           0 :                 BUNappend(bv, (hp->newstorage == STORE_MEM) ? "malloced" : (hp->newstorage == STORE_PRIV) ? "private map" : "memory mapped", false) != GDK_SUCCEED)
     601           0 :                 return GDK_FAIL;
     602           0 :         strcpy(p, "filename");
     603           0 :         if (BUNappend(bk, buf, false) != GDK_SUCCEED ||
     604           0 :                 BUNappend(bv, hp->filename[0] ? hp->filename : "no file",
     605             :                                   false) != GDK_SUCCEED)
     606           0 :                 return GDK_FAIL;
     607             :         return GDK_SUCCEED;
     608             : }
     609             : 
     610             : #define COLLISION (8 * sizeof(size_t))
     611             : 
     612             : static gdk_return
     613           0 : HASHinfo(BAT *bk, BAT *bv, Hash *h, const char *s)
     614             : {
     615           0 :         BUN i;
     616           0 :         BUN j;
     617           0 :         BUN k;
     618           0 :         BUN cnt[COLLISION + 1];
     619           0 :         char buf[32];
     620           0 :         char prebuf[64];
     621             : 
     622           0 :         if (BUNappend(bk, pre(s, "type", prebuf), false) != GDK_SUCCEED ||
     623           0 :                 BUNappend(bv, ATOMname(h->type), false) != GDK_SUCCEED ||
     624           0 :                 BUNappend(bk, pre(s, "mask", prebuf), false) != GDK_SUCCEED ||
     625           0 :                 BUNappend(bv, local_utoa(h->nbucket, buf), false) != GDK_SUCCEED)
     626           0 :                 return GDK_FAIL;
     627             : 
     628           0 :         for (i = 0; i < COLLISION + 1; i++) {
     629           0 :                 cnt[i] = 0;
     630             :         }
     631           0 :         for (i = 0; i < h->nbucket; i++) {
     632           0 :                 j = HASHlist(h, i);
     633           0 :                 for (k = 0; j; k++)
     634           0 :                         j >>= 1;
     635           0 :                 cnt[k]++;
     636             :         }
     637             : 
     638           0 :         for (i = 0; i < COLLISION + 1; i++)
     639           0 :                 if (cnt[i]) {
     640           0 :                         if (BUNappend(bk,
     641           0 :                                                   pre(s, local_utoa(i ? (((size_t) 1) << (i - 1)) : 0,
     642             :                                                                                         buf),
     643             :                                                           prebuf),
     644             :                                                   false) != GDK_SUCCEED
     645           0 :                                 || BUNappend(bv, local_utoa((size_t) cnt[i], buf),
     646             :                                                          false) != GDK_SUCCEED)
     647           0 :                                 return GDK_FAIL;
     648             :                 }
     649             :         return GDK_SUCCEED;
     650             : }
     651             : 
     652             : static str
     653           0 : BKCinfo(bat *ret1, bat *ret2, const bat *bid)
     654             : {
     655           0 :         const char *mode, *accessmode;
     656           0 :         BAT *bk = NULL, *bv = NULL, *b;
     657           0 :         char bf[oidStrlen];
     658           0 :         char buf[32];
     659             : 
     660           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     661           0 :                 throw(MAL, "bat.info", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     662             :         }
     663             : 
     664           0 :         bk = COLnew(0, TYPE_str, 128, TRANSIENT);
     665           0 :         bv = COLnew(0, TYPE_str, 128, TRANSIENT);
     666           0 :         if (bk == NULL || bv == NULL) {
     667           0 :                 BBPreclaim(bk);
     668           0 :                 BBPreclaim(bv);
     669           0 :                 BBPunfix(b->batCacheid);
     670           0 :                 throw(MAL, "bat.info", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     671             :         }
     672             : 
     673           0 :         BATiter bi = bat_iterator(b);
     674           0 :         if (bi.transient) {
     675             :                 mode = "transient";
     676             :         } else {
     677           0 :                 mode = "persistent";
     678             :         }
     679             : 
     680           0 :         switch (bi.restricted) {
     681             :         case BAT_READ:
     682             :                 accessmode = "read-only";
     683             :                 break;
     684           0 :         case BAT_WRITE:
     685           0 :                 accessmode = "updatable";
     686           0 :                 break;
     687           0 :         case BAT_APPEND:
     688           0 :                 accessmode = "append-only";
     689           0 :                 break;
     690           0 :         default:
     691           0 :                 accessmode = "unknown";
     692             :         }
     693             : 
     694           0 :         if (BUNappend(bk, "batId", false) != GDK_SUCCEED
     695           0 :                 || BUNappend(bv, BATgetId(b), false) != GDK_SUCCEED
     696           0 :                 || BUNappend(bk, "batCacheid", false) != GDK_SUCCEED
     697           0 :                 || BUNappend(bv, local_itoa((ssize_t) b->batCacheid, buf),
     698             :                                          false) != GDK_SUCCEED
     699           0 :                 || BUNappend(bk, "tparentid", false) != GDK_SUCCEED
     700           0 :                 || BUNappend(bv, local_itoa((ssize_t) bi.h->parentid, buf),
     701             :                                          false) != GDK_SUCCEED
     702           0 :                 || BUNappend(bk, "batCount", false) != GDK_SUCCEED
     703           0 :                 || BUNappend(bv, local_utoa((size_t) bi.count, buf),
     704             :                                          false) != GDK_SUCCEED
     705           0 :                 || BUNappend(bk, "batCapacity", false) != GDK_SUCCEED
     706           0 :                 || BUNappend(bv, local_utoa((size_t) b->batCapacity, buf),
     707             :                                          false) != GDK_SUCCEED
     708           0 :                 || BUNappend(bk, "head", false) != GDK_SUCCEED
     709           0 :                 || BUNappend(bv, ATOMname(TYPE_void), false) != GDK_SUCCEED
     710           0 :                 || BUNappend(bk, "tail", false) != GDK_SUCCEED
     711           0 :                 || BUNappend(bv, ATOMname(bi.type), false) != GDK_SUCCEED
     712           0 :                 || BUNappend(bk, "batPersistence", false) != GDK_SUCCEED
     713           0 :                 || BUNappend(bv, mode, false) != GDK_SUCCEED
     714           0 :                 || BUNappend(bk, "batRestricted", false) != GDK_SUCCEED
     715           0 :                 || BUNappend(bv, accessmode, false) != GDK_SUCCEED
     716           0 :                 || BUNappend(bk, "batRefcnt", false) != GDK_SUCCEED
     717           0 :                 || BUNappend(bv, local_itoa((ssize_t) BBP_refs(b->batCacheid), buf),
     718             :                                          false) != GDK_SUCCEED
     719           0 :                 || BUNappend(bk, "batLRefcnt", false) != GDK_SUCCEED
     720           0 :                 || BUNappend(bv, local_itoa((ssize_t) BBP_lrefs(b->batCacheid), buf),
     721             :                                          false) != GDK_SUCCEED
     722           0 :                 || BUNappend(bk, "batDirty", false) != GDK_SUCCEED
     723           0 :                 || BUNappend(bv, BATdirtybi(bi) ? "dirty" : "clean",
     724             :                                          false) != GDK_SUCCEED
     725           0 :                 || BUNappend(bk, "hseqbase", false) != GDK_SUCCEED
     726           0 :                 || BUNappend(bv, oidtostr(b->hseqbase, bf, sizeof(bf)),
     727             :                                          FALSE) != GDK_SUCCEED
     728           0 :                 || BUNappend(bk, "tdense", false) != GDK_SUCCEED
     729           0 :                 || BUNappend(bv, local_itoa((ssize_t) BATtdensebi(&bi), buf),
     730             :                                          false) != GDK_SUCCEED
     731           0 :                 || BUNappend(bk, "tseqbase", false) != GDK_SUCCEED
     732           0 :                 || BUNappend(bv, oidtostr(bi.tseq, bf, sizeof(bf)),
     733             :                                          FALSE) != GDK_SUCCEED
     734           0 :                 || BUNappend(bk, "tsorted", false) != GDK_SUCCEED
     735           0 :                 || BUNappend(bv, local_itoa((ssize_t) bi.sorted, buf),
     736             :                                          false) != GDK_SUCCEED
     737           0 :                 || BUNappend(bk, "trevsorted", false) != GDK_SUCCEED
     738           0 :                 || BUNappend(bv, local_itoa((ssize_t) bi.revsorted, buf),
     739             :                                          false) != GDK_SUCCEED
     740           0 :                 || BUNappend(bk, "tkey", false) != GDK_SUCCEED
     741           0 :                 || BUNappend(bv, local_itoa((ssize_t) bi.key, buf),
     742             :                                          false) != GDK_SUCCEED
     743           0 :                 || BUNappend(bk, "tvarsized", false) != GDK_SUCCEED
     744           0 :                 || BUNappend(bv,
     745           0 :                                          local_itoa((ssize_t)
     746             :                                                                 (bi.type == TYPE_void
     747           0 :                                                                  || bi.vh != NULL), buf), false) != GDK_SUCCEED
     748           0 :                 || BUNappend(bk, "tnosorted", false) != GDK_SUCCEED
     749           0 :                 || BUNappend(bv, local_utoa(bi.nosorted, buf), false) != GDK_SUCCEED
     750           0 :                 || BUNappend(bk, "tnorevsorted", false) != GDK_SUCCEED
     751           0 :                 || BUNappend(bv, local_utoa(bi.norevsorted, buf), false) != GDK_SUCCEED
     752           0 :                 || BUNappend(bk, "tnokey[0]", false) != GDK_SUCCEED
     753           0 :                 || BUNappend(bv, local_utoa(bi.nokey[0], buf), false) != GDK_SUCCEED
     754           0 :                 || BUNappend(bk, "tnokey[1]", false) != GDK_SUCCEED
     755           0 :                 || BUNappend(bv, local_utoa(bi.nokey[1], buf), false) != GDK_SUCCEED
     756           0 :                 || BUNappend(bk, "tnonil", false) != GDK_SUCCEED
     757           0 :                 || BUNappend(bv, local_utoa(bi.nonil, buf), false) != GDK_SUCCEED
     758           0 :                 || BUNappend(bk, "tnil", false) != GDK_SUCCEED
     759           0 :                 || BUNappend(bv, local_utoa(bi.nil, buf), false) != GDK_SUCCEED
     760           0 :                 || BUNappend(bk, "batInserted", false) != GDK_SUCCEED
     761           0 :                 || BUNappend(bv, local_utoa(b->batInserted, buf), false) != GDK_SUCCEED
     762           0 :                 || BUNappend(bk, "ttop", false) != GDK_SUCCEED
     763           0 :                 || BUNappend(bv, local_utoa(bi.hfree, buf), false) != GDK_SUCCEED
     764           0 :                 || BUNappend(bk, "batCopiedtodisk", false) != GDK_SUCCEED
     765           0 :                 || BUNappend(bv, local_itoa((ssize_t) bi.copiedtodisk, buf),
     766             :                                          false) != GDK_SUCCEED
     767           0 :                 || BUNappend(bk, "theap.dirty", false) != GDK_SUCCEED
     768           0 :                 || BUNappend(bv, bi.hdirty ? "dirty" : "clean", false) != GDK_SUCCEED
     769           0 :                 || infoHeap(bk, bv, bi.h, "tail.") != GDK_SUCCEED
     770           0 :                 || BUNappend(bk, "tvheap->dirty", false) != GDK_SUCCEED
     771           0 :                 || BUNappend(bv, bi.vhdirty ? "dirty" : "clean", false) != GDK_SUCCEED
     772           0 :                 || infoHeap(bk, bv, bi.vh, "theap.") != GDK_SUCCEED) {
     773           0 :                 bat_iterator_end(&bi);
     774           0 :                 BBPreclaim(bk);
     775           0 :                 BBPreclaim(bv);
     776           0 :                 BBPunfix(b->batCacheid);
     777           0 :                 throw(MAL, "bat.info", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     778             :         }
     779             :         /* dump index information */
     780           0 :         MT_rwlock_rdlock(&b->thashlock);
     781           0 :         if (b->thash && HASHinfo(bk, bv, b->thash, "thash->") != GDK_SUCCEED) {
     782           0 :                 MT_rwlock_rdunlock(&b->thashlock);
     783           0 :                 bat_iterator_end(&bi);
     784           0 :                 BBPreclaim(bk);
     785           0 :                 BBPreclaim(bv);
     786           0 :                 BBPunfix(b->batCacheid);
     787           0 :                 throw(MAL, "bat.info", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     788             :         }
     789           0 :         MT_rwlock_rdunlock(&b->thashlock);
     790           0 :         bat_iterator_end(&bi);
     791           0 :         assert(BATcount(bk) == BATcount(bv));
     792           0 :         BBPunfix(b->batCacheid);
     793           0 :         *ret1 = bk->batCacheid;
     794           0 :         BBPkeepref(bk);
     795           0 :         *ret2 = bv->batCacheid;
     796           0 :         BBPkeepref(bv);
     797           0 :         return MAL_SUCCEED;
     798             : }
     799             : 
     800             : // get the actual size of all constituents, also for views
     801             : #define ROUND_UP(x,y) ((y)*(((x)+(y)-1)/(y)))
     802             : 
     803             : static str
     804           0 : BKCgetSize(lng *tot, const bat *bid)
     805             : {
     806           0 :         BAT *b;
     807           0 :         lng size = 0;
     808           0 :         lng blksize = (lng) MT_pagesize();
     809           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     810           0 :                 throw(MAL, "bat.getDiskSize", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     811             :         }
     812             : 
     813           0 :         size = sizeof(bat);
     814             : 
     815           0 :         MT_lock_set(&b->theaplock);
     816           0 :         if (!isVIEW(b)) {
     817           0 :                 BUN cnt = BATcapacity(b);
     818           0 :                 size += ROUND_UP(b->theap->free, blksize);
     819           0 :                 if (b->tvheap)
     820           0 :                         size += ROUND_UP(b->tvheap->free, blksize);
     821           0 :                 MT_lock_unset(&b->theaplock);
     822             : 
     823           0 :                 if (b->thash)
     824           0 :                         size += ROUND_UP(sizeof(BUN) * cnt, blksize);
     825             :         } else {
     826           0 :                 MT_lock_unset(&b->theaplock);
     827             :         }
     828           0 :         *tot = size;
     829           0 :         BBPunfix(*bid);
     830           0 :         return MAL_SUCCEED;
     831             : }
     832             : 
     833             : static str
     834           3 : BKCgetVHeapSize(lng *tot, const bat *bid)
     835             : {
     836           3 :         BAT *b;
     837           3 :         lng size = 0;
     838           3 :         if ((b = BATdescriptor(*bid)) == NULL) {
     839           0 :                 throw(MAL, "bat.getVHeapSize", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     840             :         }
     841           3 :         if (ATOMvarsized(b->ttype)) {
     842           3 :                 MT_lock_set(&b->theaplock);
     843           3 :                 if (b->tvheap)
     844           3 :                         size += b->tvheap->size;
     845           3 :                 MT_lock_unset(&b->theaplock);
     846             :         }
     847             : 
     848           3 :         *tot = size;
     849           3 :         BBPunfix(*bid);
     850           3 :         return MAL_SUCCEED;
     851             : }
     852             : 
     853             : /*
     854             :  * Synced BATs
     855             :  */
     856             : static str
     857           0 : BKCisSynced(bit *ret, const bat *bid1, const bat *bid2)
     858             : {
     859           0 :         BAT *b1, *b2;
     860             : 
     861           0 :         if ((b1 = BATdescriptor(*bid1)) == NULL) {
     862           0 :                 throw(MAL, "bat.isSynced", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     863             :         }
     864           0 :         if ((b2 = BATdescriptor(*bid2)) == NULL) {
     865           0 :                 BBPunfix(b1->batCacheid);
     866           0 :                 throw(MAL, "bat.isSynced", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     867             :         }
     868           0 :         *ret = ALIGNsynced(b1, b2) != 0;
     869           0 :         BBPunfix(b1->batCacheid);
     870           0 :         BBPunfix(b2->batCacheid);
     871           0 :         return MAL_SUCCEED;
     872             : }
     873             : 
     874             : /*
     875             :  * Role Management
     876             :  */
     877             : static str
     878           9 : BKCsetName(void *r, const bat *bid, const char *const *s)
     879             : {
     880           9 :         BAT *b;
     881           9 :         int ret;
     882           9 :         int c;
     883           9 :         const char *t = *s;
     884             : 
     885           9 :         (void) r;
     886           9 :         if ((b = BATdescriptor(*bid)) == NULL)
     887           0 :                 throw(MAL, "bat.setName", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     888             : 
     889          52 :         for (; (c = *t) != 0; t++)
     890          43 :                 if (c != '_' && !GDKisalnum(c)) {
     891           0 :                         BBPunfix(b->batCacheid);
     892           0 :                         throw(MAL, "bat.setName",
     893             :                                   ILLEGAL_ARGUMENT ": identifier expected: %s", *s);
     894             :                 }
     895             : 
     896           9 :         t = *s;
     897           9 :         ret = BBPrename(b, t);
     898           9 :         BBPunfix(b->batCacheid);
     899           9 :         switch (ret) {
     900           0 :         case BBPRENAME_ILLEGAL:
     901           0 :                 GDKclrerr();
     902           0 :                 throw(MAL, "bat.setName",
     903             :                           ILLEGAL_ARGUMENT ": illegal temporary name: '%s'", t);
     904           0 :         case BBPRENAME_LONG:
     905           0 :                 GDKclrerr();
     906           0 :                 throw(MAL, "bat.setName", ILLEGAL_ARGUMENT ": name too long: '%s'", t);
     907           0 :         case BBPRENAME_MEMORY:
     908           0 :                 GDKclrerr();
     909           0 :                 throw(MAL, "bat.setName", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     910           1 :         case BBPRENAME_ALREADY:
     911           1 :                 GDKclrerr();
     912             :                 /* fall through */
     913             :         case 0:
     914             :                 break;
     915             :         }
     916             :         return MAL_SUCCEED;
     917             : }
     918             : 
     919             : static str
     920           0 : BKCgetBBPname(str *ret, const bat *bid)
     921             : {
     922           0 :         BAT *b;
     923             : 
     924           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     925           0 :                 throw(MAL, "bat.getName", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     926             :         }
     927           0 :         *ret = GDKstrdup(BBP_logical(b->batCacheid));
     928           0 :         BBPunfix(b->batCacheid);
     929           0 :         return *ret ? MAL_SUCCEED : createException(MAL, "bat.getName",
     930             :                                                                                                 SQLSTATE(HY013)
     931             :                                                                                                 MAL_MALLOC_FAIL);
     932             : }
     933             : 
     934             : static str
     935           0 : BKCsave(bit *res, const char *const *input)
     936             : {
     937           0 :         bat bid = BBPindex(*input);
     938           0 :         BAT *b;
     939             : 
     940           0 :         *res = FALSE;
     941           0 :         if (!is_bat_nil(bid)) {
     942           0 :                 if ((b = BATdescriptor(bid)) != NULL) {
     943           0 :                         if (BATdirty(b)) {
     944           0 :                                 if (BBPsave(b) == GDK_SUCCEED)
     945           0 :                                         *res = TRUE;
     946             :                         }
     947           0 :                         BBPunfix(bid);
     948           0 :                         return MAL_SUCCEED;
     949             :                 }
     950           0 :                 throw(MAL, "bat.save", "fix failed");
     951             :         }
     952             :         return MAL_SUCCEED;
     953             : }
     954             : 
     955             : static str
     956           0 : BKCsave2(void *r, const bat *bid)
     957             : {
     958           0 :         BAT *b;
     959             : 
     960           0 :         (void) r;
     961           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     962           0 :                 throw(MAL, "bat.save", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     963             :         }
     964           0 :         MT_lock_set(&b->theaplock);
     965           0 :         if (!b->batTransient) {
     966           0 :                 MT_lock_unset(&b->theaplock);
     967           0 :                 BBPunfix(b->batCacheid);
     968           0 :                 throw(MAL, "bat.save", "Only save transient columns.");
     969             :         }
     970           0 :         MT_lock_unset(&b->theaplock);
     971             : 
     972           0 :         if (b && BATdirty(b))
     973           0 :                 BBPsave(b);
     974           0 :         BBPunfix(b->batCacheid);
     975           0 :         return MAL_SUCCEED;
     976             : }
     977             : 
     978             : /*
     979             :  * Accelerator Control
     980             :  */
     981             : static str
     982           0 : BKCsetHash(bit *ret, const bat *bid)
     983             : {
     984           0 :         BAT *b;
     985             : 
     986           0 :         (void) ret;
     987           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
     988           0 :                 throw(MAL, "bat.setHash", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
     989             :         }
     990           0 :         *ret = BAThash(b) == GDK_SUCCEED;
     991           0 :         BBPunfix(b->batCacheid);
     992           0 :         return MAL_SUCCEED;
     993             : }
     994             : 
     995             : static str
     996           0 : BKCgetSequenceBase(oid *r, const bat *bid)
     997             : {
     998           0 :         BAT *b;
     999             : 
    1000           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
    1001           0 :                 throw(MAL, "bat.setSequenceBase",
    1002             :                           SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1003             :         }
    1004           0 :         *r = b->hseqbase;
    1005           0 :         BBPunfix(b->batCacheid);
    1006           0 :         return MAL_SUCCEED;
    1007             : }
    1008             : 
    1009             : static str
    1010      113403 : BKCmergecand(bat *ret, const bat *aid, const bat *bid)
    1011             : {
    1012      113403 :         BAT *a, *b, *bn;
    1013             : 
    1014      113403 :         if ((a = BATdescriptor(*aid)) == NULL) {
    1015           0 :                 throw(MAL, "bat.mergecand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1016             :         }
    1017      113684 :         if ((b = BATdescriptor(*bid)) == NULL) {
    1018           0 :                 BBPunfix(a->batCacheid);
    1019           0 :                 throw(MAL, "bat.mergecand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1020             :         }
    1021      113684 :         bn = BATmergecand(a, b);
    1022      113365 :         BBPunfix(a->batCacheid);
    1023      113554 :         BBPunfix(b->batCacheid);
    1024      113604 :         if (bn == NULL)
    1025           0 :                 throw(MAL, "bat.mergecand", GDK_EXCEPTION);
    1026      113604 :         *ret = bn->batCacheid;
    1027      113604 :         BBPkeepref(bn);
    1028      113604 :         return MAL_SUCCEED;
    1029             : }
    1030             : 
    1031             : static str
    1032           0 : BKCintersectcand(bat *ret, const bat *aid, const bat *bid)
    1033             : {
    1034           0 :         BAT *a, *b, *bn;
    1035             : 
    1036           0 :         if ((a = BATdescriptor(*aid)) == NULL) {
    1037           0 :                 throw(MAL, "bat.intersectcand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1038             :         }
    1039           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
    1040           0 :                 BBPunfix(a->batCacheid);
    1041           0 :                 throw(MAL, "bat.intersectcand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1042             :         }
    1043           0 :         bn = BATintersectcand(a, b);
    1044           0 :         BBPunfix(a->batCacheid);
    1045           0 :         BBPunfix(b->batCacheid);
    1046           0 :         if (bn == NULL)
    1047           0 :                 throw(MAL, "bat.intersectcand", GDK_EXCEPTION);
    1048           0 :         *ret = bn->batCacheid;
    1049           0 :         BBPkeepref(bn);
    1050           0 :         return MAL_SUCCEED;
    1051             : }
    1052             : 
    1053             : static str
    1054           0 : BKCdiffcand(bat *ret, const bat *aid, const bat *bid)
    1055             : {
    1056           0 :         BAT *a, *b, *bn;
    1057             : 
    1058           0 :         if ((a = BATdescriptor(*aid)) == NULL) {
    1059           0 :                 throw(MAL, "bat.diffcand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1060             :         }
    1061           0 :         if ((b = BATdescriptor(*bid)) == NULL) {
    1062           0 :                 BBPunfix(a->batCacheid);
    1063           0 :                 throw(MAL, "bat.diffcand", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
    1064             :         }
    1065           0 :         bn = BATdiffcand(a, b);
    1066           0 :         BBPunfix(a->batCacheid);
    1067           0 :         BBPunfix(b->batCacheid);
    1068           0 :         if (bn == NULL)
    1069           0 :                 throw(MAL, "bat.diffcand", GDK_EXCEPTION);
    1070           0 :         *ret = bn->batCacheid;
    1071           0 :         BBPkeepref(bn);
    1072           0 :         return MAL_SUCCEED;
    1073             : }
    1074             : 
    1075             : #include "mel.h"
    1076             : mel_func bat5_init_funcs[] = {
    1077             :  command("bat", "mirror", BKCmirror, false, "Returns the head-mirror image of a BAT (two head columns).", args(1,2, batarg("",oid),batargany("b",1))),
    1078             :  command("bat", "delete", BKCdelete, false, "Delete BUN indicated by head value, exchanging with last BUN", args(1,3, batargany("",1),batargany("b",1),arg("h",oid))),
    1079             :  command("bat", "delete", BKCdelete_multi, false, "Delete multiple BUN, shifting BUNs up", args(1,3, batargany("",1),batargany("b",1),batarg("d",oid))),
    1080             :  command("bat", "delete", BKCdelete_all, false, "Delete all entries.", args(1,2, batargany("",1),batargany("b",1))),
    1081             :  command("bat", "replace", BKCbun_inplace, false, "Replace the tail value of one BUN that has some head value.", args(1,4, batargany("",1),batargany("b",1),arg("h",oid),argany("t",1))),
    1082             :  command("bat", "replace", BKCbun_inplace_force, false, "Replace the tail value of one BUN that has some head value.", args(1,5, batargany("",1),batargany("b",1),arg("h",oid),argany("t",1),arg("force",bit))),
    1083             :  command("bat", "replace", BKCbat_inplace, false, "Perform replace for all BUNs of the second BAT into the first.", args(1,4, batargany("",1),batargany("b",1),batarg("rid",oid),batargany("val",1))),
    1084             :  command("bat", "replace", BKCbat_inplace_force, false, "Perform replace for all BUNs of the second BAT into the first.", args(1,5, batargany("",1),batargany("b",1),batarg("rid",oid),batargany("val",1),arg("force",bit))),
    1085             :  command("bat", "append", BKCappend_wrap, false, "append the content of u to i", args(1,3, batargany("",1),batargany("i",1),batargany("u",1))),
    1086             :  command("bat", "append", BKCappend_force_wrap, false, "append the content of u to i", args(1,4, batargany("",1),batargany("i",1),batargany("u",1),arg("force",bit))),
    1087             :  command("bat", "append", BKCappend_cand_wrap, false, "append the content of u with candidate list s to i", args(1,4, batargany("",1),batargany("i",1),batargany("u",1),batarg("s",oid))),
    1088             :  command("bat", "append", BKCappend_cand_force_wrap, false, "append the content of u with candidate list s to i", args(1,5, batargany("",1),batargany("i",1),batargany("u",1),batarg("s",oid),arg("force",bit))),
    1089             :  command("bat", "append", BKCappend_val_force_wrap, false, "append the value u to i", args(1,4, batargany("",1),batargany("i",1),argany("u",1),arg("force",bit))),
    1090             :  command("bat", "densebat", BKCdensebat, false, "Creates a new [void,void] BAT of size 'sz'.", args(1,2, batarg("",oid),arg("sz",lng))),
    1091             :  command("bat", "info", BKCinfo, false, "Produce a table containing information about a BAT in [attribute,value] format. \nIt contains all properties of the BAT record. ", args(2,3, batarg("",str),batarg("",str),batargany("b",1))),
    1092             :  command("bat", "getSize", BKCgetSize, false, "Calculate the actual size of the BAT descriptor, heaps, hashes in bytes\nrounded to the memory page size (see bbp.getPageSize()).", args(1,2, arg("",lng),batargany("b",1))),
    1093             :  command("bat", "getVHeapSize", BKCgetVHeapSize, false, "Calculate the vheap size for varsized bats", args(1,2, arg("",lng),batargany("b",1))),
    1094             :  command("bat", "getCapacity", BKCgetCapacity, false, "Returns the current allocation size (in max number of elements) of a BAT.", args(1,2, arg("",lng),batargany("b",1))),
    1095             :  command("bat", "getColumnType", BKCgetColumnType, false, "Returns the type of the tail column of a BAT, as an integer type number.", args(1,2, arg("",str),batargany("b",1))),
    1096             :  command("bat", "isaKey", BKCgetKey, false, "Return whether the column tail values are unique (key).", args(1,2, arg("",bit),batargany("b",1))),
    1097             :  command("bat", "setAccess", BKCsetAccess, false, "Try to change the update access privileges \nto this BAT. Mode:\nr[ead-only]      - allow only read access.\na[append-only]   - allow reads and update.\nw[riteable]      - allow all operations.\nBATs are updatable by default. On making a BAT read-only, \nall subsequent updates fail with an error message.\nReturns the BAT itself.", args(1,3, batargany("",1),batargany("b",1),arg("mode",str))),
    1098             :  command("bat", "getAccess", BKCgetAccess, false, "Return the access mode attached to this BAT as a character.", args(1,2, arg("",str),batargany("b",1))),
    1099             :  command("bat", "getSequenceBase", BKCgetSequenceBase, false, "Get the sequence base for the void column of a BAT.", args(1,2, arg("",oid),batargany("b",1))),
    1100             :  command("bat", "isSorted", BKCisSorted, false, "Returns true if BAT values are ordered.", args(1,2, arg("",bit),batargany("b",1))),
    1101             :  command("bat", "isSortedReverse", BKCisSortedReverse, false, "Returns true if BAT values are reversely ordered.", args(1,2, arg("",bit),batargany("b",1))),
    1102             :  command("bat", "append", BKCappend_val_wrap, false, "append the value u to i", args(1,3, batargany("",1),batargany("i",1),argany("u",1))),
    1103             :  command("bat", "setName", BKCsetName, false, "Give a logical name to a BAT. ", args(1,3, arg("",void),batargany("b",1),arg("s",str))),
    1104             :  command("bat", "getName", BKCgetBBPname, false, "Gives back the logical name of a BAT.", args(1,2, arg("",str),batargany("b",1))),
    1105             :  command("bat", "isTransient", BKCisTransient, false, "", args(1,2, arg("",bit),batargany("b",1))),
    1106             :  command("bat", "setTransient", BKCsetTransient, false, "Make the BAT transient.  Returns \nboolean which indicates if the\nBAT administration has indeed changed.", args(1,2, arg("",void),batargany("b",1))),
    1107             :  command("bat", "isPersistent", BKCisPersistent, false, "", args(1,2, arg("",bit),batargany("b",1))),
    1108             :  command("bat", "setPersistent", BKCsetPersistent, false, "Make the BAT persistent.", args(1,2, arg("",void),batargany("b",1))),
    1109             :  command("bat", "save", BKCsave2, false, "", args(1,2, arg("",void),batargany("nme",1))),
    1110             :  command("bat", "save", BKCsave, false, "Save a BAT to storage, if it was loaded and dirty.  \nReturns whether IO was necessary.  Please realize that \ncalling this function violates the atomic commit protocol!!", args(1,2, arg("",bit),arg("nme",str))),
    1111             :  command("bat", "setHash", BKCsetHash, false, "Create a hash structure on the column", args(1,2, arg("",bit),batargany("b",1))),
    1112             :  command("bat", "isSynced", BKCisSynced, false, "Tests whether two BATs are synced or not. ", args(1,3, arg("",bit),batargany("b1",1),batargany("b2",2))),
    1113             :  command("bat", "mergecand", BKCmergecand, false, "Merge two candidate lists into one", args(1,3, batarg("",oid),batarg("a",oid),batarg("b",oid))),
    1114             :  command("bat", "intersectcand", BKCintersectcand, false, "Intersect two candidate lists into one", args(1,3, batarg("",oid),batarg("a",oid),batarg("b",oid))),
    1115             :  command("bat", "diffcand", BKCdiffcand, false, "Calculate difference of two candidate lists", args(1,3, batarg("",oid),batarg("a",oid),batarg("b",oid))),
    1116             :  { .imp=NULL }
    1117             : };
    1118             : #include "mal_import.h"
    1119             : #ifdef _MSC_VER
    1120             : #undef read
    1121             : #pragma section(".CRT$XCU",read)
    1122             : #endif
    1123         350 : LIB_STARTUP_FUNC(init_bat5_mal)
    1124         350 : { mal_module("bat5", NULL, bat5_init_funcs); }

Generated by: LCOV version 1.14