*** glibc detected *** free(): invalid next size (fast) — should work?

问:

I'm working on an existing c project (spglib on sourceforge), and I'm running into the following problem after cleaning up some array initializations:

* glibc detected tests/spglibtest: free(): invalid next size (fast): 0x08ab46e0 **

The backtrace is:

#0  0xb7fe1424 in __kernel_vsyscall ()
#1  0xb5cfdd61 in raise () from /lib/libc.so.6
#2  0xb5cff5ee in abort () from /lib/libc.so.6
#3  0xb5d397ed in ?? () from /lib/libc.so.6
#4  0xb5d3f7b1 in ?? () from /lib/libc.so.6
#5  0xb5d4052b in ?? () from /lib/libc.so.6
#6  0xb5d441cd in free () from /lib/libc.so.6
#7  0xb6681484 in sym_get_multiplicity (cell=0xbfffe1f0, symprec=0.050000000000000003) at /git/xtalopt-public/src/spglib/symmetry.c:168
#8  0xb6680550 in spg_find_primitive (lattice=0xbfffe2a8, position=0x813c6f0, types=0x813c700, num_atom=2, symprec=0.050000000000000003)
    at /git/xtalopt-public/src/spglib/spglib.c:253

The error is in the "free(trans)" line below:

int sym_get_multiplicity(const Cell *cell, const double symprec)
{
  int i, rc;
  double **trans;
  trans = (double**)malloc(cell->size * sizeof(double*));
  for (i = 0; i < cell->size; i++) {
    trans[i] = (double*)malloc(3 * sizeof(double));
  }

  rc = get_translation(&trans[0][0], identity, cell, symprec);

  for (i = 0; i < cell->size; i++) {
    free(trans[i]);
  }
  free(trans);

  return rc;
}

get_translation assigns values to trans like so:

static int get_translation(double trans[][3], const int rot[3][3], const Cell *cell,
                           const double symprec)
{
...
  for (j = 0; j < 3; j++) {
    trans[num_trans][j] = someDouble;
  }
...
}

Valgrind is showing the following when writing to the array in get_translation:

==17929== Invalid write of size 8
==17929==    at 0x56BE8A7: get_translation (symmetry.c:285)
==17929==    by 0x56BE44B: sym_get_multiplicity (symmetry.c:163)
...
==17929==  Address 0x9cb5868 is 0 bytes after a block of size 8 alloc'd
==17929==    at 0x4024918: malloc (in /usr/lib/valgrind/vgpreload_memcheck-x86-linux.so)
==17929==    by 0x56BE3F7: sym_get_multiplicity (symmetry.c:158)
....

This suggests to me that it's trying to write past the end of the allocated memory for trans, but it's writing to trans[0][0], and trans is of dimension [2][3]. This should work, AFAIK, can anyone see something that I'm missing?


回答:


Your types are wrong, you can't pass a pointer to an array of pointers to a function expecting an array of arrays (i.e. a pointer to an array).

For the signature of get_translation that you have, you need:

double (*trans)[3] = malloc(cell->size * sizeof(double[3]));

### 解决 Glibc 编译过程中 `make` 安装错误 当遇到 Glibc 编译时的 `make install Error 2` 错误,通常是因为使用的 Make 版本过低。为了确保编译顺利进行,建议先更新到更高版本的 Make 工具。 #### 更新 Make 到最新稳定版 可以按照如下方法来安装较新版本的 Make: ```bash wget http://ftp.gnu.org/gnu/make/make-3.82.tar.gz tar zxvf make-3.82.tar.gz cd make-3.82 ./configure --prefix=/usr/local/make_3_82 make sudo make install export PATH=/usr/local/make_3_82/bin:$PATH ``` 通过上述命令序列,可以从 GNU FTP 站点获取并解压指定版本的 Make 文件包,配置构建环境,并最终完成安装和路径设置[^2]。 #### 验证 Make 是否成功更新 在执行以上操作之后,可以通过下面这条指令验证当前系统的 Make 版本是否已被正确替换为新的版本: ```bash make --version ``` 如果显示的是刚刚安装的新版本号,则说明更新成功;否则可能需要检查环境变量 `$PATH` 的顺序或权限问题。 #### 准备工作前注意事项 对于 Glibc 这样的核心库来说,在重新编译之前还需要注意一些准备工作: 1. **备份现有系统**:因为任何对基础库的操作都可能导致不可预见的问题。 2. **创建独立的工作目录**:避免污染源码树以及影响其他软件包。 3. **调整依赖关系**:某些情况下还需同步升级相关联的工具链组件。 #### 开始正式编译流程 一旦确认 Make 被适当更新后,就可以继续尝试重新编译 Glibc: ```bash mkdir build && cd build ../configure --prefix=/opt/glibc-2.25 make -j$(nproc) sudo make install ``` 这里使用 `-j$(nproc)` 参数可以让多核 CPU 并行处理加快编译速度。最后一步会把新建好的 Glibc 库文件放置于 `/opt/glibc-2.25/lib` 中[^4]。 #### 设置动态链接器路径 为了让程序能够找到新版 Glibc 提供的标准 C 库和其他共享对象,需临时修改 `LD_LIBRARY_PATH` 或者永久性地编辑 `/etc/ld.so.conf.d/new_glibc.conf` 文件加入相应路径后再运行 `ldconfig` 命令刷新缓存。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值