Skip to content

Commit

Permalink
Implement failure points (#218)
Browse files Browse the repository at this point in the history
This patch adds API to trigger execution of C code from SQL to test
corner cases.

`test/sql/hnsw_failure_point.sql` has an example of how to trigger a
process crash using failure points and how to see that a space leak
happens if a crash happens after a block is allocated, but before a
record for the block is added to the index during blockmaps creation.

* src/hnsw/failure_point: fix use-after-free bug when strings are deallocated at the end of a query

* src/hnsw/failure_point: elog(INFO, ...) when a failure point is enabled
  • Loading branch information
medvied authored Nov 3, 2023
1 parent 69a9e3d commit 09c3dfe
Show file tree
Hide file tree
Showing 10 changed files with 247 additions and 3 deletions.
8 changes: 8 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ option(BUILD_WITH_USEARCH "Build with usearch as hnsw provider" ON)
option(BUILD_LIBHNSW "Build libhnsw as hnsw provider" OFF)
option(CODECOVERAGE "Enable code coverage for the build" OFF)
option(BENCH "Enable benchmarking" OFF)
option(FAILURE_POINTS "Enable failure points" ON)

if(CODECOVERAGE)
message(STATUS "Code coverage is enabled.")
Expand Down Expand Up @@ -149,6 +150,13 @@ if (${BUILD_WITH_LIBHNSW})
target_link_libraries(lantern PRIVATE hnsw)
target_compile_definitions(lantern PRIVATE LANTERN_USE_LIBHNSW)
endif()
if (FAILURE_POINTS)
message(STATUS "Failure points are enabled.")
target_compile_definitions(lantern PRIVATE LANTERN_FAILURE_POINTS_ARE_ENABLED=1)
else()
message(STATUS "Failure points are disabled.")
target_compile_definitions(lantern PRIVATE LANTERN_FAILURE_POINTS_ARE_ENABLED=0)
endif()
if (${LANTERNDB_COPYNODES})
target_compile_definitions(lantern PRIVATE LANTERNDB_COPYNODES)
endif()
Expand Down
3 changes: 3 additions & 0 deletions sql/lantern.sql
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,9 @@ CREATE SCHEMA _lantern_internal;
CREATE FUNCTION _lantern_internal.validate_index(index regclass, print_info boolean DEFAULT true) RETURNS VOID
AS 'MODULE_PATHNAME', 'lantern_internal_validate_index' LANGUAGE C STABLE STRICT PARALLEL UNSAFE;

CREATE FUNCTION _lantern_internal.failure_point_enable(func TEXT, name TEXT, dont_trigger_first_nr INTEGER DEFAULT 0) RETURNS VOID
AS 'MODULE_PATHNAME', 'lantern_internal_failure_point_enable' LANGUAGE C STABLE STRICT PARALLEL UNSAFE;

-- operator classes
CREATE OR REPLACE FUNCTION _lantern_internal._create_ldb_operator_classes(access_method_name TEXT) RETURNS BOOLEAN AS $$
DECLARE
Expand Down
13 changes: 13 additions & 0 deletions src/hnsw.c
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,15 @@
#include <commands/vacuum.h>
#include <float.h>
#include <math.h>
#include <utils/builtins.h>
#include <utils/guc.h>
#include <utils/lsyscache.h>
#include <utils/selfuncs.h>
#include <utils/spccache.h>

#include "hnsw/build.h"
#include "hnsw/delete.h"
#include "hnsw/failure_point.h"
#include "hnsw/insert.h"
#include "hnsw/options.h"
#include "hnsw/scan.h"
Expand Down Expand Up @@ -369,6 +371,17 @@ Datum lantern_internal_validate_index(PG_FUNCTION_ARGS)
PG_RETURN_VOID();
}

PGDLLEXPORT PG_FUNCTION_INFO_V1(lantern_internal_failure_point_enable);
Datum lantern_internal_failure_point_enable(PG_FUNCTION_ARGS)
{
const char *func = text_to_cstring(PG_GETARG_TEXT_PP(0));
const char *name = text_to_cstring(PG_GETARG_TEXT_PP(1));
uint32 dont_trigger_first_nr = PG_GETARG_UINT32(2);

ldb_failure_point_enable(func, name, dont_trigger_first_nr);
PG_RETURN_VOID();
}

/*
* Get data type for give oid
* */
Expand Down
3 changes: 3 additions & 0 deletions src/hnsw/external_index.c
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
#include <utils/relcache.h>

#include "extra_dirtied.h"
#include "failure_point.h"
#include "htab_cache.h"
#include "insert.h"
#include "options.h"
Expand Down Expand Up @@ -67,6 +68,8 @@ int CreateBlockMapGroup(
Buffer buf = ReadBufferExtended(index, forkNum, P_NEW, RBM_NORMAL, NULL);
LockBuffer(buf, BUFFER_LOCK_EXCLUSIVE);

LDB_FAILURE_POINT_CRASH_IF_ENABLED("crash_after_buf_allocation");

if(blockmap_id == 0) {
hdr->blockmap_page_groups = blockmap_groupno;
hdr->blockmap_page_group_index[ blockmap_groupno ] = BufferGetBlockNumber(buf);
Expand Down
90 changes: 90 additions & 0 deletions src/hnsw/failure_point.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
#include <postgres.h>

#include "hnsw/failure_point.h"

#include <inttypes.h> /* PRIu32 */

struct failure_point_state
{
bool enabled;
char func[ 0x100 ];
char name[ 0x100 ];
uint32 remaining;
};

static struct failure_point_state *failure_point_get_state(void)
{
static struct failure_point_state state = {};

return &state;
}

void ldb_failure_point_enable(const char *func, const char *name, uint32 dont_trigger_first_nr)
{
struct failure_point_state *state = failure_point_get_state();

if(!LANTERN_FAILURE_POINTS_ARE_ENABLED) {
elog(WARNING,
"Can't enable failure point for (func=%s name=%s), "
"because failure points are disabled in compile time.",
func,
name);
}
if(state->enabled) {
elog(WARNING,
"ldb_failure_point_enable(): another failure point is enabled already."
" old failure point: func=%s name=%s remaining=%" PRIu32
" new failure point: func=%s name=%s dont_trigger_first_nr=%" PRIu32,
state->func,
state->name,
state->remaining,
func,
name,
dont_trigger_first_nr);
}
if(strlen(func) >= lengthof(state->func)) {
elog(ERROR,
"failure point function name is too large: "
"func=%s strlen(func)=%zu lengthof(state->func)=%zu",
func,
strlen(func),
lengthof(state->func));
}
if(strlen(name) >= lengthof(state->name)) {
elog(ERROR,
"failure point name is too large: "
"name=%s strlen(name)=%zu lengthof(state->name)=%zu",
name,
strlen(name),
lengthof(state->name));
}
state->enabled = true;
state->remaining = dont_trigger_first_nr;
strncpy(state->func, func, lengthof(state->func));
strncpy(state->name, name, lengthof(state->name));
elog(INFO, "Failure point (func=%s name=%s) is enabled.", state->func, state->name);
}

bool ldb_failure_point_is_enabled(const char *func, const char *name)
{
struct failure_point_state *state = failure_point_get_state();

if(!LANTERN_FAILURE_POINTS_ARE_ENABLED) return false;
if(!state->enabled) return false;
if(strcmp(func, state->func) == 0 && strcmp(name, state->name) == 0) {
if(state->remaining == 0) {
state->enabled = false;
elog(INFO, "Failure point (func=%s name=%s) has been triggered.", state->func, state->name);
return true;
} else {
--state->remaining;
}
}
return false;
}

void ldb_failure_point_crash(void)
{
elog(ERROR, "ldb_failure_point_crash()");
pg_unreachable();
}
47 changes: 47 additions & 0 deletions src/hnsw/failure_point.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#ifndef LDB_HNSW_FAILURE_POINT_H
#define LDB_HNSW_FAILURE_POINT_H

/*
* Failure points implementation.
*
* An example on how to use from test/sql/hnsw_failure_point.sql.
*
* 1) Add this to CreateBlockMapGroup():
*
LDB_FAILURE_POINT_CRASH_IF_ENABLED("crash_after_buf_allocation");
*
* 2) Enable the failure point somewhere in the test:
*
* SELECT _lantern_internal.failure_point_enable('CreateBlockMapGroup', 'crash_after_buf_allocation', 0);
*
* 3) Trigger the failure point, the output looks like this:
*
* INFO: Failure point (func=CreateBlockMapGroup name=crash_after_buf_allocation) has been triggered.
*
* 4) Now check that the failure actually happens, for example with validate_index():
*
* SELECT _lantern_internal.validate_index('small_world_v_idx', false);
*
* 5) The output tells that the block is allocated, but it's not being used:
*
* INFO: validate_index() start for small_world_v_idx
* ERROR: vi_blocks[48].vp_type == LDB_VI_BLOCK_UNKNOWN (but it should be known now)
*
*
* Limitations
*
* 1) A single static per-process variable holds the state.
* 2) Only one failure point active at a time is supported.
* 3) The API is not thread-safe.
*/

#define LDB_FAILURE_POINT_IS_ENABLED(_name) \
(LANTERN_FAILURE_POINTS_ARE_ENABLED && ldb_failure_point_is_enabled(__func__, (_name)))
#define LDB_FAILURE_POINT_CRASH_IF_ENABLED(_name) \
if(LDB_FAILURE_POINT_IS_ENABLED(_name)) ldb_failure_point_crash()

void ldb_failure_point_enable(const char *func, const char *name, uint32 dont_trigger_first_nr);
bool ldb_failure_point_is_enabled(const char *func, const char *name);
void ldb_failure_point_crash(void);

#endif // LDB_HNSW_FAILURE_POINT_H
5 changes: 3 additions & 2 deletions test/expected/ext_relocation.out
Original file line number Diff line number Diff line change
Expand Up @@ -35,14 +35,15 @@ ORDER BY 1, 3;
extschema | proname | proschema
-----------+------------------------------+-------------------
schema1 | validate_index | _lantern_internal
schema1 | failure_point_enable | _lantern_internal
schema1 | _create_ldb_operator_classes | _lantern_internal
schema1 | ldb_generic_dist | schema1
schema1 | l2sq_dist | schema1
schema1 | hnsw_handler | schema1
schema1 | hamming_dist | schema1
schema1 | cos_dist | schema1
schema1 | ldb_generic_dist | schema1
(8 rows)
schema1 | ldb_generic_dist | schema1
(9 rows)

-- show all the extension operators
SELECT ne.nspname AS extschema, op.oprname, np.nspname AS proschema
Expand Down
46 changes: 46 additions & 0 deletions test/expected/hnsw_failure_point.out
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
------------------------------
-- Test HNSW failure points --
------------------------------
CREATE TABLE small_world (
id SERIAL PRIMARY KEY,
v REAL[2]
);
CREATE INDEX ON small_world USING hnsw (v) WITH (dim=3);
INFO: done init usearch index
INFO: inserted 0 elements
INFO: done saving 0 vectors
-- let's insert HNSW_BLOCKMAP_BLOCKS_PER_PAGE (2000) record to fill the first blockmap page
do $$
BEGIN
FOR i IN 1..2000 LOOP
INSERT INTO small_world (v) VALUES (array_replace(ARRAY[0,0,-1], -1, i));
END LOOP;
END $$;
-- everything is fine, the index is valid
SELECT _lantern_internal.validate_index('small_world_v_idx', false);
INFO: validate_index() start for small_world_v_idx
INFO: validate_index() done, no issues found.
validate_index
----------------

(1 row)

-- now let's crash after a buffer for a blockmap is allocated during insert,
-- but it hasn't been recorded yet
SELECT _lantern_internal.failure_point_enable('CreateBlockMapGroup', 'crash_after_buf_allocation');
INFO: Failure point (func=CreateBlockMapGroup name=crash_after_buf_allocation) is enabled.
failure_point_enable
----------------------

(1 row)

-- here is the insert where the crash happens
\set ON_ERROR_STOP off
INSERT INTO small_world (v) VALUES ('{2,2,2}');
INFO: Failure point (func=CreateBlockMapGroup name=crash_after_buf_allocation) has been triggered.
ERROR: ldb_failure_point_crash()
\set ON_ERROR_STOP on
-- now we see that the index has an extra free page, so the index validation fails
SELECT _lantern_internal.validate_index('small_world_v_idx', false);
INFO: validate_index() start for small_world_v_idx
ERROR: vi_blocks[48].vp_type == LDB_VI_BLOCK_UNKNOWN (but it should be known now)
2 changes: 1 addition & 1 deletion test/schedule.txt
Original file line number Diff line number Diff line change
Expand Up @@ -4,4 +4,4 @@
# - 'test' lines may have multiple space-separated tests. All tests in a single 'test' line will be run in parallel

test_pgvector: hnsw_vector
test: hnsw_config hnsw_correct hnsw_create hnsw_create_expr hnsw_dist_func hnsw_insert hnsw_select hnsw_todo hnsw_index_from_file hnsw_cost_estimate ext_relocation hnsw_ef_search
test: hnsw_config hnsw_correct hnsw_create hnsw_create_expr hnsw_dist_func hnsw_insert hnsw_select hnsw_todo hnsw_index_from_file hnsw_cost_estimate ext_relocation hnsw_ef_search hnsw_failure_point
33 changes: 33 additions & 0 deletions test/sql/hnsw_failure_point.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
------------------------------
-- Test HNSW failure points --
------------------------------

CREATE TABLE small_world (
id SERIAL PRIMARY KEY,
v REAL[2]
);
CREATE INDEX ON small_world USING hnsw (v) WITH (dim=3);

-- let's insert HNSW_BLOCKMAP_BLOCKS_PER_PAGE (2000) record to fill the first blockmap page

do $$
BEGIN
FOR i IN 1..2000 LOOP
INSERT INTO small_world (v) VALUES (array_replace(ARRAY[0,0,-1], -1, i));
END LOOP;
END $$;

-- everything is fine, the index is valid
SELECT _lantern_internal.validate_index('small_world_v_idx', false);

-- now let's crash after a buffer for a blockmap is allocated during insert,
-- but it hasn't been recorded yet
SELECT _lantern_internal.failure_point_enable('CreateBlockMapGroup', 'crash_after_buf_allocation');

-- here is the insert where the crash happens
\set ON_ERROR_STOP off
INSERT INTO small_world (v) VALUES ('{2,2,2}');
\set ON_ERROR_STOP on

-- now we see that the index has an extra free page, so the index validation fails
SELECT _lantern_internal.validate_index('small_world_v_idx', false);

0 comments on commit 09c3dfe

Please sign in to comment.