[Date Prev][Date Next] [Thread Prev][Thread Next] [Date Index] [Thread Index]

Bug#200619: gcc: parisc: compiling dietlibc-dev with -Os causes segfault



Package: gcc

This applies to gcc_3.2.3-6 and gcc_3.3.1-0pre0 on hppa; gcc_3.0.4-16
doesn't have this problem.

The dietlibc_0.22-2 builds with -Os by default on hppa, and this test
program linked against the dietlibc dies with a segfault after fopen():

#include <stdio.h>
int main() {
  FILE *f;
  fopen("foo", "a");
  fclose(f);
  return(0);
}

If the dietlibc is built without -Os, the program works fine.

The bug is triggered by the following changes in
dietlibc/libstdio/fdglue2.c (I attach fdglue2.c for reference):

diff -u -r1.10 -r1.11
--- libstdio/fdglue2.c  9 Feb 2002 00:45:18 -0000       1.10
+++ libstdio/fdglue2.c  4 Dec 2002 18:32:33 -0000       1.11
@@ -1,5 +1,6 @@
 #include <unistd.h>
 #include <fcntl.h>
+#include <sys/stat.h>
 #include <errno.h>
 #include "dietstdio.h"
 #include <stdlib.h>
@@ -23,7 +24,12 @@
   tmp->bm=0;
   tmp->bs=0;
   tmp->buflen=BUFSIZE;
-  tmp->flags=0;
+  {
+    struct stat st;
+    fstat(fd,&st);
+    tmp->flags=(S_ISFIFO(st.st_mode))?FDPIPE:0;
+  }
+  tmp->popen_kludge=0;
   if (__stdio_atexit==0) {
     __stdio_atexit=1;
     atexit(__stdio_flushall);
  
The dietlibc compiled with revision 1.10 doesn't segfault, revision 1.11
does.

The code changes look fine to me, and it works on other architectures,
so I suppose it's a problem with gcc version >=3.2 on hppa optimizing
the code.

I've a complete test environment on paer, and did run this through gdb
several times.  If you need more details, just ask for it, I should be
able to provide them.

Regards, Gerrit.
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <errno.h>
#include "dietstdio.h"
#include <stdlib.h>
#include <pthread.h>

extern int __stdio_atexit;

FILE* __stdio_init_file_nothreads(int fd,int closeonerror);
FILE* __stdio_init_file_nothreads(int fd,int closeonerror) {
  FILE *tmp=(FILE*)malloc(sizeof(FILE));
  if (!tmp) goto err_out;
  tmp->buf=(char*)malloc(BUFSIZE);
  if (!tmp->buf) {
    free(tmp);
err_out:
    if (closeonerror) close(fd);
    errno=ENOMEM;
    return 0;
  }
  tmp->fd=fd;
  tmp->bm=0;
  tmp->bs=0;
  tmp->buflen=BUFSIZE;
  {
    struct stat st;
    fstat(fd,&st);
    tmp->flags=(S_ISFIFO(st.st_mode))?FDPIPE:0;
  }
  tmp->popen_kludge=0;
  if (__stdio_atexit==0) {
    __stdio_atexit=1;
    atexit(__stdio_flushall);
  }
  tmp->next=__stdio_root;
  __stdio_root=tmp;
  tmp->ungotten=0;
  return tmp;
}

FILE* __stdio_init_file(int fd,int closeonerror) __attribute__((weak,alias("__stdio_init_file_nothreads")));

Reply to: