| File: | obj-scan-build/tmpfs/../../tmpfs/node.c |
| Location: | line 238, column 19 |
| Description: | Call to 'alloca' has an allocation size of 0 bytes |
| 1 | /* Node state and file contents for tmpfs. | |||||
| 2 | Copyright (C) 2000,01,02 Free Software Foundation, Inc. | |||||
| 3 | ||||||
| 4 | This file is part of the GNU Hurd. | |||||
| 5 | ||||||
| 6 | The GNU Hurd 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, or (at your option) | |||||
| 9 | any later version. | |||||
| 10 | ||||||
| 11 | The GNU Hurd 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 the GNU Hurd; see the file COPYING. If not, write to | |||||
| 18 | the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ | |||||
| 19 | ||||||
| 20 | #include "tmpfs.h" | |||||
| 21 | #include <stddef.h> | |||||
| 22 | #include <stdlib.h> | |||||
| 23 | #include <fcntl.h> | |||||
| 24 | #include <hurd/hurd_types.h> | |||||
| 25 | #include <hurd/store.h> | |||||
| 26 | #include "default_pager_U.h" | |||||
| 27 | ||||||
| 28 | unsigned int num_files; | |||||
| 29 | static unsigned int gen; | |||||
| 30 | ||||||
| 31 | struct node *all_nodes; | |||||
| 32 | ||||||
| 33 | error_t | |||||
| 34 | diskfs_alloc_node (struct node *dp, mode_t mode, struct node **npp) | |||||
| 35 | { | |||||
| 36 | struct disknode *dn; | |||||
| 37 | ||||||
| 38 | dn = calloc (1, sizeof *dn); | |||||
| 39 | if (dn == 0) | |||||
| 40 | return ENOSPC((0x10 << 26) | ((28) & 0x3fff)); | |||||
| 41 | pthread_spin_lock (&diskfs_node_refcnt_lock); | |||||
| 42 | if (round_page (tmpfs_space_used + sizeof *dn)((((vm_offset_t) (tmpfs_space_used + sizeof *dn) + __vm_page_size - 1) / __vm_page_size) * __vm_page_size) / vm_page_size | |||||
| 43 | > tmpfs_page_limit) | |||||
| 44 | { | |||||
| 45 | pthread_spin_unlock (&diskfs_node_refcnt_lock); | |||||
| 46 | free (dn); | |||||
| 47 | return ENOSPC((0x10 << 26) | ((28) & 0x3fff)); | |||||
| 48 | } | |||||
| 49 | dn->gen = gen++; | |||||
| 50 | ++num_files; | |||||
| 51 | tmpfs_space_used += sizeof *dn; | |||||
| 52 | pthread_spin_unlock (&diskfs_node_refcnt_lock); | |||||
| 53 | ||||||
| 54 | dn->type = IFTODT (mode & S_IFMT)(((mode & 0170000) & 0170000) >> 12); | |||||
| 55 | return diskfs_cached_lookup ((ino_t) (uintptr_t) dn, npp); | |||||
| 56 | } | |||||
| 57 | ||||||
| 58 | void | |||||
| 59 | diskfs_free_node (struct node *np, mode_t mode) | |||||
| 60 | { | |||||
| 61 | switch (np->dn->type) | |||||
| 62 | { | |||||
| 63 | case DT_REGDT_REG: | |||||
| 64 | if (np->dn->u.reg.memobj != MACH_PORT_NULL((mach_port_t) 0)) { | |||||
| 65 | vm_deallocate (mach_task_self ()((__mach_task_self_ + 0)), np->dn->u.reg.memref, 4096); | |||||
| 66 | mach_port_deallocate (mach_task_self ()((__mach_task_self_ + 0)), np->dn->u.reg.memobj); | |||||
| 67 | } | |||||
| 68 | break; | |||||
| 69 | case DT_DIRDT_DIR: | |||||
| 70 | assert (np->dn->u.dir.entries == 0)((np->dn->u.dir.entries == 0) ? (void) (0) : __assert_fail ("np->dn->u.dir.entries == 0", "../../tmpfs/node.c", 70 , __PRETTY_FUNCTION__)); | |||||
| 71 | break; | |||||
| 72 | case DT_LNKDT_LNK: | |||||
| 73 | free (np->dn->u.lnk); | |||||
| 74 | break; | |||||
| 75 | } | |||||
| 76 | *np->dn->hprevp = np->dn->hnext; | |||||
| 77 | if (np->dn->hnext != 0) | |||||
| 78 | np->dn->hnext->dn->hprevp = np->dn->hprevp; | |||||
| 79 | free (np->dn); | |||||
| 80 | np->dn = 0; | |||||
| 81 | ||||||
| 82 | --num_files; | |||||
| 83 | tmpfs_space_used -= sizeof *np->dn; | |||||
| 84 | } | |||||
| 85 | ||||||
| 86 | void | |||||
| 87 | diskfs_node_norefs (struct node *np) | |||||
| 88 | { | |||||
| 89 | if (np->dn != 0) | |||||
| 90 | { | |||||
| 91 | /* We don't bother to do this in diskfs_write_disknode, since it only | |||||
| 92 | ever matters here. The node state goes back into the `struct | |||||
| 93 | disknode' while it has no associated diskfs node. */ | |||||
| 94 | ||||||
| 95 | np->dn->size = np->dn_stat.st_size; | |||||
| 96 | np->dn->mode = np->dn_stat.st_mode; | |||||
| 97 | np->dn->nlink = np->dn_stat.st_nlink; | |||||
| 98 | np->dn->uid = np->dn_stat.st_uid; | |||||
| 99 | np->dn->author = np->dn_stat.st_author; | |||||
| 100 | np->dn->gid = np->dn_stat.st_gid; | |||||
| 101 | np->dn->atime = np->dn_stat.st_atim; | |||||
| 102 | np->dn->mtime = np->dn_stat.st_mtim; | |||||
| 103 | np->dn->ctime = np->dn_stat.st_ctim; | |||||
| 104 | np->dn->flags = np->dn_stat.st_flags; | |||||
| 105 | ||||||
| 106 | switch (np->dn->type) | |||||
| 107 | { | |||||
| 108 | case DT_REGDT_REG: | |||||
| 109 | assert (np->allocsize % vm_page_size == 0)((np->allocsize % vm_page_size == 0) ? (void) (0) : __assert_fail ("np->allocsize % vm_page_size == 0", "../../tmpfs/node.c" , 109, __PRETTY_FUNCTION__)); | |||||
| 110 | np->dn->u.reg.allocpages = np->allocsize / vm_page_size; | |||||
| 111 | break; | |||||
| 112 | case DT_CHRDT_CHR: | |||||
| 113 | case DT_BLKDT_BLK: | |||||
| 114 | np->dn->u.chr = np->dn_stat.st_rdev; | |||||
| 115 | break; | |||||
| 116 | } | |||||
| 117 | ||||||
| 118 | /* Remove this node from the cache list rooted at `all_nodes'. */ | |||||
| 119 | *np->dn->hprevp = np->dn->hnext; | |||||
| 120 | if (np->dn->hnext != 0) | |||||
| 121 | np->dn->hnext->dn->hprevp = np->dn->hprevp; | |||||
| 122 | np->dn->hnext = 0; | |||||
| 123 | np->dn->hprevp = 0; | |||||
| 124 | } | |||||
| 125 | ||||||
| 126 | free (np); | |||||
| 127 | } | |||||
| 128 | ||||||
| 129 | static void | |||||
| 130 | recompute_blocks (struct node *np) | |||||
| 131 | { | |||||
| 132 | struct disknode *const dn = np->dn; | |||||
| 133 | struct stat *const st = &np->dn_stat; | |||||
| 134 | ||||||
| 135 | st->st_blocks = sizeof *dn + dn->translen; | |||||
| 136 | switch (dn->type) | |||||
| 137 | { | |||||
| 138 | case DT_REGDT_REG: | |||||
| 139 | np->allocsize = dn->u.reg.allocpages * vm_page_size; | |||||
| 140 | st->st_blocks += np->allocsize; | |||||
| 141 | break; | |||||
| 142 | case DT_LNKDT_LNK: | |||||
| 143 | st->st_blocks += st->st_size + 1; | |||||
| 144 | break; | |||||
| 145 | case DT_CHRDT_CHR: | |||||
| 146 | case DT_BLKDT_BLK: | |||||
| 147 | st->st_rdev = dn->u.chr; | |||||
| 148 | break; | |||||
| 149 | case DT_DIRDT_DIR: | |||||
| 150 | st->st_blocks += dn->size; | |||||
| 151 | break; | |||||
| 152 | } | |||||
| 153 | st->st_blocks = (st->st_blocks + 511) / 512; | |||||
| 154 | } | |||||
| 155 | ||||||
| 156 | /* Fetch inode INUM, set *NPP to the node structure; | |||||
| 157 | gain one user reference and lock the node. */ | |||||
| 158 | error_t | |||||
| 159 | diskfs_cached_lookup (ino_t inum, struct node **npp) | |||||
| 160 | { | |||||
| 161 | struct disknode *dn = (void *) (uintptr_t) inum; | |||||
| 162 | struct node *np; | |||||
| 163 | ||||||
| 164 | assert (npp)((npp) ? (void) (0) : __assert_fail ("npp", "../../tmpfs/node.c" , 164, __PRETTY_FUNCTION__)); | |||||
| 165 | ||||||
| 166 | if (dn->hprevp != 0) /* There is already a node. */ | |||||
| 167 | { | |||||
| 168 | np = *dn->hprevp; | |||||
| 169 | assert (np->dn == dn)((np->dn == dn) ? (void) (0) : __assert_fail ("np->dn == dn" , "../../tmpfs/node.c", 169, __PRETTY_FUNCTION__)); | |||||
| 170 | assert (*dn->hprevp == np)((*dn->hprevp == np) ? (void) (0) : __assert_fail ("*dn->hprevp == np" , "../../tmpfs/node.c", 170, __PRETTY_FUNCTION__)); | |||||
| 171 | ||||||
| 172 | diskfs_nref (np); | |||||
| 173 | } | |||||
| 174 | else | |||||
| 175 | /* Create the new node. */ | |||||
| 176 | { | |||||
| 177 | struct stat *st; | |||||
| 178 | ||||||
| 179 | np = diskfs_make_node (dn); | |||||
| 180 | np->cache_id = (ino_t) (uintptr_t) dn; | |||||
| 181 | ||||||
| 182 | pthread_spin_lock (&diskfs_node_refcnt_lock); | |||||
| 183 | dn->hnext = all_nodes; | |||||
| 184 | if (dn->hnext) | |||||
| 185 | dn->hnext->dn->hprevp = &dn->hnext; | |||||
| 186 | dn->hprevp = &all_nodes; | |||||
| 187 | all_nodes = np; | |||||
| 188 | pthread_spin_unlock (&diskfs_node_refcnt_lock); | |||||
| 189 | ||||||
| 190 | st = &np->dn_stat; | |||||
| 191 | memset (st, 0, sizeof *st); | |||||
| 192 | st->st_fstype = FSTYPE_MEMFS0x00000019; | |||||
| 193 | st->st_fsid = getpid (); | |||||
| 194 | st->st_blksize = vm_page_size; | |||||
| 195 | ||||||
| 196 | st->st_ino = (ino_t) (uintptr_t) dn; | |||||
| 197 | st->st_gen = dn->gen; | |||||
| 198 | ||||||
| 199 | st->st_size = dn->size; | |||||
| 200 | st->st_mode = dn->mode; | |||||
| 201 | st->st_nlink = dn->nlink; | |||||
| 202 | st->st_uid = dn->uid; | |||||
| 203 | st->st_author = dn->author; | |||||
| 204 | st->st_gid = dn->gid; | |||||
| 205 | st->st_atim = dn->atime; | |||||
| 206 | st->st_mtim = dn->mtime; | |||||
| 207 | st->st_ctim = dn->ctime; | |||||
| 208 | st->st_flags = dn->flags; | |||||
| 209 | ||||||
| 210 | st->st_rdev = 0; | |||||
| 211 | np->allocsize = 0; | |||||
| 212 | recompute_blocks (np); | |||||
| 213 | } | |||||
| 214 | ||||||
| 215 | pthread_mutex_lock (&np->lock); | |||||
| 216 | *npp = np; | |||||
| 217 | return 0; | |||||
| 218 | } | |||||
| 219 | ||||||
| 220 | error_t | |||||
| 221 | diskfs_node_iterate (error_t (*fun) (struct node *)) | |||||
| 222 | { | |||||
| 223 | error_t err = 0; | |||||
| 224 | unsigned int num_nodes = 0; | |||||
| 225 | struct node *node, **node_list, **p; | |||||
| 226 | ||||||
| 227 | pthread_spin_lock (&diskfs_node_refcnt_lock); | |||||
| 228 | ||||||
| 229 | /* We must copy everything from the hash table into another data structure | |||||
| 230 | to avoid running into any problems with the hash-table being modified | |||||
| 231 | during processing (normally we delegate access to hash-table with | |||||
| 232 | diskfs_node_refcnt_lock, but we can't hold this while locking the | |||||
| 233 | individual node locks). */ | |||||
| 234 | ||||||
| 235 | for (node = all_nodes; node != 0; node = node->dn->hnext) | |||||
| ||||||
| 236 | num_nodes++; | |||||
| 237 | ||||||
| 238 | p = node_list = alloca (num_nodes * sizeof (struct node *))__builtin_alloca (num_nodes * sizeof (struct node *)); | |||||
| ||||||
| 239 | for (node = all_nodes; node != 0; node = node->dn->hnext) | |||||
| 240 | { | |||||
| 241 | *p++ = node; | |||||
| 242 | node->references++; | |||||
| 243 | } | |||||
| 244 | ||||||
| 245 | pthread_spin_unlock (&diskfs_node_refcnt_lock); | |||||
| 246 | ||||||
| 247 | p = node_list; | |||||
| 248 | while (num_nodes-- > 0) | |||||
| 249 | { | |||||
| 250 | node = *p++; | |||||
| 251 | if (!err) | |||||
| 252 | { | |||||
| 253 | pthread_mutex_lock (&node->lock); | |||||
| 254 | err = (*fun) (node); | |||||
| 255 | pthread_mutex_unlock (&node->lock); | |||||
| 256 | } | |||||
| 257 | diskfs_nrele (node); | |||||
| 258 | } | |||||
| 259 | ||||||
| 260 | return err; | |||||
| 261 | } | |||||
| 262 | ||||||
| 263 | /* The user must define this function. Node NP has some light | |||||
| 264 | references, but has just lost its last hard references. Take steps | |||||
| 265 | so that if any light references can be freed, they are. NP is locked | |||||
| 266 | as is the pager refcount lock. This function will be called after | |||||
| 267 | diskfs_lost_hardrefs. */ | |||||
| 268 | void | |||||
| 269 | diskfs_try_dropping_softrefs (struct node *np) | |||||
| 270 | { | |||||
| 271 | } | |||||
| 272 | ||||||
| 273 | /* The user must define this funcction. Node NP has some light | |||||
| 274 | references but has just lost its last hard reference. NP is locked. */ | |||||
| 275 | void | |||||
| 276 | diskfs_lost_hardrefs (struct node *np) | |||||
| 277 | { | |||||
| 278 | } | |||||
| 279 | ||||||
| 280 | /* The user must define this function. Node NP has just acquired | |||||
| 281 | a hard reference where it had none previously. It is thus now | |||||
| 282 | OK again to have light references without real users. NP is | |||||
| 283 | locked. */ | |||||
| 284 | void | |||||
| 285 | diskfs_new_hardrefs (struct node *np) | |||||
| 286 | { | |||||
| 287 | } | |||||
| 288 | ||||||
| 289 | ||||||
| 290 | ||||||
| 291 | error_t | |||||
| 292 | diskfs_get_translator (struct node *np, char **namep, u_int *namelen) | |||||
| 293 | { | |||||
| 294 | *namelen = np->dn->translen; | |||||
| 295 | if (*namelen == 0) | |||||
| 296 | return 0; | |||||
| 297 | *namep = malloc (*namelen); | |||||
| 298 | if (*namep == 0) | |||||
| 299 | return ENOMEM((0x10 << 26) | ((12) & 0x3fff)); | |||||
| 300 | memcpy (*namep, np->dn->trans, *namelen); | |||||
| 301 | return 0; | |||||
| 302 | } | |||||
| 303 | ||||||
| 304 | error_t | |||||
| 305 | diskfs_set_translator (struct node *np, | |||||
| 306 | const char *name, u_int namelen, | |||||
| 307 | struct protid *cred) | |||||
| 308 | { | |||||
| 309 | char *new; | |||||
| 310 | if (namelen == 0) | |||||
| 311 | { | |||||
| 312 | free (np->dn->trans); | |||||
| 313 | new = 0; | |||||
| 314 | np->dn_stat.st_mode &= ~S_IPTRANS000010000000; | |||||
| 315 | } | |||||
| 316 | else | |||||
| 317 | { | |||||
| 318 | new = realloc (np->dn->trans, namelen); | |||||
| 319 | if (new == 0) | |||||
| 320 | return ENOSPC((0x10 << 26) | ((28) & 0x3fff)); | |||||
| 321 | memcpy (new, name, namelen); | |||||
| 322 | np->dn_stat.st_mode |= S_IPTRANS000010000000; | |||||
| 323 | } | |||||
| 324 | adjust_used (namelen - np->dn->translen); | |||||
| 325 | np->dn->trans = new; | |||||
| 326 | np->dn->translen = namelen; | |||||
| 327 | recompute_blocks (np); | |||||
| 328 | return 0; | |||||
| 329 | } | |||||
| 330 | ||||||
| 331 | static error_t | |||||
| 332 | create_symlink_hook (struct node *np, const char *target) | |||||
| 333 | { | |||||
| 334 | assert (np->dn->u.lnk == 0)((np->dn->u.lnk == 0) ? (void) (0) : __assert_fail ("np->dn->u.lnk == 0" , "../../tmpfs/node.c", 334, __PRETTY_FUNCTION__)); | |||||
| 335 | np->dn_stat.st_size = strlen (target); | |||||
| 336 | if (np->dn_stat.st_size > 0) | |||||
| 337 | { | |||||
| 338 | const size_t size = np->dn_stat.st_size + 1; | |||||
| 339 | np->dn->u.lnk = malloc (size); | |||||
| 340 | if (np->dn->u.lnk == 0) | |||||
| 341 | return ENOSPC((0x10 << 26) | ((28) & 0x3fff)); | |||||
| 342 | memcpy (np->dn->u.lnk, target, size); | |||||
| 343 | np->dn->type = DT_LNKDT_LNK; | |||||
| 344 | adjust_used (size); | |||||
| 345 | recompute_blocks (np); | |||||
| 346 | } | |||||
| 347 | return 0; | |||||
| 348 | } | |||||
| 349 | error_t (*diskfs_create_symlink_hook)(struct node *np, const char *target) | |||||
| 350 | = create_symlink_hook; | |||||
| 351 | ||||||
| 352 | static error_t | |||||
| 353 | read_symlink_hook (struct node *np, char *target) | |||||
| 354 | { | |||||
| 355 | memcpy (target, np->dn->u.lnk, np->dn_stat.st_size + 1); | |||||
| 356 | return 0; | |||||
| 357 | } | |||||
| 358 | error_t (*diskfs_read_symlink_hook)(struct node *np, char *target) | |||||
| 359 | = read_symlink_hook; | |||||
| 360 | ||||||
| 361 | void | |||||
| 362 | diskfs_write_disknode (struct node *np, int wait) | |||||
| 363 | { | |||||
| 364 | } | |||||
| 365 | ||||||
| 366 | void | |||||
| 367 | diskfs_file_update (struct node *np, int wait) | |||||
| 368 | { | |||||
| 369 | diskfs_node_update (np, wait); | |||||
| 370 | } | |||||
| 371 | ||||||
| 372 | error_t | |||||
| 373 | diskfs_node_reload (struct node *node) | |||||
| 374 | { | |||||
| 375 | return 0; | |||||
| 376 | } | |||||
| 377 | ||||||
| 378 | ||||||
| 379 | /* The user must define this function. Truncate locked node NP to be SIZE | |||||
| 380 | bytes long. (If NP is already less than or equal to SIZE bytes | |||||
| 381 | long, do nothing.) If this is a symlink (and diskfs_shortcut_symlink | |||||
| 382 | is set) then this should clear the symlink, even if | |||||
| 383 | diskfs_create_symlink_hook stores the link target elsewhere. */ | |||||
| 384 | error_t | |||||
| 385 | diskfs_truncate (struct node *np, off_t size) | |||||
| 386 | { | |||||
| 387 | if (np->dn->type == DT_LNKDT_LNK) | |||||
| 388 | { | |||||
| 389 | free (np->dn->u.lnk); | |||||
| 390 | adjust_used (size - np->dn_stat.st_size); | |||||
| 391 | np->dn->u.lnk = 0; | |||||
| 392 | np->dn_stat.st_size = size; | |||||
| 393 | return 0; | |||||
| 394 | } | |||||
| 395 | ||||||
| 396 | if (np->allocsize <= size) | |||||
| 397 | return 0; | |||||
| 398 | ||||||
| 399 | assert (np->dn->type == DT_REG)((np->dn->type == DT_REG) ? (void) (0) : __assert_fail ( "np->dn->type == DT_REG", "../../tmpfs/node.c", 399, __PRETTY_FUNCTION__ )); | |||||
| 400 | ||||||
| 401 | if (default_pager == MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 402 | return EIO((0x10 << 26) | ((5) & 0x3fff)); | |||||
| 403 | ||||||
| 404 | np->dn_stat.st_size = size; | |||||
| 405 | ||||||
| 406 | off_t set_size = size; | |||||
| 407 | size = round_page (size)((((vm_offset_t) (size) + __vm_page_size - 1) / __vm_page_size ) * __vm_page_size); | |||||
| 408 | ||||||
| 409 | if (np->dn->u.reg.memobj != MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 410 | { | |||||
| 411 | error_t err = default_pager_object_set_size (np->dn->u.reg.memobj, set_size); | |||||
| 412 | if (err == MIG_BAD_ID-303) | |||||
| 413 | /* This is an old default pager. We have no way to truncate the | |||||
| 414 | memory object. Note that the behavior here will be wrong in | |||||
| 415 | two ways: user accesses past the end won't fault; and, more | |||||
| 416 | importantly, later growing the file won't zero the contents | |||||
| 417 | past the size we just supposedly truncated to. For proper | |||||
| 418 | behavior, use a new default pager. */ | |||||
| 419 | return 0; | |||||
| 420 | if (err) | |||||
| 421 | return err; | |||||
| 422 | } | |||||
| 423 | /* Otherwise it never had any real contents. */ | |||||
| 424 | ||||||
| 425 | adjust_used (size - np->allocsize); | |||||
| 426 | np->dn_stat.st_blocks += (size - np->allocsize) / 512; | |||||
| 427 | np->allocsize = size; | |||||
| 428 | ||||||
| 429 | return 0; | |||||
| 430 | } | |||||
| 431 | ||||||
| 432 | /* The user must define this function. Grow the disk allocated to locked node | |||||
| 433 | NP to be at least SIZE bytes, and set NP->allocsize to the actual | |||||
| 434 | allocated size. (If the allocated size is already SIZE bytes, do | |||||
| 435 | nothing.) CRED identifies the user responsible for the call. */ | |||||
| 436 | error_t | |||||
| 437 | diskfs_grow (struct node *np, off_t size, struct protid *cred) | |||||
| 438 | { | |||||
| 439 | assert (np->dn->type == DT_REG)((np->dn->type == DT_REG) ? (void) (0) : __assert_fail ( "np->dn->type == DT_REG", "../../tmpfs/node.c", 439, __PRETTY_FUNCTION__ )); | |||||
| 440 | ||||||
| 441 | if (np->allocsize >= size) | |||||
| 442 | return 0; | |||||
| 443 | ||||||
| 444 | off_t set_size = size; | |||||
| 445 | size = round_page (size)((((vm_offset_t) (size) + __vm_page_size - 1) / __vm_page_size ) * __vm_page_size); | |||||
| 446 | if (round_page (tmpfs_space_used + size - np->allocsize)((((vm_offset_t) (tmpfs_space_used + size - np->allocsize) + __vm_page_size - 1) / __vm_page_size) * __vm_page_size) | |||||
| 447 | / vm_page_size > tmpfs_page_limit) | |||||
| 448 | return ENOSPC((0x10 << 26) | ((28) & 0x3fff)); | |||||
| 449 | ||||||
| 450 | if (default_pager == MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 451 | return EIO((0x10 << 26) | ((5) & 0x3fff)); | |||||
| 452 | ||||||
| 453 | if (np->dn->u.reg.memobj != MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 454 | { | |||||
| 455 | /* Increase the limit the memory object will allow to be accessed. */ | |||||
| 456 | error_t err = default_pager_object_set_size (np->dn->u.reg.memobj, set_size); | |||||
| 457 | if (err == MIG_BAD_ID-303) /* Old default pager, never limited it. */ | |||||
| 458 | err = 0; | |||||
| 459 | if (err) | |||||
| 460 | return err; | |||||
| 461 | } | |||||
| 462 | ||||||
| 463 | adjust_used (size - np->allocsize); | |||||
| 464 | np->dn_stat.st_blocks += (size - np->allocsize) / 512; | |||||
| 465 | np->allocsize = size; | |||||
| 466 | return 0; | |||||
| 467 | } | |||||
| 468 | ||||||
| 469 | mach_port_t | |||||
| 470 | diskfs_get_filemap (struct node *np, vm_prot_t prot) | |||||
| 471 | { | |||||
| 472 | error_t err; | |||||
| 473 | ||||||
| 474 | if (np->dn->type != DT_REGDT_REG) | |||||
| 475 | { | |||||
| 476 | errno(*__errno_location ()) = EOPNOTSUPP((0x10 << 26) | ((45) & 0x3fff)); /* ? */ | |||||
| 477 | return MACH_PORT_NULL((mach_port_t) 0); | |||||
| 478 | } | |||||
| 479 | ||||||
| 480 | if (default_pager == MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 481 | { | |||||
| 482 | errno(*__errno_location ()) = EIO((0x10 << 26) | ((5) & 0x3fff)); | |||||
| 483 | return MACH_PORT_NULL((mach_port_t) 0); | |||||
| 484 | } | |||||
| 485 | ||||||
| 486 | /* We don't bother to create the memory object until the first time we | |||||
| 487 | need it (i.e. first mapping or i/o). This way we might have a clue | |||||
| 488 | what size it's going to be beforehand, so we can tell the default | |||||
| 489 | pager how big to make its bitmaps. This is just an optimization for | |||||
| 490 | the default pager; the memory object can be expanded at any time just | |||||
| 491 | by accessing more of it. (It also optimizes the case of empty files | |||||
| 492 | so we might never make a memory object at all.) */ | |||||
| 493 | if (np->dn->u.reg.memobj == MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 494 | { | |||||
| 495 | error_t err = default_pager_object_create (default_pager, | |||||
| 496 | &np->dn->u.reg.memobj, | |||||
| 497 | np->allocsize); | |||||
| 498 | if (err) | |||||
| 499 | { | |||||
| 500 | errno(*__errno_location ()) = err; | |||||
| 501 | return MACH_PORT_NULL((mach_port_t) 0); | |||||
| 502 | } | |||||
| 503 | assert (np->dn->u.reg.memobj != MACH_PORT_NULL)((np->dn->u.reg.memobj != ((mach_port_t) 0)) ? (void) ( 0) : __assert_fail ("np->dn->u.reg.memobj != ((mach_port_t) 0)" , "../../tmpfs/node.c", 503, __PRETTY_FUNCTION__)); | |||||
| 504 | ||||||
| 505 | /* XXX we need to keep a reference to the object, or GNU Mach | |||||
| 506 | will terminate it when we release the map. */ | |||||
| 507 | vm_map (mach_task_self ()((__mach_task_self_ + 0)), &np->dn->u.reg.memref, 4096, 0, 1, | |||||
| 508 | np->dn->u.reg.memobj, 0, 0, VM_PROT_NONE((vm_prot_t) 0x00), VM_PROT_NONE((vm_prot_t) 0x00), | |||||
| 509 | VM_INHERIT_NONE((vm_inherit_t) 2)); | |||||
| 510 | assert_perror (err)(!(err) ? (void) (0) : __assert_perror_fail ((err), "../../tmpfs/node.c" , 510, __PRETTY_FUNCTION__)); | |||||
| 511 | } | |||||
| 512 | ||||||
| 513 | /* XXX always writable */ | |||||
| 514 | ||||||
| 515 | /* Add a reference for each call, the caller will deallocate it. */ | |||||
| 516 | err = mach_port_mod_refs (mach_task_self ()((__mach_task_self_ + 0)), np->dn->u.reg.memobj, | |||||
| 517 | MACH_PORT_RIGHT_SEND((mach_port_right_t) 0), +1); | |||||
| 518 | assert_perror (err)(!(err) ? (void) (0) : __assert_perror_fail ((err), "../../tmpfs/node.c" , 518, __PRETTY_FUNCTION__)); | |||||
| 519 | ||||||
| 520 | return np->dn->u.reg.memobj; | |||||
| 521 | } | |||||
| 522 | ||||||
| 523 | /* The user must define this function. Return a `struct pager *' suitable | |||||
| 524 | for use as an argument to diskfs_register_memory_fault_area that | |||||
| 525 | refers to the pager returned by diskfs_get_filemap for node NP. | |||||
| 526 | NP is locked. */ | |||||
| 527 | struct pager * | |||||
| 528 | diskfs_get_filemap_pager_struct (struct node *np) | |||||
| 529 | { | |||||
| 530 | return 0; | |||||
| 531 | } | |||||
| 532 | ||||||
| 533 | /* We have no pager of our own, so there is no need to worry about | |||||
| 534 | users of it, or to shut it down. */ | |||||
| 535 | int | |||||
| 536 | diskfs_pager_users () | |||||
| 537 | { | |||||
| 538 | return 0; | |||||
| 539 | } | |||||
| 540 | void | |||||
| 541 | diskfs_shutdown_pager () | |||||
| 542 | { | |||||
| 543 | } | |||||
| 544 | ||||||
| 545 | /* The purpose of this is to decide that it's ok to make the fs read-only. | |||||
| 546 | Turning a temporary filesystem read-only seem pretty useless. */ | |||||
| 547 | vm_prot_t | |||||
| 548 | diskfs_max_user_pager_prot () | |||||
| 549 | { | |||||
| 550 | return VM_PROT_READ((vm_prot_t) 0x01); /* Probable lie that lets us go read-only. */ | |||||
| 551 | } | |||||
| 552 | ||||||
| 553 | error_t | |||||
| 554 | diskfs_S_file_get_storage_info (struct protid *cred, | |||||
| 555 | mach_port_t **ports, | |||||
| 556 | mach_msg_type_name_t *ports_type, | |||||
| 557 | mach_msg_type_number_t *num_ports, | |||||
| 558 | int **ints, mach_msg_type_number_t *num_ints, | |||||
| 559 | off_t **offsets, | |||||
| 560 | mach_msg_type_number_t *num_offsets, | |||||
| 561 | char **data, mach_msg_type_number_t *data_len) | |||||
| 562 | { | |||||
| 563 | mach_port_t memobj = diskfs_get_filemap (cred->po->np, VM_PROT_ALL(((vm_prot_t) 0x01)|((vm_prot_t) 0x02)|((vm_prot_t) 0x04))); | |||||
| 564 | if (memobj == MACH_PORT_NULL((mach_port_t) 0)) | |||||
| 565 | return errno(*__errno_location ()); | |||||
| 566 | ||||||
| 567 | assert (*num_ports >= 1)((*num_ports >= 1) ? (void) (0) : __assert_fail ("*num_ports >= 1" , "../../tmpfs/node.c", 567, __PRETTY_FUNCTION__)); /* mig always gives us some */ | |||||
| 568 | *num_ports = 1; | |||||
| 569 | *ports_type = MACH_MSG_TYPE_MOVE_SEND17; | |||||
| 570 | (*ports)[0] | |||||
| 571 | = (cred->po->openstat & O_RDWR(0x0001|0x0002)) == O_RDWR(0x0001|0x0002) ? memobj : MACH_PORT_NULL((mach_port_t) 0); | |||||
| 572 | ||||||
| 573 | assert (*num_offsets >= 2)((*num_offsets >= 2) ? (void) (0) : __assert_fail ("*num_offsets >= 2" , "../../tmpfs/node.c", 573, __PRETTY_FUNCTION__)); /* mig always gives us some */ | |||||
| 574 | *num_offsets = 2; | |||||
| 575 | (*offsets)[0] = 0; | |||||
| 576 | (*offsets)[1] = cred->po->np->dn_stat.st_size; | |||||
| 577 | ||||||
| 578 | assert (*num_ints >= 6)((*num_ints >= 6) ? (void) (0) : __assert_fail ("*num_ints >= 6" , "../../tmpfs/node.c", 578, __PRETTY_FUNCTION__)); /* mig always gives us some */ | |||||
| 579 | *num_ints = 6; | |||||
| 580 | (*ints)[0] = STORAGE_MEMORY; | |||||
| 581 | (*ints)[1] = (cred->po->openstat & O_WRITE0x0002) ? 0 : STORE_READONLY0x0100; | |||||
| 582 | (*ints)[2] = 1; /* block size */ | |||||
| 583 | (*ints)[3] = 1; /* 1 run in offsets list */ | |||||
| 584 | (*ints)[4] = 0; /* name len */ | |||||
| 585 | (*ints)[5] = 0; /* misc len */ | |||||
| 586 | ||||||
| 587 | *data_len = 0; | |||||
| 588 | ||||||
| 589 | return 0; | |||||
| 590 | } |