Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[memory] Fix all deadlocks, implement monitors #316

Merged
merged 42 commits into from
Nov 28, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
0350383
[memory] Remove section locks
NikitaZotov Aug 18, 2023
6e89213
[memory] Remove check levels from iterators
NikitaZotov Aug 20, 2023
de40b3c
[memory][segment] Remake segments caching and creating
NikitaZotov Sep 10, 2023
bb5b7de
[memory] Remake processes sync flow
NikitaZotov Sep 15, 2023
196e79a
[memory] Implement monitors for reading and writing
NikitaZotov Sep 17, 2023
6a28551
[memory] Setup monitors in storages
NikitaZotov Sep 17, 2023
5e49828
[memory][tests] Fix tests
NikitaZotov Sep 17, 2023
f5f4ce6
[memory][fs-storage] Implements monitors
NikitaZotov Sep 17, 2023
a52de99
[memory][fs-memory] Fix base64 encode
NikitaZotov Sep 17, 2023
6e2bcd4
[memory][events] Inject monitors instead muteses
NikitaZotov Sep 18, 2023
1f39cbb
[memory] Reuse freed segment elements
NikitaZotov Sep 22, 2023
c82bddc
[memory] Implements monitors for addrs group
NikitaZotov Sep 23, 2023
9abf97e
[memory][monitor] Fix processes starvations
NikitaZotov Sep 28, 2023
a2383d3
[memory] Redefine glib structures
NikitaZotov Sep 28, 2023
da25ccb
[memory] Add monitors for dump and save memory
NikitaZotov Sep 29, 2023
c71dbbd
[memory][monitor][docs] Write API docs
NikitaZotov Sep 29, 2023
3949ae4
[memory] Optimized released elements usage
NikitaZotov Oct 15, 2023
93ca183
[memory][fs-memory] Sync save, load and dump statistics
NikitaZotov Oct 15, 2023
6a4b7e1
[memory] Process to segment handling
NikitaZotov Oct 15, 2023
e57f39f
[memory] Save and load released segments and elements
NikitaZotov Oct 15, 2023
b6da6f9
[memory][fs-memory] Fix sync in sc-dictionary
NikitaZotov Oct 15, 2023
2d63dbb
[memory] Optimize segment parallel handling
NikitaZotov Oct 16, 2023
ec02c36
[memory] Process management
NikitaZotov Oct 18, 2023
ee18815
[memory][tests] Update benchmark tests
NikitaZotov Oct 21, 2023
38d3e46
[memory][refactor] Restruct logs
NikitaZotov Oct 21, 2023
3cc46dd
[memory][tests] Update benchmark tests for sc-links
NikitaZotov Oct 22, 2023
779e4b8
[memory][fs-memory] Update fs-memory processes sync
NikitaZotov Oct 22, 2023
4ad6787
[memory][fs-memory] Optimize links appending
NikitaZotov Oct 22, 2023
0651ab9
[memory][tests] Update benchmark tests for links
NikitaZotov Oct 22, 2023
9e29482
[memory][tests] Fix parallel remove elements
NikitaZotov Oct 22, 2023
ccc8c40
[memory] Fix double edge iterations
NikitaZotov Oct 24, 2023
0a823df
[memory][event][context] Fix events and contexts deletion
NikitaZotov Oct 26, 2023
0305b07
[memory][events][contexts] Reorder memory, events and contexts shutdo…
NikitaZotov Nov 4, 2023
8c1a3a0
[memory] Add checking if reuse empty elements
NikitaZotov Nov 4, 2023
dd7de75
[memory][events][refactor] Divide events logic into managers
NikitaZotov Nov 9, 2023
0b12eb4
[memory][context] Refactor memory contexts, implement context manager
NikitaZotov Nov 10, 2023
fb6a519
[memory] Restruct processes distribution logic
NikitaZotov Nov 24, 2023
24d846e
[memory][fs-memory] Optimize imports, secure interfaces
NikitaZotov Nov 26, 2023
51d37dd
[docs] Apply changes
NikitaZotov Nov 26, 2023
b49a307
[memory][fs-memory][tests] Add tests for invalid params
NikitaZotov Nov 26, 2023
628b19a
[sc-memory][tests] Upgrade iterator3 tests
NikitaZotov Nov 27, 2023
a38c389
[docs] Apply changes
NikitaZotov Nov 28, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
[memory][fs-memory] Sync save, load and dump statistics
  • Loading branch information
NikitaZotov committed Nov 28, 2023
commit 93ca183b61a39abbb178eecb90420e020f83ec9a
4 changes: 2 additions & 2 deletions sc-memory/sc-core/sc-store/sc-base/sc_monitor.c
Original file line number Diff line number Diff line change
Expand Up @@ -124,8 +124,8 @@ sc_bool sc_monitor_try_acquire_write(sc_monitor * monitor)
sc_cond_init(&current_request->condition);
sc_queue_push(monitor->queue, current_request);

sc_bool const result
= sc_queue_front(monitor->queue) != current_request || monitor->active_writer || monitor->active_readers > 0;
sc_bool const result =
sc_queue_front(monitor->queue) != current_request || monitor->active_writer || monitor->active_readers > 0;
sc_mem_free(sc_queue_pop(monitor->queue));

if (!result)
Expand Down
29 changes: 21 additions & 8 deletions sc-memory/sc-core/sc-store/sc-fs-memory/sc_fs_memory.c
Original file line number Diff line number Diff line change
Expand Up @@ -148,7 +148,8 @@ sc_bool _sc_fs_memory_load_sc_memory_segments(sc_segment ** segments, sc_addr_se
sc_uint32 element_size = is_no_deprecated_segments ? sizeof(sc_element) : OLD_SC_ELEMENT_SIZE;
if (is_no_deprecated_segments)
{
if (sc_io_channel_read_chars(segments_channel, (sc_char *)segments_num, sizeof(sc_addr_seg), &read_bytes, null_ptr) !=
if (sc_io_channel_read_chars(
segments_channel, (sc_char *)segments_num, sizeof(sc_addr_seg), &read_bytes, null_ptr) !=
SC_FS_IO_STATUS_NORMAL ||
read_bytes != sizeof(sc_addr_seg))
{
Expand Down Expand Up @@ -260,26 +261,38 @@ sc_bool _sc_fs_memory_save_sc_memory_segments(sc_segment ** segments, sc_addr_se

for (sc_addr_seg idx = 0; idx < segments_num; ++idx)
{
sc_segment const * segment = segments[idx];
sc_segment * segment = segments[idx];
if (segment == null_ptr)
{
sc_fs_memory_error("Error while attribute `segment` writing");
goto error;
}

if (segment == null_ptr ||
sc_io_channel_write_chars(
sc_monitor_acquire_read(&segment->monitor);

if (sc_io_channel_write_chars(
segments_channel, (sc_char *)segment->elements, SC_SEG_ELEMENTS_SIZE_BYTE, &written_bytes, null_ptr) !=
SC_FS_IO_STATUS_NORMAL ||
written_bytes != SC_SEG_ELEMENTS_SIZE_BYTE)
{
sc_fs_memory_error("Error while attribute `segment->elements` writing");
goto error;
goto segment_save_error;
}

if (sc_io_channel_write_chars(
segments_channel, (sc_char *)&segment->last_engaged_offset, sizeof(sc_addr_offset), &written_bytes, null_ptr) !=
SC_FS_IO_STATUS_NORMAL ||
segments_channel,
(sc_char *)&segment->last_engaged_offset,
sizeof(sc_addr_offset),
&written_bytes,
null_ptr) != SC_FS_IO_STATUS_NORMAL ||
written_bytes != sizeof(sc_addr_offset))
{
sc_fs_memory_error("Error while attribute `segment->last_element_offset` writing");
goto error;
goto segment_save_error;
}

segment_save_error:
sc_monitor_release_read(&segment->monitor);
}

// rename main file
Expand Down
92 changes: 46 additions & 46 deletions sc-memory/sc-core/sc-store/sc_storage.c
Original file line number Diff line number Diff line change
Expand Up @@ -22,26 +22,26 @@ sc_storage * storage;

sc_bool sc_storage_initialize(sc_memory_params const * params)
{
sc_bool result = sc_fs_memory_initialize_ext(params);
if (result == SC_FALSE)
if (sc_fs_memory_initialize_ext(params) == SC_FALSE)
return SC_FALSE;

storage = sc_mem_new(sc_storage, 1);
storage->max_segments_count = params->max_loaded_segments;
storage->segments_count = 0;
storage->segments = sc_mem_new(sc_segment *, params->max_loaded_segments);
sc_list_init(&storage->no_fully_engaged_segments);
sc_list_init(&storage->segments_with_released_elements);
sc_monitor_init(&storage->segments_monitor);
_sc_monitor_global_init(&storage->addr_monitors_table);

sc_result result = SC_TRUE;
if (params->clear == SC_FALSE)
{
if (sc_fs_memory_load(storage->segments, &storage->segments_count) != SC_TRUE)
return SC_FALSE;
sc_monitor_acquire_write(&storage->segments_monitor);
result = sc_fs_memory_load(storage->segments, &storage->segments_count);
sc_monitor_release_write(&storage->segments_monitor);
}

return SC_TRUE;
return result;
}

sc_bool sc_storage_shutdown(sc_bool save_state)
Expand All @@ -55,15 +55,19 @@ sc_bool sc_storage_shutdown(sc_bool save_state)
if (sc_fs_memory_shutdown() == SC_FALSE)
return SC_FALSE;

for (sc_uint32 idx = 0; idx < storage->segments_count; idx++)
sc_monitor_acquire_write(&storage->segments_monitor);

for (sc_addr_seg idx = 0; idx < storage->segments_count; idx++)
{
if (storage->segments[idx] == null_ptr)
continue; // skip segments, that are not loaded
sc_segment_free(storage->segments[idx]);
sc_segment * segment = storage->segments[idx];
if (segment == null_ptr)
continue;
sc_segment_free(segment);
}

sc_monitor_release_write(&storage->segments_monitor);

sc_mem_free(storage->segments);
sc_list_destroy(storage->no_fully_engaged_segments);
sc_list_destroy(storage->segments_with_released_elements);
sc_monitor_destroy(&storage->segments_monitor);
_sc_monitor_global_destroy(&storage->addr_monitors_table);
Expand Down Expand Up @@ -104,7 +108,7 @@ sc_result sc_storage_get_element_by_addr(sc_addr addr, sc_element ** el)
goto error;

*el = sc_segment_get_element_by_offset(segment, addr.offset);
if (((*el)->flags.access_levels & SC_ACCESS_LVL_ELEMENT_EXIST) == 0)
if (((*el)->flags.access_levels & SC_ACCESS_LVL_ELEMENT_EXIST) != SC_ACCESS_LVL_ELEMENT_EXIST)
goto error;

result = SC_RESULT_OK;
Expand Down Expand Up @@ -177,7 +181,7 @@ sc_segment * _sc_storage_get_last_free_segment()
if (storage->segments_count == 0)
goto error;

sc_uint32 last_segment_idx = storage->segments_count - 1;
sc_addr_seg last_segment_idx = storage->segments_count - 1;
segment = storage->segments[last_segment_idx];
if (segment->last_engaged_offset == SC_SEGMENT_ELEMENTS_COUNT)
{
Expand Down Expand Up @@ -249,8 +253,8 @@ sc_element * sc_storage_append_el_into_segments(sc_memory_context const * ctx, s

sc_result sc_storage_element_free(sc_memory_context const * ctx, sc_addr addr)
{
GHashTable * remove_table = null_ptr;
GSList * remove_list = null_ptr;
sc_hash_table * remove_table = null_ptr;
sc_hash_table_list * remove_list = null_ptr;

sc_result result;

Expand All @@ -260,45 +264,45 @@ sc_result sc_storage_element_free(sc_memory_context const * ctx, sc_addr addr)
if (result != SC_RESULT_OK)
goto error;

remove_table = g_hash_table_new(g_direct_hash, g_direct_equal);
g_hash_table_insert(remove_table, GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(addr)), el);
remove_table = sc_hash_table_init(g_direct_hash, g_direct_equal, null_ptr, null_ptr);
sc_hash_table_insert(remove_table, GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(addr)), el);

remove_list = g_slist_append(remove_list, GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(addr)));
remove_list = sc_hash_table_list_append(remove_list, GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(addr)));
while (remove_list != null_ptr)
{
// get sc-addr for removing
sc_uint32 addr_int = GPOINTER_TO_UINT(remove_list->data);
sc_addr_hash addr_int = GPOINTER_TO_UINT(remove_list->data);
sc_addr _addr;
_addr.seg = SC_ADDR_LOCAL_SEG_FROM_INT(addr_int);
_addr.offset = SC_ADDR_LOCAL_OFFSET_FROM_INT(addr_int);

gpointer p_addr = GUINT_TO_POINTER(addr_int);
sc_pointer p_addr = GUINT_TO_POINTER(addr_int);

// go to next sc-addr in list
remove_list = g_slist_delete_link(remove_list, remove_list);
remove_list = sc_hash_table_list_remove_sublist(remove_list, remove_list);

result = sc_storage_get_element_by_addr(_addr, &el);
if (result != SC_RESULT_OK)
goto error;

g_hash_table_insert(remove_table, p_addr, el);
sc_hash_table_insert(remove_table, p_addr, el);

// Iterate all connectors for deleted element and append them into remove_list
_addr = el->first_out_arc;
while (SC_ADDR_IS_NOT_EMPTY(_addr))
{
p_addr = GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(_addr));
sc_element * el2 = g_hash_table_lookup(remove_table, p_addr);
sc_element * el2 = sc_hash_table_get(remove_table, p_addr);

if (el2 == null_ptr)
{
result = sc_storage_get_element_by_addr(_addr, &el2);
if (result != SC_RESULT_OK)
goto error;

g_hash_table_insert(remove_table, p_addr, el2);
sc_hash_table_insert(remove_table, p_addr, el2);

remove_list = g_slist_append(remove_list, p_addr);
remove_list = sc_hash_table_list_append(remove_list, p_addr);
}

_addr = el2->arc.next_out_arc;
Expand All @@ -308,17 +312,17 @@ sc_result sc_storage_element_free(sc_memory_context const * ctx, sc_addr addr)
while (SC_ADDR_IS_NOT_EMPTY(_addr))
{
p_addr = GUINT_TO_POINTER(SC_ADDR_LOCAL_TO_INT(_addr));
sc_element * el2 = g_hash_table_lookup(remove_table, p_addr);
sc_element * el2 = sc_hash_table_get(remove_table, p_addr);

if (el2 == null_ptr)
{
result = sc_storage_get_element_by_addr(_addr, &el2);
if (result != SC_RESULT_OK)
goto error;

g_hash_table_insert(remove_table, p_addr, el2);
sc_hash_table_insert(remove_table, p_addr, el2);

remove_list = g_slist_append(remove_list, p_addr);
remove_list = sc_hash_table_list_append(remove_list, p_addr);
}

_addr = el2->arc.next_in_arc;
Expand All @@ -331,12 +335,12 @@ sc_result sc_storage_element_free(sc_memory_context const * ctx, sc_addr addr)
// now we need to erase all elements
GHashTableIter iter;
g_hash_table_iter_init(&iter, remove_table);
gpointer key, value;
sc_pointer key, value;
while (g_hash_table_iter_next(&iter, &key, &value) == SC_TRUE)
{
el = value;
sc_uint32 uint_addr = GPOINTER_TO_UINT(key);
gpointer p_addr;
sc_addr_hash uint_addr = GPOINTER_TO_UINT(key);
sc_pointer p_addr;
addr.offset = SC_ADDR_LOCAL_OFFSET_FROM_INT(uint_addr);
addr.seg = SC_ADDR_LOCAL_SEG_FROM_INT(uint_addr);

Expand Down Expand Up @@ -424,9 +428,9 @@ sc_result sc_storage_element_free(sc_memory_context const * ctx, sc_addr addr)
result = SC_RESULT_OK;
error:
if (remove_list != null_ptr)
g_slist_free(remove_list);
sc_hash_table_list_destroy(remove_list);
if (remove_table != null_ptr)
g_hash_table_destroy(remove_table);
sc_hash_table_destroy(remove_table);
return result;
}

Expand Down Expand Up @@ -882,26 +886,22 @@ sc_result sc_storage_get_elements_stat(sc_stat * stat)
sc_mem_set(stat, 0, sizeof(sc_stat));

sc_monitor_acquire_read(&storage->segments_monitor);
sc_addr_seg count = storage->segments_count;
sc_monitor_release_read(&storage->segments_monitor);

sc_int32 i;
for (i = 0; i < storage->segments_count; ++i)
for (sc_addr_seg i = 0; i < storage->segments_count; ++i)
{
sc_segment * seg = storage->segments[i];
sc_segment_collect_elements_stat(seg, stat);
}
sc_segment * segment = storage->segments[i];

sc_monitor_release_read(&storage->segments_monitor);
sc_monitor_acquire_read(&segment->monitor);
sc_segment_collect_elements_stat(segment, stat);
sc_monitor_release_read(&segment->monitor);
}

return SC_RESULT_OK;
}

sc_result sc_storage_save(sc_memory_context const * ctx)
{
sc_monitor_acquire_read(&storage->segments_monitor);

sc_fs_memory_save(storage->segments, storage->segments_count);

sc_monitor_release_read(&storage->segments_monitor);

return SC_RESULT_OK;
return sc_fs_memory_save(storage->segments, storage->segments_count) ? SC_RESULT_OK : SC_RESULT_ERROR;
}
1 change: 0 additions & 1 deletion sc-memory/sc-core/sc-store/sc_storage.h
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@ typedef struct
sc_segment ** segments;
sc_addr_seg segments_count;
sc_addr_seg max_segments_count;
sc_list * no_fully_engaged_segments;
sc_list * segments_with_released_elements;
sc_monitor_table addr_monitors_table;
sc_monitor segments_monitor;
Expand Down