/* Private declarations for fileserver library Copyright (C) 1994 Free Software Foundation This program 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 2, or (at your option) any later version. This program 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 this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifndef DISKFS_PRIV_H #define DISKFS_PRIV_H #include #include #include #include #include #include #include "diskfs.h" extern mach_port_t fs_control_port; /* receive right */ enum porttype { PT_NONE, PT_PROTID, PT_PAGER, PT_CTL, PT_EXECBOOT, PT_INITBOOT, PT_TRANSBOOT, }; spin_lock_t _diskfs_node_refcnt_lock = SPIN_LOCK_INITIALIZER; volatile struct mapped_time_value *_diskfs_mtime; /* Needed for MiG. */ typedef struct protid *protid_t; /* Called by MiG to translate ports into struct protid *. fsmutations.h arranges for this to happen for the io and fs interfaces. */ extern inline struct protid * begin_using_protid_port (file_t port) { return ports_check_port_type (port, PT_PROTID); } /* Called by MiG after server routines have been run; this balances begin_using_protid_port, and is arranged for the io and fs interfaces by fsmutations.h. */ extern inline void end_using_protid_port (struct protid *cred) { ports_done_with_port (cred); } /* Add a reference to a node. */ extern inline void diskfs_nref (struct node *np) { spin_lock (&_diskfs_node_refcnt_lock); np->references++; spin_unlock (&_diskfs_node_refcnt_lock); } /* Unlock node NP and release a reference */ extern inline void diskfs_nput (struct node *np) { spin_lock (&_diskfs_node_refcnt_lock); np->references--; if (np->references == 0) { diskfs_drop_node (np); spin_unlock (&_diskfs_node_refcnt_lock); } else { spin_unlock (&_diskfs_node_refcnt_lock); mutex_unlock (&np->lock); } } /* Release a reference on NP. If NP is locked by anyone, then this cannot be the last reference (because you must hold a reference in order to hold the lock). */ extern inline void diskfs_nrele (struct node *np) { spin_lock (&_diskfs_node_refcnt_lock); np->references--; if (np->references == 0) diskfs_drop_node (np); spin_unlock (&_diskfs_node_refcnt_lock); } /* Actually read or write a file. The file size must already permit the requested access. NP is the file to read/write. DATA is a buffer to write from or fill on read. OFFSET is the absolute address (-1 not permitted here); AMT is the size of the read/write to perform; DIR is set for writing and clear for reading. The inode must be locked. */ error_t _diskfs_rdwr_internal (struct node *np, char *data, int offset, int amt, int dir); /* This macro locks the node associated with PROTID, and then evaluates the expression OPERATION; then it syncs the inode (without waiting) and unlocks everything, and then returns the value `err' (which can be set by OPERATION if desired). */ #define CHANGE_NODE_FIELD(PROTID, OPERATION) \ ({ \ error_t err = 0; \ struct node *np; \ \ if (!(PROTID)) \ return EOPNOTSUPP; \ \ if (diskfs_readonly) \ return EROFS; \ \ np = (PROTID)->po->np; \ \ mutex_lock (&np->lock); \ (OPERATION); \ diskfs_node_update (np, 0); \ mutex_unlock (&np->lock); \ return err; \ }) #define HONORED_STATE_MODES (O_APPEND|O_ASYNC|O_FSYNC|O_NONBLOCK) #endif