/*-
 * Copyright (c) 2014-present MongoDB, Inc.
 * Copyright (c) 2008-2014 WiredTiger, Inc.
 *	All rights reserved.
 *
 * See the file LICENSE for redistribution information.
 */

#include "wt_internal.h"

/*
 * __clsm_close_bulk --
 *     WT_CURSOR->close method for LSM bulk cursors.
 */
static int
__clsm_close_bulk(WT_CURSOR *cursor)
{
    WT_CURSOR *bulk_cursor;
    WT_CURSOR_LSM *clsm;
    WT_LSM_CHUNK *chunk;
    WT_LSM_TREE *lsm_tree;
    WT_SESSION_IMPL *session;
    uint64_t avg_chunks, total_chunks;

    clsm = (WT_CURSOR_LSM *)cursor;
    lsm_tree = clsm->lsm_tree;
    chunk = lsm_tree->chunk[0];
    session = CUR2S(clsm);

    /* Close the bulk cursor to ensure the chunk is written to disk. */
    bulk_cursor = clsm->chunks[0]->cursor;
    WT_RET(bulk_cursor->close(bulk_cursor));
    clsm->nchunks = 0;

    /* Set ondisk, and flush the metadata */
    F_SET(chunk, WT_LSM_CHUNK_ONDISK);
    /*
     * Setup a generation in our chunk based on how many chunk_size pieces fit into a chunk of a
     * given generation. This allows future LSM merges choose reasonable sets of chunks.
     */
    avg_chunks = (lsm_tree->merge_min + lsm_tree->merge_max) / 2;
    for (total_chunks = chunk->size / lsm_tree->chunk_size; total_chunks > 1;
         total_chunks /= avg_chunks)
        ++chunk->generation;

    WT_RET(__wti_lsm_meta_write(session, lsm_tree, NULL));
    ++lsm_tree->dsk_gen;

    /* Close the LSM cursor */
    WT_RET(__wti_clsm_close(cursor));
    WT_STAT_CONN_DECR_ATOMIC(session, cursor_bulk_count);

    return (0);
}
/*
 * __clsm_insert_bulk --
 *     WT_CURSOR->insert method for LSM bulk cursors.
 */
static int
__clsm_insert_bulk(WT_CURSOR *cursor)
{
    WT_CURSOR *bulk_cursor;
    WT_CURSOR_LSM *clsm;
    WT_LSM_CHUNK *chunk;
    WT_LSM_TREE *lsm_tree;
    WT_SESSION_IMPL *session;

    clsm = (WT_CURSOR_LSM *)cursor;
    lsm_tree = clsm->lsm_tree;
    chunk = lsm_tree->chunk[0];
    session = CUR2S(clsm);

    WT_ASSERT(session, lsm_tree->nchunks == 1 && clsm->nchunks == 1);
    ++chunk->count;
    chunk->size += cursor->key.size + cursor->value.size;
    bulk_cursor = clsm->chunks[0]->cursor;
    bulk_cursor->set_key(bulk_cursor, &cursor->key);
    bulk_cursor->set_value(bulk_cursor, &cursor->value);
    WT_RET(bulk_cursor->insert(bulk_cursor));

    return (0);
}

/*
 * __wti_clsm_open_bulk --
 *     WT_SESSION->open_cursor method for LSM bulk cursors.
 */
int
__wti_clsm_open_bulk(WT_CURSOR_LSM *clsm, const char *cfg[])
{
    WT_CURSOR *cursor, *bulk_cursor;
    WT_DECL_RET;
    WT_LSM_TREE *lsm_tree;
    WT_SESSION_IMPL *session;

    bulk_cursor = NULL;
    cursor = &clsm->iface;
    lsm_tree = clsm->lsm_tree;
    session = CUR2S(clsm);

    F_SET(clsm, WT_CLSM_BULK);

    /* Bulk cursors are limited to insert and close. */
    __wt_cursor_set_notsup(cursor);
    cursor->insert = __clsm_insert_bulk;
    cursor->close = __clsm_close_bulk;

    /*
     * Setup the first chunk in the tree. This is the only time we switch without using the LSM
     * worker threads, it's safe to do here since we have an exclusive lock on the LSM tree. We need
     * to do this switch inline, since switch needs a schema lock and online index creation opens a
     * bulk cursor while holding the schema lock.
     */
    WT_WITH_SCHEMA_LOCK(session, ret = __wti_lsm_tree_switch(session, lsm_tree));
    WT_RET(ret);

    /*
     * Open a bulk cursor on the first chunk, it's not a regular LSM chunk cursor, but use the
     * standard storage locations. Allocate the space for a bloom filter - it makes cleanup simpler.
     * Cleaned up by cursor close on error.
     */
    WT_RET(__wt_realloc_def(session, &clsm->chunks_alloc, 1, &clsm->chunks));
    WT_RET(__wt_calloc_one(session, &clsm->chunks[0]));
    clsm->chunks_count = clsm->nchunks = 1;

    /*
     * Open a bulk cursor on the first chunk in the tree - take a read lock on the LSM tree while we
     * are opening the chunk, to ensure that the first chunk has been fully created before we
     * succeed. Pass through the application config to ensure the tree is open for bulk access.
     */
    WT_RET(__wt_open_cursor(session, lsm_tree->chunk[0]->uri, &clsm->iface, cfg, &bulk_cursor));
    clsm->chunks[0]->cursor = bulk_cursor;
    /* LSM cursors are always raw */
    F_SET(bulk_cursor, WT_CURSTD_RAW);

    WT_STAT_CONN_INCR_ATOMIC(session, cursor_bulk_count);

    return (0);
}
