// update operation codes. these codes get stuffed into update messages, so they can not change. enum { UPDATE_OP_COL_ADD_OR_DROP = 0, UPDATE_OP_EXPAND_VARCHAR_OFFSETS = 1, UPDATE_OP_EXPAND_INT = 2, UPDATE_OP_EXPAND_CHAR = 3, UPDATE_OP_EXPAND_BINARY = 4, UPDATE_OP_ADD_INT = 5, UPDATE_OP_SUB_INT = 6, }; #define UP_COL_ADD_OR_DROP UPDATE_OP_COL_ADD_OR_DROP // add or drop column sub-operations #define COL_DROP 0xaa #define COL_ADD 0xbb // add or drop column types #define COL_FIXED 0xcc #define COL_VAR 0xdd #define COL_BLOB 0xee #define STATIC_ROW_MUTATOR_SIZE 1+8+2+8+8+8 // how much space do I need for the mutators? // static stuff first: // operation 1 == UP_COL_ADD_OR_DROP // 8 - old null, new null // 2 - old num_offset, new num_offset // 8 - old fixed_field size, new fixed_field_size // 8 - old and new length of offsets // 8 - old and new starting null bit position // TOTAL: 27 // dynamic stuff: // 4 - number of columns // for each column: // 1 - add or drop // 1 - is nullable // 4 - if nullable, position // 1 - if add, whether default is null or not // 1 - if fixed, var, or not // for fixed, entire default // for var, 4 bytes length, then entire default // for blob, nothing // So, an upperbound is 4 + num_fields(12) + all default stuff // static blob stuff: // 4 - num blobs // 1 byte for each num blobs in old table // So, an upperbound is 4 + kc_info->num_blobs // dynamic blob stuff: // for each blob added: // 1 - state if we are adding or dropping // 4 - blob index // if add, 1 len bytes // at most, 4 0's // So, upperbound is num_blobs(1+4+1+4) = num_columns*10 // operation 1 == UPDATE_OP_EXPAND_VARCHAR_OFFSETS // offset_start 4 starting offset of the variable length field offsets // offset end 4 ending offset of the variable length field offsets // operation 1 == UPDATE_OP_EXPAND_INT, UPDATE_OP_EXPAND_CHAR, UPDATE_OP_EXPAND_BINARY // old offset 4 // old length 4 // new offset 4 // new length 4 // if operation == UPDATE_OP_EXPAND_INT // is unsigned 1 // operation 1 == UPDATE_OP_INT_ADD or UPDATE_OP_INT_SUB // offset 4 starting offset of the int type field // length 4 length of the int type field // value 4 value to add or subtract (common use case is increment or decrement by 1) // is unsigned 1 // // checks whether the bit at index pos in data is set or not // static inline bool is_overall_null_position_set(uchar* data, uint32_t pos) { uint32_t offset = pos/8; uchar remainder = pos%8; uchar null_bit = 1<data; extra_pos = (uchar *)extra->data; operation = extra_pos[0]; extra_pos++; assert(operation == UP_COL_ADD_OR_DROP); memcpy(&old_num_null_bytes, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&new_num_null_bytes, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); old_num_offset_bytes = extra_pos[0]; extra_pos++; new_num_offset_bytes = extra_pos[0]; extra_pos++; memcpy(&old_fixed_field_size, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&new_fixed_field_size, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&old_len_of_offsets, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&new_len_of_offsets, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); max_num_bytes = old_val->size + extra->size + new_len_of_offsets + new_fixed_field_size; new_val_data = (uchar *)my_malloc( max_num_bytes, MYF(MY_FAE) ); if (new_val_data == NULL) { goto cleanup; } old_fixed_field_ptr = (uchar *) old_val->data; old_fixed_field_ptr += old_num_null_bytes; new_fixed_field_ptr = new_val_data + new_num_null_bytes; curr_old_fixed_offset = 0; curr_new_fixed_offset = 0; old_num_var_fields = old_len_of_offsets/old_num_offset_bytes; // new_num_var_fields = new_len_of_offsets/new_num_offset_bytes; // following fields will change as we write the variable data old_var_field_offset_ptr = old_fixed_field_ptr + old_fixed_field_size; new_var_field_offset_ptr = new_fixed_field_ptr + new_fixed_field_size; old_var_field_data_ptr = old_var_field_offset_ptr + old_len_of_offsets; new_var_field_data_ptr = new_var_field_offset_ptr + new_len_of_offsets; curr_new_var_field_offset_ptr = new_var_field_offset_ptr; curr_new_var_field_data_ptr = new_var_field_data_ptr; curr_old_num_var_field = 0; curr_new_num_var_field = 0; old_null_bytes = (uchar *)old_val->data; new_null_bytes = new_val_data; memcpy(&curr_old_null_pos, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&curr_new_null_pos, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&num_columns, extra_pos, sizeof(num_columns)); extra_pos += sizeof(num_columns); // // now go through and apply the change into new_val_data // for (uint32_t i = 0; i < num_columns; i++) { uchar op_type = extra_pos[0]; bool is_null_default = false; extra_pos++; assert(op_type == COL_DROP || op_type == COL_ADD); bool nullable = (extra_pos[0] != 0); extra_pos++; if (nullable) { uint32_t null_bit_position; memcpy(&null_bit_position, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); uint32_t num_bits; if (op_type == COL_DROP) { assert(curr_old_null_pos <= null_bit_position); num_bits = null_bit_position - curr_old_null_pos; } else { assert(curr_new_null_pos <= null_bit_position); num_bits = null_bit_position - curr_new_null_pos; } copy_null_bits( curr_old_null_pos, curr_new_null_pos, num_bits, old_null_bytes, new_null_bytes ); // update the positions curr_new_null_pos += num_bits; curr_old_null_pos += num_bits; if (op_type == COL_DROP) { curr_old_null_pos++; // account for dropped column } else { is_null_default = (extra_pos[0] != 0); extra_pos++; set_overall_null_position( new_null_bytes, null_bit_position, is_null_default ); curr_new_null_pos++; //account for added column } } uchar col_type = extra_pos[0]; extra_pos++; if (col_type == COL_FIXED) { uint32_t col_offset; uint32_t col_size; uint32_t num_bytes_to_copy; memcpy(&col_offset, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); memcpy(&col_size, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); if (op_type == COL_DROP) { num_bytes_to_copy = col_offset - curr_old_fixed_offset; } else { num_bytes_to_copy = col_offset - curr_new_fixed_offset; } memcpy( new_fixed_field_ptr + curr_new_fixed_offset, old_fixed_field_ptr + curr_old_fixed_offset, num_bytes_to_copy ); curr_old_fixed_offset += num_bytes_to_copy; curr_new_fixed_offset += num_bytes_to_copy; if (op_type == COL_DROP) { // move old_fixed_offset val to skip OVER column that is being dropped curr_old_fixed_offset += col_size; } else { if (is_null_default) { // copy zeroes memset(new_fixed_field_ptr + curr_new_fixed_offset, 0, col_size); } else { // copy data from extra_pos into new row memcpy( new_fixed_field_ptr + curr_new_fixed_offset, extra_pos, col_size ); extra_pos += col_size; } curr_new_fixed_offset += col_size; } } else if (col_type == COL_VAR) { uint32_t var_col_index; memcpy(&var_col_index, extra_pos, sizeof(uint32_t)); extra_pos += sizeof(uint32_t); if (op_type == COL_DROP) { num_var_fields_to_copy = var_col_index - curr_old_num_var_field; } else { num_var_fields_to_copy = var_col_index - curr_new_num_var_field; } copy_var_fields( curr_old_num_var_field, num_var_fields_to_copy, old_var_field_offset_ptr, old_num_offset_bytes, curr_new_var_field_data_ptr, curr_new_var_field_offset_ptr, new_var_field_data_ptr, // pointer to beginning of var fields in new row old_var_field_data_ptr, // pointer to beginning of var fields in old row new_num_offset_bytes, // number of offset bytes used in new row &num_data_bytes_written, &num_offset_bytes_written ); curr_new_var_field_data_ptr += num_data_bytes_written; curr_new_var_field_offset_ptr += num_offset_bytes_written; curr_new_num_var_field += num_var_fields_to_copy; curr_old_num_var_field += num_var_fields_to_copy; if (op_type == COL_DROP) { curr_old_num_var_field++; // skip over dropped field } else { if (is_null_default) { curr_new_var_field_data_ptr = write_var_field( curr_new_var_field_offset_ptr, curr_new_var_field_data_ptr, new_var_field_data_ptr, NULL, //copying no data 0, //copying 0 bytes new_num_offset_bytes ); curr_new_var_field_offset_ptr += new_num_offset_bytes; } else { uint32_t data_length; memcpy(&data_length, extra_pos, sizeof(data_length)); extra_pos += sizeof(data_length); curr_new_var_field_data_ptr = write_var_field( curr_new_var_field_offset_ptr, curr_new_var_field_data_ptr, new_var_field_data_ptr, extra_pos, //copying data from mutator data_length, //copying data_length bytes new_num_offset_bytes ); extra_pos += data_length; curr_new_var_field_offset_ptr += new_num_offset_bytes; } curr_new_num_var_field++; //account for added column } } else if (col_type == COL_BLOB) { // handle blob data later continue; } else { assert(false); } } // finish copying the null stuff old_null_bits_left = 8*old_num_null_bytes - curr_old_null_pos; new_null_bits_left = 8*new_num_null_bytes - curr_new_null_pos; overall_null_bits_left = old_null_bits_left; set_if_smaller(overall_null_bits_left, new_null_bits_left); copy_null_bits( curr_old_null_pos, curr_new_null_pos, overall_null_bits_left, old_null_bytes, new_null_bytes ); // finish copying fixed field stuff num_bytes_left = old_fixed_field_size - curr_old_fixed_offset; memcpy( new_fixed_field_ptr + curr_new_fixed_offset, old_fixed_field_ptr + curr_old_fixed_offset, num_bytes_left ); curr_old_fixed_offset += num_bytes_left; curr_new_fixed_offset += num_bytes_left; // sanity check assert(curr_new_fixed_offset == new_fixed_field_size); // finish copying var field stuff num_var_fields_to_copy = old_num_var_fields - curr_old_num_var_field; copy_var_fields( curr_old_num_var_field, num_var_fields_to_copy, old_var_field_offset_ptr, old_num_offset_bytes, curr_new_var_field_data_ptr, curr_new_var_field_offset_ptr, new_var_field_data_ptr, // pointer to beginning of var fields in new row old_var_field_data_ptr, // pointer to beginning of var fields in old row new_num_offset_bytes, // number of offset bytes used in new row &num_data_bytes_written, &num_offset_bytes_written ); curr_new_var_field_offset_ptr += num_offset_bytes_written; curr_new_var_field_data_ptr += num_data_bytes_written; // sanity check assert(curr_new_var_field_offset_ptr == new_var_field_data_ptr); // start handling blobs get_blob_field_info( &start_blob_offset, old_len_of_offsets, old_var_field_data_ptr, old_num_offset_bytes ); start_blob_ptr = old_var_field_data_ptr + start_blob_offset; // if nothing else in extra, then there are no blobs to add or drop, so can copy blobs straight if ((extra_pos - extra_pos_start) == extra->size) { num_blob_bytes = old_val->size - (start_blob_ptr - old_null_bytes); memcpy(curr_new_var_field_data_ptr, start_blob_ptr, num_blob_bytes); curr_new_var_field_data_ptr += num_blob_bytes; } // else, there is blob information to process else { uchar* len_bytes = NULL; uint32_t curr_old_blob = 0; uint32_t curr_new_blob = 0; uint32_t num_old_blobs = 0; uchar* curr_old_blob_ptr = start_blob_ptr; memcpy(&num_old_blobs, extra_pos, sizeof(num_old_blobs)); extra_pos += sizeof(num_old_blobs); len_bytes = extra_pos; extra_pos += num_old_blobs; // copy over blob fields one by one while ((extra_pos - extra_pos_start) < extra->size) { uchar op_type = extra_pos[0]; extra_pos++; uint32_t num_blobs_to_copy = 0; uint32_t blob_index; memcpy(&blob_index, extra_pos, sizeof(blob_index)); extra_pos += sizeof(blob_index); assert (op_type == COL_DROP || op_type == COL_ADD); if (op_type == COL_DROP) { num_blobs_to_copy = blob_index - curr_old_blob; } else { num_blobs_to_copy = blob_index - curr_new_blob; } for (uint32_t i = 0; i < num_blobs_to_copy; i++) { uint32_t num_bytes_written = copy_toku_blob( curr_new_var_field_data_ptr, curr_old_blob_ptr, len_bytes[curr_old_blob + i], false ); curr_old_blob_ptr += num_bytes_written; curr_new_var_field_data_ptr += num_bytes_written; } curr_old_blob += num_blobs_to_copy; curr_new_blob += num_blobs_to_copy; if (op_type == COL_DROP) { // skip over blob in row uint32_t num_bytes = copy_toku_blob( NULL, curr_old_blob_ptr, len_bytes[curr_old_blob], true ); curr_old_blob++; curr_old_blob_ptr += num_bytes; } else { // copy new data uint32_t new_len_bytes = extra_pos[0]; extra_pos++; uint32_t num_bytes = copy_toku_blob( curr_new_var_field_data_ptr, extra_pos, new_len_bytes, false ); curr_new_blob++; curr_new_var_field_data_ptr += num_bytes; extra_pos += num_bytes; } } num_blob_bytes = old_val->size - (curr_old_blob_ptr - old_null_bytes); memcpy(curr_new_var_field_data_ptr, curr_old_blob_ptr, num_blob_bytes); curr_new_var_field_data_ptr += num_blob_bytes; } new_val.data = new_val_data; new_val.size = curr_new_var_field_data_ptr - new_val_data; set_val(&new_val, set_extra); error = 0; cleanup: my_free(new_val_data, MYF(MY_ALLOW_ZERO_PTR)); return error; } // Decode the expand varchar offsets message and expand the varchar offsets array in the old // val to a new val. Call the set_val callback with the new val. static int tokudb_expand_varchar_offsets( DB* db, const DBT *key, const DBT *old_val, const DBT *extra, void (*set_val)(const DBT *new_val, void *set_extra), void *set_extra ) { int error = 0; uchar *extra_pos = (uchar *)extra->data; // decode the operation uchar operation = extra_pos[0]; assert(operation == UPDATE_OP_EXPAND_VARCHAR_OFFSETS); extra_pos += sizeof operation; // decode the offset start uint32_t offset_start; memcpy(&offset_start, extra_pos, sizeof offset_start); extra_pos += sizeof offset_start; // decode the offset end uint32_t offset_end; memcpy(&offset_end, extra_pos, sizeof offset_end); extra_pos += sizeof offset_end; // number of variable fields is the diff of the offset end and start uint32_t number_of_offsets = offset_end - offset_start; assert(extra_pos == (uchar *)extra->data + extra->size); assert(offset_start < old_val->size); assert(offset_start + number_of_offsets < old_val->size); DBT new_val = {}; if (old_val != NULL) { // compute the new val from the old val uchar *old_val_ptr = (uchar *)old_val->data; // allocate space for the new val's data uchar *new_val_ptr = (uchar *)my_malloc(number_of_offsets + old_val->size, MYF(MY_FAE)); if (!new_val_ptr) { error = ENOMEM; goto cleanup; } new_val.data = new_val_ptr; // copy up to the start of the varchar offset memcpy(new_val_ptr, old_val_ptr, offset_start); new_val_ptr += offset_start; old_val_ptr += offset_start; // expand each offset from 1 to 2 bytes for (uint32_t i = 0; i < number_of_offsets; i++) { uint16_t new_offset = *old_val_ptr; int2store(new_val_ptr, new_offset); new_val_ptr += 2; old_val_ptr += 1; } // copy the rest of the row size_t n = old_val->size - (old_val_ptr - (uchar *)old_val->data); memcpy(new_val_ptr, old_val_ptr, n); new_val_ptr += n; old_val_ptr += n; new_val.size = new_val_ptr - (uchar *)new_val.data; assert(new_val_ptr == (uchar *)new_val.data + new_val.size); assert(old_val_ptr == (uchar *)old_val->data + old_val->size); // set the new val set_val(&new_val, set_extra); } error = 0; cleanup: my_free(new_val.data, MYF(MY_ALLOW_ZERO_PTR)); return error; } // Given a description of a fixed length field and the old value of a row, build a new value for the row and // update it in the fractal tree. static int tokudb_expand_field( DB* db, const DBT *key, const DBT *old_val, const DBT *extra, void (*set_val)(const DBT *new_val, void *set_extra), void *set_extra ) { int error = 0; uchar *extra_pos = (uchar *)extra->data; uchar operation = extra_pos[0]; assert(operation == UPDATE_OP_EXPAND_INT || operation == UPDATE_OP_EXPAND_CHAR || operation == UPDATE_OP_EXPAND_BINARY); extra_pos += sizeof operation; uint32_t old_offset; memcpy(&old_offset, extra_pos, sizeof old_offset); extra_pos += sizeof old_offset; uint32_t old_length; memcpy(&old_length, extra_pos, sizeof old_length); extra_pos += sizeof old_length; uint32_t new_offset; memcpy(&new_offset, extra_pos, sizeof new_offset); extra_pos += sizeof new_offset; uint32_t new_length; memcpy(&new_length, extra_pos, sizeof new_length); extra_pos += sizeof new_length; uchar is_unsigned; // for int expansion switch (operation) { case UPDATE_OP_EXPAND_INT: is_unsigned = extra_pos[0]; extra_pos += sizeof is_unsigned; assert(is_unsigned == 0 || is_unsigned == 1); break; case UPDATE_OP_EXPAND_CHAR: case UPDATE_OP_EXPAND_BINARY: break; default: assert(0); } assert(extra_pos == (uchar *)extra->data + extra->size); // consumed the entire message assert(old_offset == new_offset); // only expand one field per update, so the offset must be the same assert(new_length >= old_length); // expand only assert(old_offset + old_length <= old_val->size); // old field within the old val DBT new_val = {}; if (old_val != NULL) { // compute the new val from the old val uchar *old_val_ptr = (uchar *)old_val->data; // allocate space for the new val's data uchar *new_val_ptr = (uchar *)my_malloc(old_val->size + (new_length - old_length), MYF(MY_FAE)); if (!new_val_ptr) { error = ENOMEM; goto cleanup; } new_val.data = new_val_ptr; // copy up to the old offset memcpy(new_val_ptr, old_val_ptr, old_offset); new_val_ptr += new_offset; old_val_ptr += old_offset; // read the old field, expand it, write to the new offset switch (operation) { case UPDATE_OP_EXPAND_INT: if (is_unsigned) { memset(new_val_ptr, 0, new_length); } else { if (old_val_ptr[old_length-1] & 0x80) // sign bit on? memset(new_val_ptr, 0xff, new_length); // sign extend else memset(new_val_ptr, 0, new_length); } memcpy(new_val_ptr, old_val_ptr, old_length); new_val_ptr += new_length; old_val_ptr += old_length; break; case UPDATE_OP_EXPAND_CHAR: memset(new_val_ptr, ' ', new_length); // expand the field with the char pad memcpy(new_val_ptr, old_val_ptr, old_length); new_val_ptr += new_length; old_val_ptr += old_length; break; case UPDATE_OP_EXPAND_BINARY: memset(new_val_ptr, 0, new_length); // expand the field with the binary pad memcpy(new_val_ptr, old_val_ptr, old_length); new_val_ptr += new_length; old_val_ptr += old_length; break; default: assert(0); } // copy the rest size_t n = old_val->size - (old_val_ptr - (uchar *)old_val->data); memcpy(new_val_ptr, old_val_ptr, n); new_val_ptr += n; old_val_ptr += n; new_val.size = new_val_ptr - (uchar *)new_val.data; assert(new_val_ptr == (uchar *)new_val.data + new_val.size); assert(old_val_ptr == (uchar *)old_val->data + old_val->size); // set the new val set_val(&new_val, set_extra); } error = 0; cleanup: my_free(new_val.data, MYF(MY_ALLOW_ZERO_PTR)); return error; } int tokudb_update_fun( DB* db, const DBT *key, const DBT *old_val, const DBT *extra, void (*set_val)(const DBT *new_val, void *set_extra), void *set_extra ) { uchar *extra_pos = (uchar *)extra->data; uchar operation = extra_pos[0]; int error = 0; switch (operation) { case UPDATE_OP_COL_ADD_OR_DROP: error = tokudb_hcad_update_fun(db, key, old_val, extra, set_val, set_extra); break; case UPDATE_OP_EXPAND_VARCHAR_OFFSETS: error = tokudb_expand_varchar_offsets(db, key, old_val, extra, set_val, set_extra); break; case UPDATE_OP_EXPAND_INT: case UPDATE_OP_EXPAND_CHAR: case UPDATE_OP_EXPAND_BINARY: error = tokudb_expand_field(db, key, old_val, extra, set_val, set_extra); break; default: error = EINVAL; break; } return error; }