aafecf0950332e6a6300ef3149225db87e19fa90
[samba.git] / source4 / ntvfs / posix / pvfs_fileinfo.c
1 /* 
2    Unix SMB/CIFS implementation.
3
4    POSIX NTVFS backend - 
5
6    Copyright (C) Andrew Tridgell 2004
7
8    This program is free software; you can redistribute it and/or modify
9    it under the terms of the GNU General Public License as published by
10    the Free Software Foundation; either version 2 of the License, or
11    (at your option) any later version.
12    
13    This program is distributed in the hope that it will be useful,
14    but WITHOUT ANY WARRANTY; without even the implied warranty of
15    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16    GNU General Public License for more details.
17    
18    You should have received a copy of the GNU General Public License
19    along with this program; if not, write to the Free Software
20    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 */
22
23 #include "includes.h"
24 #include "vfs_posix.h"
25
26
27 /* UNIX filetype mappings. */
28 #define UNIX_TYPE_FILE 0
29 #define UNIX_TYPE_DIR 1
30 #define UNIX_TYPE_SYMLINK 2
31 #define UNIX_TYPE_CHARDEV 3
32 #define UNIX_TYPE_BLKDEV 4
33 #define UNIX_TYPE_FIFO 5
34 #define UNIX_TYPE_SOCKET 6
35 #define UNIX_TYPE_UNKNOWN 0xFFFFFFFF
36
37
38 /****************************************************************************
39  Change a unix mode to a dos mode.
40 ****************************************************************************/
41 static uint32_t dos_mode_from_stat(struct pvfs_state *pvfs, struct stat *st)
42 {
43         int result = 0;
44
45         if ((st->st_mode & S_IWUSR) == 0)
46                 result |= FILE_ATTRIBUTE_READONLY;
47         
48         if ((pvfs->flags & PVFS_FLAG_MAP_ARCHIVE) && ((st->st_mode & S_IXUSR) != 0))
49                 result |= FILE_ATTRIBUTE_ARCHIVE;
50         
51         if ((pvfs->flags & PVFS_FLAG_MAP_SYSTEM) && ((st->st_mode & S_IXGRP) != 0))
52                 result |= FILE_ATTRIBUTE_SYSTEM;
53         
54         if ((pvfs->flags & PVFS_FLAG_MAP_HIDDEN) && ((st->st_mode & S_IXOTH) != 0))
55                 result |= FILE_ATTRIBUTE_HIDDEN;
56   
57         if (S_ISDIR(st->st_mode))
58                 result = FILE_ATTRIBUTE_DIRECTORY | (result & FILE_ATTRIBUTE_READONLY);
59
60         if (!(result & 
61               (FILE_ATTRIBUTE_READONLY|
62                FILE_ATTRIBUTE_ARCHIVE|
63                FILE_ATTRIBUTE_SYSTEM|
64                FILE_ATTRIBUTE_HIDDEN|
65                FILE_ATTRIBUTE_DIRECTORY))) {
66                 result |= FILE_ATTRIBUTE_NORMAL;
67         }
68  
69         return result;
70 }
71
72
73
74 /*
75   fill in the dos file attributes for a file
76 */
77 NTSTATUS pvfs_fill_dos_info(struct pvfs_state *pvfs, struct pvfs_filename *name, int fd)
78 {
79         /* make directories appear as size 0 */
80         if (S_ISDIR(name->st.st_mode)) {
81                 name->st.st_size = 0;
82         }
83
84         /* for now just use the simple samba mapping */
85         unix_to_nt_time(&name->dos.create_time, name->st.st_ctime);
86         unix_to_nt_time(&name->dos.access_time, name->st.st_atime);
87         unix_to_nt_time(&name->dos.write_time,  name->st.st_mtime);
88         unix_to_nt_time(&name->dos.change_time, name->st.st_ctime);
89 #ifdef HAVE_STAT_TV_NSEC
90         name->dos.create_time += name->st.st_ctim.tv_nsec / 100;
91         name->dos.access_time += name->st.st_atim.tv_nsec / 100;
92         name->dos.write_time  += name->st.st_mtim.tv_nsec / 100;
93         name->dos.change_time += name->st.st_ctim.tv_nsec / 100;
94 #endif
95         name->dos.attrib = dos_mode_from_stat(pvfs, &name->st);
96         name->dos.alloc_size = pvfs_round_alloc_size(pvfs, name->st.st_size);
97         name->dos.nlink = name->st.st_nlink;
98         name->dos.ea_size = 4;
99         name->dos.file_id = (((uint64_t)name->st.st_dev)<<32) | name->st.st_ino;
100         name->dos.flags = 0;
101
102         return pvfs_dosattrib_load(pvfs, name, fd);
103 }
104
105
106 /*
107   return a set of unix file permissions for a new file or directory
108 */
109 mode_t pvfs_fileperms(struct pvfs_state *pvfs, uint32 attrib)
110 {
111         mode_t mode = S_IRUSR | S_IRGRP | S_IROTH;
112
113         if (attrib & FILE_ATTRIBUTE_DIRECTORY) {
114                 mode |= S_IXUSR | S_IXGRP | S_IXOTH;
115         }
116
117         if (!(attrib & FILE_ATTRIBUTE_READONLY) ||
118             (pvfs->flags & PVFS_FLAG_XATTR_ENABLE)) {
119                 mode |= S_IWUSR;
120         }
121
122         if (!(pvfs->flags & PVFS_FLAG_XATTR_ENABLE)) {
123                 if ((attrib & FILE_ATTRIBUTE_ARCHIVE) &&
124                     (pvfs->flags & PVFS_FLAG_MAP_ARCHIVE)) {
125                         mode |= S_IXUSR;
126                 }
127                 
128                 if ((attrib & FILE_ATTRIBUTE_SYSTEM) &&
129                     (pvfs->flags & PVFS_FLAG_MAP_SYSTEM)) {
130                         mode |= S_IXGRP;
131                 }
132                 
133                 if ((attrib & FILE_ATTRIBUTE_HIDDEN) &&
134                     (pvfs->flags & PVFS_FLAG_MAP_HIDDEN)) {
135                         mode |= S_IXOTH;
136                 }
137         }
138
139         return mode;
140 }
141