* configure.ac: Require libgcrypt 1.1.94.
[gnupg.git] / common / sysutils.c
1 /* sysutils.c -  system helpers
2  * Copyright (C) 1998, 1999, 2000, 2001 Free Software Foundation, Inc.
3  *
4  * This file is part of GnuPG.
5  *
6  * GnuPG is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * GnuPG is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
19  */
20
21 #include <config.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <unistd.h>
26 #include <errno.h>
27 #ifdef HAVE_STAT
28 #include <sys/stat.h>
29 #endif
30 #if defined(__linux__) && defined(__alpha__) && __GLIBC__ < 2
31   #include <asm/sysinfo.h>
32   #include <asm/unistd.h>
33 #endif
34 #ifdef HAVE_SETRLIMIT
35   #include <time.h>
36   #include <sys/time.h>
37   #include <sys/resource.h>
38 #endif
39 #include "util.h"
40 #include "i18n.h"
41
42 #include "sysutils.h"
43
44 #if defined(__linux__) && defined(__alpha__) && __GLIBC__ < 2
45 #warning using trap_unaligned
46 static int
47 setsysinfo(unsigned long op, void *buffer, unsigned long size,
48                      int *start, void *arg, unsigned long flag)
49 {
50     return syscall(__NR_osf_setsysinfo, op, buffer, size, start, arg, flag);
51 }
52
53 void
54 trap_unaligned(void)
55 {
56     unsigned int buf[2];
57
58     buf[0] = SSIN_UACPROC;
59     buf[1] = UAC_SIGBUS | UAC_NOPRINT;
60     setsysinfo(SSI_NVPAIRS, buf, 1, 0, 0, 0);
61 }
62 #else
63 void
64 trap_unaligned(void)
65 {  /* dummy */
66 }
67 #endif
68
69
70 int
71 disable_core_dumps (void)
72 {
73  #ifdef HAVE_DOSISH_SYSTEM
74     return 0;
75  #else
76   #ifdef HAVE_SETRLIMIT
77     struct rlimit limit;
78
79     limit.rlim_cur = 0;
80     limit.rlim_max = 0;
81     if( !setrlimit( RLIMIT_CORE, &limit ) )
82         return 0;
83     if( errno != EINVAL && errno != ENOSYS )
84         log_fatal (_("can't disable core dumps: %s\n"), strerror(errno) );
85   #endif
86     return 1;
87  #endif
88 }
89
90
91
92 /* Return a string which is used as a kind of process ID */
93 const byte *
94 get_session_marker( size_t *rlen )
95 {
96     static byte marker[SIZEOF_UNSIGNED_LONG*2];
97     static int initialized;
98
99     if ( !initialized ) {
100         volatile ulong aa, bb; /* we really want the uninitialized value */
101         ulong a, b;
102
103         initialized = 1;
104         /* also this marker is guessable it is not easy to use this 
105          * for a faked control packet because an attacker does not
106          * have enough control about the time the verification does 
107          * take place.  Of course, we can add just more random but 
108          * than we need the random generator even for verification
109          * tasks - which does not make sense. */
110         a = aa ^ (ulong)getpid();
111         b = bb ^ (ulong)time(NULL);
112         memcpy( marker, &a, SIZEOF_UNSIGNED_LONG );
113         memcpy( marker+SIZEOF_UNSIGNED_LONG, &b, SIZEOF_UNSIGNED_LONG );
114     }
115     *rlen = sizeof(marker);
116     return marker;
117 }
118
119
120 #if 0 /* not yet needed - Note that this will require inclusion of
121          cmacros.am in Makefile.am */
122 int
123 check_permissions(const char *path,int extension,int checkonly)
124 {
125 #if defined(HAVE_STAT) && !defined(HAVE_DOSISH_SYSTEM)
126   char *tmppath;
127   struct stat statbuf;
128   int ret=1;
129   int isdir=0;
130
131   if(opt.no_perm_warn)
132     return 0;
133
134   if(extension && path[0]!=DIRSEP_C)
135     {
136       if(strchr(path,DIRSEP_C))
137         tmppath=make_filename(path,NULL);
138       else
139         tmppath=make_filename(GNUPG_LIBDIR,path,NULL);
140     }
141   else
142     tmppath=m_strdup(path);
143
144   /* It's okay if the file doesn't exist */
145   if(stat(tmppath,&statbuf)!=0)
146     {
147       ret=0;
148       goto end;
149     }
150
151   isdir=S_ISDIR(statbuf.st_mode);
152
153   /* Per-user files must be owned by the user.  Extensions must be
154      owned by the user or root. */
155   if((!extension && statbuf.st_uid != getuid()) ||
156      (extension && statbuf.st_uid!=0 && statbuf.st_uid!=getuid()))
157     {
158       if(!checkonly)
159         log_info(_("Warning: unsafe ownership on %s \"%s\"\n"),
160                  isdir?"directory":extension?"extension":"file",path);
161       goto end;
162     }
163
164   /* This works for both directories and files - basically, we don't
165      care what the owner permissions are, so long as the group and
166      other permissions are 0 for per-user files, and non-writable for
167      extensions. */
168   if((extension && (statbuf.st_mode & (S_IWGRP|S_IWOTH)) !=0) ||
169      (!extension && (statbuf.st_mode & (S_IRWXG|S_IRWXO)) != 0))
170     {
171       char *dir;
172
173       /* However, if the directory the directory/file is in is owned
174          by the user and is 700, then this is not a problem.
175          Theoretically, we could walk this test up to the root
176          directory /, but for the sake of sanity, I'm stopping at one
177          level down. */
178
179       dir= make_dirname (tmppath);
180       if(stat(dir,&statbuf)==0 && statbuf.st_uid==getuid() &&
181          S_ISDIR(statbuf.st_mode) && (statbuf.st_mode & (S_IRWXG|S_IRWXO))==0)
182         {
183           xfree (dir);
184           ret=0;
185           goto end;
186         }
187
188       m_free(dir);
189
190       if(!checkonly)
191         log_info(_("Warning: unsafe permissions on %s \"%s\"\n"),
192                  isdir?"directory":extension?"extension":"file",path);
193       goto end;
194     }
195
196   ret=0;
197
198  end:
199   m_free(tmppath);
200
201   return ret;
202
203 #endif /* HAVE_STAT && !HAVE_DOSISH_SYSTEM */
204
205   return 0;
206 }
207 #endif