File: anonfile.c

package info (click to toggle)
bash 5.3-1
  • links: PTS
  • area: main
  • in suites: forky, sid
  • size: 43,860 kB
  • sloc: ansic: 134,738; sh: 8,866; yacc: 5,966; makefile: 4,697; perl: 4,105; asm: 48; awk: 23; sed: 16
file content (120 lines) | stat: -rw-r--r-- 2,529 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/* anonfile.c - open and close temporary files (anonymous and memory-backed if possible). */

/* Copyright (C) 2023-2024 Free Software Foundation, Inc.

   This file is part of GNU Bash, the Bourne Again SHell.

   Bash is free software: you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation, either version 3 of the License, or
   (at your option) any later version.

   Bash is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with Bash.  If not, see <http://www.gnu.org/licenses/>.
*/
   
#include <config.h>

#if defined (HAVE_UNISTD_H)
#  include <unistd.h>
#endif
#include <bashtypes.h>

#if defined (HAVE_MEMFD_CREATE) || defined (HAVE_SHM_MKSTEMP)
#  include <sys/mman.h>
#endif
#include <filecntl.h>

#include <errno.h>

#include <shell.h>
#include <bashansi.h>

static int anonunlink (const char *);

#if defined (HAVE_MEMFD_CREATE) && !defined (MFD_NOEXEC_SEAL)
#  define MFD_NOEXEC_SEAL 0
#endif

#if defined (HAVE_SHM_MKSTEMP)
static int
anonshmopen (const char *name, int flags, char **fn)
{
  int fd;
  char *fname;

  fd = -1;
  if (fn)
    *fn = 0;

  fname = savestring ("/shm-XXXXXXXXXX");
  fd = shm_mkstemp (fname);
  if (fd < 0)
    {
      free (fname);
      return fd;
    }

  if (fn)
    *fn = fname;
  else
    free (fname);

  return fd;  
}
#endif

int
anonopen (const char *name, int flags, char **fn)
{
  int fd, flag;
  char *fname;

#if defined (HAVE_MEMFD_CREATE)
  /* "Names do not affect the behavior of the file descriptor." */
  fd = memfd_create ("anonopen", MFD_NOEXEC_SEAL);
  if (fd >= 0)
    {
      if (fn)
	*fn = 0;
      return fd;
    }
  /* If memfd_create fails, we fall through to the unlinked-shm-or-regular-file
     implementation. */
#endif

  /* Heuristic */
  flag = (name && *name == '/') ? MT_TEMPLATE : MT_USETMPDIR;

#if defined (HAVE_SHM_MKSTEMP)
  fd = anonshmopen (name, flag, fn);
  if (fd >= 0)
    return fd;		/* anonshmopen sets *FN appropriately */
#endif

  fd = sh_mktmpfd (name, flag|MT_USERANDOM|MT_READWRITE|MT_UNLINK, fn);
  return fd;
}

int
anonclose (int fd, const char *name)
{
  int r;

  r = close (fd);
  return r;
}

static int
anonunlink (const char *fn)
{
  int r;

  r = unlink (fn);
  return r;
}