私はしばらくの間、オープン ソース プロジェクトhttp://gtkworkbook.sourceforge.net/に取り組んできましたが、最近、まるで輪になっているかのような問題に遭遇しました。ヒープの問題があると確信していますが、このコードをあまりにも長く見てきたため、それが何であるかを正確に把握できませんでした。
つまり、簡単に言えば、libcsv パーサーで作業しながら、メモリのブロックを N ポインターから M ポインターに再割り当てしています。追加の列がある場合は、配列の最大サイズを現在のサイズの 2 倍に増やしたいと考えています。現在のコードは次のとおりです。
struct csv_column {
Sheet * sheet;
Cell ** array;
int & array_max;
int & array_size;
int row;
int field;
char * value;
};
static void
cb1 (void * s, size_t length, void * data) {
struct csv_column * column = (struct csv_column *) data;
int & array_max = column->array_max;
// Resize the cell array here.
if (column->field >= array_max) {
int max = (2 * array_max);
(column->array) = (Cell **) g_realloc ((column->array), max * sizeof (Cell*));
for (int ii = array_max; ii array)[column->field] == NULL)
(column->array)[column->field] = cell_new();
Cell * cell = (column->array)[column->field];
cell->set_row (cell, column->row);
cell->set_column (cell, column->field++);
cell->set_value_length (cell, s, length);
}
CsvParser::CsvParser (Workbook * wb,
FILE * log,
int verbosity,
int maxOfFields) {
this->wb = wb;
this->log = log;
this->verbosity = verbosity;
this->sizeOfFields = 0;
this->maxOfFields = maxOfFields;
this->fields = (Cell **) g_malloc (maxOfFields * sizeof (Cell*));
for (int ii = 0; ii maxOfFields; ii++)
this->fields[ii] = NULL;
}
CsvParser::~CsvParser (void) {
for (int ii = 0; ii maxOfFields; ii++) {
if (this->fields[ii])
this->fields[ii]->destroy (this->fields[ii]);
}
g_free (this->fields);
}
valgrind の出力は次のとおりです。
==28476== スレッド 9: ==28476== サイズ 8 の無効な読み取り ==28476== 0x771AF4F: sheet_method_apply_cellarray (sheet.c:351) ==28476== by 0xD930DB7: largefile::CsvParser::run(void*) (CsvParser.cpp:147) ==28476== by 0xDD624C8: 同時実行::thread_run(void*) (Thread.cpp:28) ==28476== by 0xA7B73B9: start_thread (/lib/libpthread-2.9.so 内) ==28476== by 0x80DBFCC: クローン (/lib/libc-2.9.so 内) ==28476== アドレス 0xbc5d4a8 は、アクセス中のポインタの内部で 0 バイトです ==28476== かつては正当な範囲だったが、サイズ 160 のブロックが解放された ==28476== 0x4C25D4F: フリー (vg_replace_malloc.c:323) ==28476== by 0xD9314CA: largefile::cb1(void*, unsigned long, void*) (CsvParser.cpp:57) ==28476== by 0xDB42681: csv_parse (/home/jbellone/work/gtkworkbook/lib/libcsv.so 内) ==28476== by 0xD930D00: largefile::CsvParser::run(void*) (CsvParser.cpp:136) ==28476== by 0xDD624C8: 同時実行::thread_run(void*) (Thread.cpp:28) ==28476== by 0xA7B73B9: start_thread (/lib/libpthread-2.9.so 内) ==28476== by 0x80DBFCC: クローン (/lib/libc-2.9.so 内) ==28476== ==28476== サイズ 8 の無効な読み取り ==28476== 0x771AF66: sheet_method_apply_cellarray (sheet.c:351) ==28476== by 0xD930DB7: largefile::CsvParser::run(void*) (CsvParser.cpp:147) ==28476== by 0xDD624C8: 同時実行::thread_run(void*) (Thread.cpp:28) ==28476== by 0xA7B73B9: start_thread (/lib/libpthread-2.9.so 内) ==28476== by 0x80DBFCC: クローン (/lib/libc-2.9.so 内) ==28476== アドレス 0xbc5d4a8 は、アクセス中のポインタの内部で 0 バイトです ==28476== かつては正当な範囲だったが、サイズ 160 のブロックが解放された ==28476== 0x4C25D4F: フリー (vg_replace_malloc.c:323) ==28476== by 0xD9314CA: largefile::cb1(void*, unsigned long, void*) (CsvParser.cpp:57) ==28476== by 0xDB42681: csv_parse (/home/jbellone/work/gtkworkbook/lib/libcsv.so 内) ==28476== by 0xD930D00: largefile::CsvParser::run(void*) (CsvParser.cpp:136) ==28476== by 0xDD624C8: 同時実行::thread_run(void*) (Thread.cpp:28) ==28476== by 0xA7B73B9: start_thread (/lib/libpthread-2.9.so 内) ==28476== by 0x80DBFCC: クローン (/lib/libc-2.9.so 内)
sheet.c 行 351
gtk_sheet_set_cell_text (GTK_SHEET (sheet->gtk_sheet),
array[ii]->row,
array[ii]->column,
array[ii]->value->str);
sheet.c の関数全体:
static void
sheet_method_apply_cellarray (Sheet * sheet,
Cell ** array,
gint size)
{
ASSERT (sheet != NULL);
g_return_if_fail (array != NULL);
gdk_threads_enter ();
/* We'll see how this performs for now. In the future we may want to go
directly into the GtkSheet structures to get a little more performance
boost (mainly because we should not have to check all the bounds each
time we want to update). */
for (gint ii = 0; ii gtk_sheet),
array[ii]->row,
array[ii]->column,
array[ii]->value->str);
if (!IS_NULLSTR (array[ii]->attributes.bgcolor->str))
sheet->range_set_background (sheet,
&array[ii]->range,
array[ii]->attributes.bgcolor->str);
if (!IS_NULLSTR (array[ii]->attributes.fgcolor->str))
sheet->range_set_foreground (sheet,
&array[ii]->range,
array[ii]->attributes.fgcolor->str);
/* Clear all of the strings */
g_string_assign (array[ii]->value, "");
g_string_assign (array[ii]->attributes.bgcolor, "");
g_string_assign (array[ii]->attributes.fgcolor, "");
}
gdk_threads_leave ();
}
CSV パーサー スレッド
void *
CsvParser::run (void * null) {
this->running = true;
std::queue queue;
struct csv_parser csv;
struct csv_column column = {this->wb->sheet_first,
this->fields,
this->maxOfFields,
this->sizeOfFields,
0,
0,
new char [1024]};
if (csv_init (&csv, CSV_STRICT) != 0) {
std::cerr running == true) {
if (this->inputQueue.size() > 0) {
// Lock, copy, clear, unlock. - Free this up.
this->inputQueue.lock();
this->inputQueue.copy (queue);
this->inputQueue.clear();
this->inputQueue.unlock();
while (queue.size() > 0) {
std::string buf = queue.front(); queue.pop();
size_t bytes = buf.length();
if (this->running == false)
break;
if ((bytes = csv_parse (&csv, buf.c_str(), bytes, cb1, cb2, &column)) == bytes) {
if (csv_error (&csv) == CSV_EPARSE) {
std::cerr wb->sheet_first->apply_array (this->wb->sheet_first,
this->fields,
this->sizeOfFields);
if (column.row >= (column.sheet)->max_rows)
column.row = 0;
}
}
concurrent::Thread::sleep(1);
}
return NULL;
}