/[rdesktop]/sourceforge.net/trunk/rdesktop/disk.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /sourceforge.net/trunk/rdesktop/disk.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 596 by n-ki, Tue Feb 3 14:03:42 2004 UTC revision 986 by astrand, Wed Aug 24 11:31:58 2005 UTC
# Line 18  Line 18 
18     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19  */  */
20    
21  #define FILE_ATTRIBUTE_READONLY                 0x00000001  #include "disk.h"
 #define FILE_ATTRIBUTE_HIDDEN                   0x00000002  
 #define FILE_ATTRIBUTE_SYSTEM                   0x00000004  
 #define FILE_ATTRIBUTE_DIRECTORY                0x00000010  
 #define FILE_ATTRIBUTE_ARCHIVE                  0x00000020  
 #define FILE_ATTRIBUTE_DEVICE                   0x00000040  
 #define FILE_ATTRIBUTE_NORMAL                   0x00000080  
 #define FILE_ATTRIBUTE_TEMPORARY                0x00000100  
 #define FILE_ATTRIBUTE_SPARSE_FILE              0x00000200  
 #define FILE_ATTRIBUTE_REPARSE_POINT            0x00000400  
 #define FILE_ATTRIBUTE_COMPRESSED               0x00000800  
 #define FILE_ATTRIBUTE_OFFLINE                  0x00001000  
 #define FILE_ATTRIBUTE_NOT_CONTENT_INDEXED      0x00002000  
 #define FILE_ATTRIBUTE_ENCRYPTED                0x00004000  
   
 #define FILE_BASIC_INFORMATION                  0x04  
 #define FILE_STANDARD_INFORMATION               0x05  
   
 #define FS_CASE_SENSITIVE                       0x00000001  
 #define FS_CASE_IS_PRESERVED                    0x00000002  
 #define FS_UNICODE_STORED_ON_DISK               0x00000004  
 #define FS_PERSISTENT_ACLS                      0x00000008  
 #define FS_FILE_COMPRESSION                     0x00000010  
 #define FS_VOLUME_QUOTAS                        0x00000020  
 #define FS_SUPPORTS_SPARSE_FILES                0x00000040  
 #define FS_SUPPORTS_REPARSE_POINTS              0x00000080  
 #define FS_SUPPORTS_REMOTE_STORAGE              0X00000100  
 #define FS_VOL_IS_COMPRESSED                    0x00008000  
 #define FILE_READ_ONLY_VOLUME                   0x00080000  
   
 #define OPEN_EXISTING                           1  
 #define CREATE_NEW                              2  
 #define OPEN_ALWAYS                             3  
 #define TRUNCATE_EXISTING                       4  
 #define CREATE_ALWAYS                           5  
   
 #define GENERIC_READ                            0x80000000  
 #define GENERIC_WRITE                           0x40000000  
 #define GENERIC_EXECUTE                         0x20000000  
 #define GENERIC_ALL                             0x10000000  
   
 #define ERROR_FILE_NOT_FOUND                    2L  
 #define ERROR_ALREADY_EXISTS                    183L  
   
 #define MAX_OPEN_FILES  0x100  
   
 #if (defined(sun) && (defined(__svr4__) || defined(__SVR4)))  
 #define SOLARIS  
 #endif  
   
 #ifdef SOLARIS  
 #define DIRFD(a) ((a)->dd_fd)  
 #else  
 #define DIRFD(a) (dirfd(a))  
 #endif  
22    
23  #include <sys/types.h>  #include <sys/types.h>
24  #include <sys/stat.h>  #include <sys/stat.h>
# Line 81  Line 27 
27  #include <dirent.h>             /* opendir, closedir, readdir */  #include <dirent.h>             /* opendir, closedir, readdir */
28  #include <fnmatch.h>  #include <fnmatch.h>
29  #include <errno.h>              /* errno */  #include <errno.h>              /* errno */
30    #include <stdio.h>
31    
32  #if defined(SOLARIS)  #include <utime.h>
33  #include <sys/statvfs.h>        /* solaris statvfs */  #include <time.h>               /* ctime */
34  #define STATFS_FN(path, buf) (statvfs(path,buf))  
35  #define STATFS_T statvfs  #if (defined(HAVE_DIRFD) || (HAVE_DECL_DIRFD == 1))
36  #define F_NAMELEN(buf) ((buf).f_namemax)  #define DIRFD(a) (dirfd(a))
37    #else
38    #define DIRFD(a) ((a)->DIR_FD_MEMBER_NAME)
39    #endif
40    
41    /* TODO: Fix mntent-handling for solaris
42     * #include <sys/mntent.h> */
43    #if (defined(HAVE_MNTENT_H) && defined(HAVE_SETMNTENT))
44    #include <mntent.h>
45    #define MNTENT_PATH "/etc/mtab"
46    #define USE_SETMNTENT
47    #endif
48    
49    #ifdef HAVE_SYS_VFS_H
50    #include <sys/vfs.h>
51    #endif
52    
53    #ifdef HAVE_SYS_STATVFS_H
54    #include <sys/statvfs.h>
55    #endif
56    
57    #ifdef HAVE_SYS_STATFS_H
58    #include <sys/statfs.h>
59    #endif
60    
61  #elif (defined(__OpenBSD__) || defined(__NetBSD__) || defined(__FreeBSD__))  #ifdef HAVE_SYS_PARAM_H
62  #include <sys/param.h>  #include <sys/param.h>
63    #endif
64    
65    #ifdef HAVE_SYS_MOUNT_H
66  #include <sys/mount.h>  #include <sys/mount.h>
67  #define STATFS_FN(path, buf) (statfs(path,buf))  #endif
68    
69    #include "rdesktop.h"
70    
71    #ifdef STAT_STATFS3_OSF1
72    #define STATFS_FN(path, buf) (statfs(path,buf,sizeof(buf)))
73  #define STATFS_T statfs  #define STATFS_T statfs
74  #define F_NAMELEN(buf) (NAME_MAX)  #define USE_STATFS
75    #endif
76    
77  #else  #ifdef STAT_STATVFS
78  #include <sys/vfs.h>            /* linux statfs */  #define STATFS_FN(path, buf) (statvfs(path,buf))
79    #define STATFS_T statvfs
80    #define USE_STATVFS
81    #endif
82    
83    #ifdef STAT_STATVFS64
84    #define STATFS_FN(path, buf) (statvfs64(path,buf))
85    #define STATFS_T statvfs64
86    #define USE_STATVFS
87    #endif
88    
89    #if (defined(STAT_STATFS2_FS_DATA) || defined(STAT_STATFS2_BSIZE) || defined(STAT_STATFS2_FSIZE))
90  #define STATFS_FN(path, buf) (statfs(path,buf))  #define STATFS_FN(path, buf) (statfs(path,buf))
91  #define STATFS_T statfs  #define STATFS_T statfs
92    #define USE_STATFS
93    #endif
94    
95    #ifdef STAT_STATFS4
96    #define STATFS_FN(path, buf) (statfs(path,buf,sizeof(buf),0))
97    #define STATFS_T statfs
98    #define USE_STATFS
99    #endif
100    
101    #if ((defined(USE_STATFS) && defined(HAVE_STRUCT_STATFS_F_NAMEMAX)) || (defined(USE_STATVFS) && defined(HAVE_STRUCT_STATVFS_F_NAMEMAX)))
102    #define F_NAMELEN(buf) ((buf).f_namemax)
103    #endif
104    
105    #if ((defined(USE_STATFS) && defined(HAVE_STRUCT_STATFS_F_NAMELEN)) || (defined(USE_STATVFS) && defined(HAVE_STRUCT_STATVFS_F_NAMELEN)))
106  #define F_NAMELEN(buf) ((buf).f_namelen)  #define F_NAMELEN(buf) ((buf).f_namelen)
107  #endif  #endif
108    
109  #include "rdesktop.h"  #ifndef F_NAMELEN
110    #define F_NAMELEN(buf) (255)
111    #endif
112    
113    /* Dummy statfs fallback */
114    #ifndef STATFS_T
115    struct dummy_statfs_t
116    {
117            long f_bfree;
118            long f_bsize;
119            long f_blocks;
120            int f_namelen;
121            int f_namemax;
122    };
123    
124    static int
125    dummy_statfs(struct dummy_statfs_t *buf)
126    {
127            buf->f_blocks = 262144;
128            buf->f_bfree = 131072;
129            buf->f_bsize = 512;
130            buf->f_namelen = 255;
131            buf->f_namemax = 255;
132    
133            return 0;
134    }
135    
136    #define STATFS_T dummy_statfs_t
137    #define STATFS_FN(path,buf) (dummy_statfs(buf))
138    #endif
139    
140  extern RDPDR_DEVICE g_rdpdr_device[];  extern RDPDR_DEVICE g_rdpdr_device[];
141    
142  struct fileinfo  FILEINFO g_fileinfo[MAX_OPEN_FILES];
143    BOOL g_notify_stamp = False;
144    
145    typedef struct
146  {  {
147          uint32 device_id, flags_and_attributes;          char name[PATH_MAX];
148          char path[256];          char label[PATH_MAX];
149          DIR *pdir;          unsigned long serial;
150          struct dirent *pdirent;          char type[PATH_MAX];
151          char pattern[64];  } FsInfoType;
152          BOOL delete_on_close;  
153    static NTSTATUS NotifyInfo(NTHANDLE handle, uint32 info_class, NOTIFY * p);
154    
155    static time_t
156    get_create_time(struct stat *st)
157    {
158            time_t ret, ret1;
159    
160            ret = MIN(st->st_ctime, st->st_mtime);
161            ret1 = MIN(ret, st->st_atime);
162    
163            if (ret1 != (time_t) 0)
164                    return ret1;
165    
166            return ret;
167  }  }
 g_fileinfo[MAX_OPEN_FILES];  
168    
169  /* Convert seconds since 1970 to a filetime */  /* Convert seconds since 1970 to a filetime */
170  void  static void
171  seconds_since_1970_to_filetime(time_t seconds, uint32 * high, uint32 * low)  seconds_since_1970_to_filetime(time_t seconds, uint32 * high, uint32 * low)
172  {  {
173          unsigned long long ticks;          unsigned long long ticks;
# Line 128  seconds_since_1970_to_filetime(time_t se Line 177  seconds_since_1970_to_filetime(time_t se
177          *high = (uint32) (ticks >> 32);          *high = (uint32) (ticks >> 32);
178  }  }
179    
180    /* Convert seconds since 1970 back to filetime */
181    static time_t
182    convert_1970_to_filetime(uint32 high, uint32 low)
183    {
184            unsigned long long ticks;
185            time_t val;
186    
187            ticks = low + (((unsigned long long) high) << 32);
188            ticks /= 10000000;
189            ticks -= 11644473600LL;
190    
191            val = (time_t) ticks;
192            return (val);
193    
194    }
195    
196    /* A wrapper for ftruncate which supports growing files, even if the
197       native ftruncate doesn't. This is needed on Linux FAT filesystems,
198       for example. */
199    static int
200    ftruncate_growable(int fd, off_t length)
201    {
202            int ret;
203            off_t pos;
204            static const char zero;
205    
206            /* Try the simple method first */
207            if ((ret = ftruncate(fd, length)) != -1)
208            {
209                    return ret;
210            }
211    
212            /*
213             * Some kind of error. Perhaps we were trying to grow. Retry
214             * in a safe way.
215             */
216    
217            /* Get current position */
218            if ((pos = lseek(fd, 0, SEEK_CUR)) == -1)
219            {
220                    perror("lseek");
221                    return -1;
222            }
223    
224            /* Seek to new size */
225            if (lseek(fd, length, SEEK_SET) == -1)
226            {
227                    perror("lseek");
228                    return -1;
229            }
230    
231            /* Write a zero */
232            if (write(fd, &zero, 1) == -1)
233            {
234                    perror("write");
235                    return -1;
236            }
237    
238            /* Truncate. This shouldn't fail. */
239            if (ftruncate(fd, length) == -1)
240            {
241                    perror("ftruncate");
242                    return -1;
243            }
244    
245            /* Restore position */
246            if (lseek(fd, pos, SEEK_SET) == -1)
247            {
248                    perror("lseek");
249                    return -1;
250            }
251    
252            return 0;
253    }
254    
255    /* Just like open(2), but if a open with O_EXCL fails, retry with
256       GUARDED semantics. This might be necessary because some filesystems
257       (such as NFS filesystems mounted from a unfsd server) doesn't
258       support O_EXCL. GUARDED semantics are subject to race conditions,
259       but we can live with that.
260    */
261    static int
262    open_weak_exclusive(const char *pathname, int flags, mode_t mode)
263    {
264            int ret;
265            struct stat statbuf;
266    
267            ret = open(pathname, flags, mode);
268            if (ret != -1 || !(flags & O_EXCL))
269            {
270                    /* Success, or not using O_EXCL */
271                    return ret;
272            }
273    
274            /* An error occured, and we are using O_EXCL. In case the FS
275               doesn't support O_EXCL, some kind of error will be
276               returned. Unfortunately, we don't know which one. Linux
277               2.6.8 seems to return 524, but I cannot find a documented
278               #define for this case. So, we'll return only on errors that
279               we know aren't related to O_EXCL. */
280            switch (errno)
281            {
282                    case EACCES:
283                    case EEXIST:
284                    case EINTR:
285                    case EISDIR:
286                    case ELOOP:
287                    case ENAMETOOLONG:
288                    case ENOENT:
289                    case ENOTDIR:
290                            return ret;
291            }
292    
293            /* Retry with GUARDED semantics */
294            if (stat(pathname, &statbuf) != -1)
295            {
296                    /* File exists */
297                    errno = EEXIST;
298                    return -1;
299            }
300            else
301            {
302                    return open(pathname, flags & ~O_EXCL, mode);
303            }
304    }
305    
306  /* Enumeration of devices from rdesktop.c        */  /* Enumeration of devices from rdesktop.c        */
307  /* returns numer of units found and initialized. */  /* returns numer of units found and initialized. */
308  /* optarg looks like ':h:=/mnt/floppy,b:=/mnt/usbdevice1' */  /* optarg looks like ':h=/mnt/floppy,b=/mnt/usbdevice1' */
309  /* when it arrives to this function.             */  /* when it arrives to this function.             */
310  int  int
311  disk_enum_devices(int *id, char *optarg)  disk_enum_devices(uint32 * id, char *optarg)
312  {  {
313          char *pos = optarg;          char *pos = optarg;
314          char *pos2;          char *pos2;
315          int count = 0;          int count = 0;
316    
317          // skip the first colon          /* skip the first colon */
318          optarg++;          optarg++;
319          while ((pos = next_arg(optarg, ',')) && *id < RDPDR_MAX_DEVICES)          while ((pos = next_arg(optarg, ',')) && *id < RDPDR_MAX_DEVICES)
320          {          {
321                  pos2 = next_arg(optarg, '=');                  pos2 = next_arg(optarg, '=');
                 strcpy(g_rdpdr_device[*id].name, optarg);  
   
                 toupper_str(g_rdpdr_device[*id].name);  
322    
323                  /* add trailing colon to name. */                  strncpy(g_rdpdr_device[*id].name, optarg, sizeof(g_rdpdr_device[*id].name) - 1);
324                  strcat(g_rdpdr_device[*id].name, ":");                  if (strlen(optarg) > (sizeof(g_rdpdr_device[*id].name) - 1))
325                            fprintf(stderr, "share name %s truncated to %s\n", optarg,
326                                    g_rdpdr_device[*id].name);
327    
328                  g_rdpdr_device[*id].local_path = xmalloc(strlen(pos2) + 1);                  g_rdpdr_device[*id].local_path = xmalloc(strlen(pos2) + 1);
329                  strcpy(g_rdpdr_device[*id].local_path, pos2);                  strcpy(g_rdpdr_device[*id].local_path, pos2);
                 printf("DISK %s to %s\n", g_rdpdr_device[*id].name, g_rdpdr_device[*id].local_path);  
330                  g_rdpdr_device[*id].device_type = DEVICE_TYPE_DISK;                  g_rdpdr_device[*id].device_type = DEVICE_TYPE_DISK;
331                  count++;                  count++;
332                  (*id)++;                  (*id)++;
# Line 164  disk_enum_devices(int *id, char *optarg) Line 337  disk_enum_devices(int *id, char *optarg)
337  }  }
338    
339  /* Opens or creates a file or directory */  /* Opens or creates a file or directory */
340  NTSTATUS  static NTSTATUS
341  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,
342              uint32 flags_and_attributes, char *filename, HANDLE * phandle)              uint32 flags_and_attributes, char *filename, NTHANDLE * phandle)
343  {  {
344          HANDLE handle;          NTHANDLE handle;
345          DIR *dirp;          DIR *dirp;
346          int flags, mode;          int flags, mode;
347          char path[256];          char path[PATH_MAX];
348            struct stat filestat;
349    
350          handle = 0;          handle = 0;
351          dirp = NULL;          dirp = NULL;
352          flags = 0;          flags = 0;
353          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
354    
355          if (filename[strlen(filename) - 1] == '/')          if (*filename && filename[strlen(filename) - 1] == '/')
356                  filename[strlen(filename) - 1] = 0;                  filename[strlen(filename) - 1] = 0;
357          sprintf(path, "%s%s", g_rdpdr_device[device_id].local_path, filename);          sprintf(path, "%s%s", g_rdpdr_device[device_id].local_path, filename);
358    
# Line 186  disk_create(uint32 device_id, uint32 acc Line 360  disk_create(uint32 device_id, uint32 acc
360          {          {
361                  case CREATE_ALWAYS:                  case CREATE_ALWAYS:
362    
363                          // Delete existing file/link.                          /* Delete existing file/link. */
364                          unlink(path);                          unlink(path);
365                          flags |= O_CREAT;                          flags |= O_CREAT;
366                          break;                          break;
367    
368                  case CREATE_NEW:                  case CREATE_NEW:
369    
370                          // If the file already exists, then fail.                          /* If the file already exists, then fail. */
371                          flags |= O_CREAT | O_EXCL;                          flags |= O_CREAT | O_EXCL;
372                          break;                          break;
373    
374                  case OPEN_ALWAYS:                  case OPEN_ALWAYS:
375    
376                          // Create if not already exists.                          /* Create if not already exists. */
377                          flags |= O_CREAT;                          flags |= O_CREAT;
378                          break;                          break;
379    
380                  case OPEN_EXISTING:                  case OPEN_EXISTING:
381    
382                          // Default behaviour                          /* Default behaviour */
383                          break;                          break;
384    
385                  case TRUNCATE_EXISTING:                  case TRUNCATE_EXISTING:
386    
387                          // If the file does not exist, then fail.                          /* If the file does not exist, then fail. */
388                          flags |= O_TRUNC;                          flags |= O_TRUNC;
389                          break;                          break;
390          }          }
391    
392          /* the sad part is that we can't trust this flag   */          /*printf("Open: \"%s\"  flags: %X, accessmask: %X sharemode: %X create disp: %X\n", path, flags_and_attributes, accessmask, sharemode, create_disposition); */
393          /* directories aren't always marked                */  
394          if (flags_and_attributes ^ FILE_DIRECTORY_FILE)          /* Get information about file and set that flag ourselfs */
395            if ((stat(path, &filestat) == 0) && (S_ISDIR(filestat.st_mode)))
396          {          {
397                  if (accessmask & GENERIC_ALL                  if (flags_and_attributes & FILE_NON_DIRECTORY_FILE)
398                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))                          return STATUS_FILE_IS_A_DIRECTORY;
                 {  
                         flags |= O_RDWR;  
                 }  
                 else if ((accessmask & GENERIC_WRITE) && !(accessmask & GENERIC_READ))  
                 {  
                         flags |= O_WRONLY;  
                 }  
399                  else                  else
400                            flags_and_attributes |= FILE_DIRECTORY_FILE;
401            }
402    
403            if (flags_and_attributes & FILE_DIRECTORY_FILE)
404            {
405                    if (flags & O_CREAT)
406                  {                  {
407                          flags |= O_RDONLY;                          mkdir(path, mode);
408                  }                  }
409    
410                  handle = open(path, flags, mode);                  dirp = opendir(path);
411                  if (handle == -1)                  if (!dirp)
412                  {                  {
413                          switch (errno)                          switch (errno)
414                          {                          {
# Line 248  disk_create(uint32 device_id, uint32 acc Line 422  disk_create(uint32 device_id, uint32 acc
422    
423                                  default:                                  default:
424    
425                                          perror("open");                                          perror("opendir");
426                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
427                          }                          }
428                  }                  }
429                    handle = DIRFD(dirp);
                 /* all read and writes of files should be non blocking */  
                 if (fcntl(handle, F_SETFL, O_NONBLOCK) == -1)  
                         perror("fcntl");  
430          }          }
431            else
         /* since we can't trust the FILE_DIRECTORY_FILE flag */  
         /* we need to double check that the file isn't a dir */  
         if (handle != 0)  
432          {          {
                 /* Must check if this file isn't actually a directory */  
                 struct stat filestat;  
433    
434                  // Get information about file and set that flag ourselfs                  if (accessmask & GENERIC_ALL
435                  if ((fstat(handle, &filestat) == 0) && (S_ISDIR(filestat.st_mode)))                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))
436                  {                  {
437                          flags_and_attributes |= FILE_DIRECTORY_FILE;                          flags |= O_RDWR;
                         close(handle);  
                         handle = 0;  
438                  }                  }
439          }                  else if ((accessmask & GENERIC_WRITE) && !(accessmask & GENERIC_READ))
   
   
         if (flags_and_attributes & FILE_DIRECTORY_FILE)  
         {  
                 if (flags & O_CREAT)  
440                  {                  {
441                          mkdir(path, mode);                          flags |= O_WRONLY;
442                    }
443                    else
444                    {
445                            flags |= O_RDONLY;
446                  }                  }
447    
448                  dirp = opendir(path);                  handle = open_weak_exclusive(path, flags, mode);
449                  if (!dirp)                  if (handle == -1)
450                  {                  {
451                          switch (errno)                          switch (errno)
452                          {                          {
453                                    case EISDIR:
454    
455                                            return STATUS_FILE_IS_A_DIRECTORY;
456    
457                                  case EACCES:                                  case EACCES:
458    
459                                          return STATUS_ACCESS_DENIED;                                          return STATUS_ACCESS_DENIED;
# Line 294  disk_create(uint32 device_id, uint32 acc Line 461  disk_create(uint32 device_id, uint32 acc
461                                  case ENOENT:                                  case ENOENT:
462    
463                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
464                                    case EEXIST:
465    
466                                            return STATUS_OBJECT_NAME_COLLISION;
467                                  default:                                  default:
468    
469                                          perror("opendir");                                          perror("open");
470                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
471                          }                          }
472                  }                  }
473                  handle = DIRFD(dirp);  
474                    /* all read and writes of files should be non blocking */
475                    if (fcntl(handle, F_SETFL, O_NONBLOCK) == -1)
476                            perror("fcntl");
477          }          }
478    
479          if (handle >= MAX_OPEN_FILES)          if (handle >= MAX_OPEN_FILES)
# Line 313  disk_create(uint32 device_id, uint32 acc Line 485  disk_create(uint32 device_id, uint32 acc
485    
486          if (dirp)          if (dirp)
487                  g_fileinfo[handle].pdir = dirp;                  g_fileinfo[handle].pdir = dirp;
488            else
489                    g_fileinfo[handle].pdir = NULL;
490    
491          g_fileinfo[handle].device_id = device_id;          g_fileinfo[handle].device_id = device_id;
492          g_fileinfo[handle].flags_and_attributes = flags_and_attributes;          g_fileinfo[handle].flags_and_attributes = flags_and_attributes;
493  //      printf("create: attrib: %u handle %u\n", g_fileinfo[handle].flags_and_attributes, handle );          g_fileinfo[handle].accessmask = accessmask;
494          strncpy(g_fileinfo[handle].path, path, 255);          strncpy(g_fileinfo[handle].path, path, PATH_MAX - 1);
495            g_fileinfo[handle].delete_on_close = False;
496            g_notify_stamp = True;
497    
498          *phandle = handle;          *phandle = handle;
499          return STATUS_SUCCESS;          return STATUS_SUCCESS;
500  }  }
501    
502  NTSTATUS  static NTSTATUS
503  disk_close(HANDLE handle)  disk_close(NTHANDLE handle)
504  {  {
505          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
506    
507          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
508    
509          if (pfinfo->flags_and_attributes & FILE_DIRECTORY_FILE)          g_notify_stamp = True;
510    
511            rdpdr_abort_io(handle, 0, STATUS_CANCELLED);
512    
513            if (pfinfo->pdir)
514          {          {
515                  closedir(pfinfo->pdir);                  if (closedir(pfinfo->pdir) < 0)
516                  //FIXME: Should check exit code                  {
517                            perror("closedir");
518                            return STATUS_INVALID_HANDLE;
519                    }
520    
521                    if (pfinfo->delete_on_close)
522                            if (rmdir(pfinfo->path) < 0)
523                            {
524                                    perror(pfinfo->path);
525                                    return STATUS_ACCESS_DENIED;
526                            }
527                    pfinfo->delete_on_close = False;
528          }          }
529          else          else
530          {          {
531                  close(handle);                  if (close(handle) < 0)
532                    {
533                            perror("close");
534                            return STATUS_INVALID_HANDLE;
535                    }
536                    if (pfinfo->delete_on_close)
537                            if (unlink(pfinfo->path) < 0)
538                            {
539                                    perror(pfinfo->path);
540                                    return STATUS_ACCESS_DENIED;
541                            }
542    
543                    pfinfo->delete_on_close = False;
544          }          }
545    
546          return STATUS_SUCCESS;          return STATUS_SUCCESS;
547  }  }
548    
549  NTSTATUS  static NTSTATUS
550  disk_read(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_read(NTHANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
551  {  {
552          int n;          int n;
553    
554    #if 0
555          /* browsing dir ????        */          /* browsing dir ????        */
556          /* each request is 24 bytes */          /* each request is 24 bytes */
557          if (g_fileinfo[handle].flags_and_attributes & FILE_DIRECTORY_FILE)          if (g_fileinfo[handle].flags_and_attributes & FILE_DIRECTORY_FILE)
# Line 354  disk_read(HANDLE handle, uint8 * data, u Line 559  disk_read(HANDLE handle, uint8 * data, u
559                  *result = 0;                  *result = 0;
560                  return STATUS_SUCCESS;                  return STATUS_SUCCESS;
561          }          }
562    #endif
563    
564            lseek(handle, offset, SEEK_SET);
565    
         if (offset)  
                 lseek(handle, offset, SEEK_SET);  
566          n = read(handle, data, length);          n = read(handle, data, length);
567    
568          if (n < 0)          if (n < 0)
569          {          {
                 perror("read");  
570                  *result = 0;                  *result = 0;
571                  return STATUS_INVALID_PARAMETER;                  switch (errno)
572                    {
573                            case EISDIR:
574                                    /* Implement 24 Byte directory read ??
575                                       with STATUS_NOT_IMPLEMENTED server doesn't read again */
576                                    /* return STATUS_FILE_IS_A_DIRECTORY; */
577                                    return STATUS_NOT_IMPLEMENTED;
578                            default:
579                                    perror("read");
580                                    return STATUS_INVALID_PARAMETER;
581                    }
582          }          }
583    
584          *result = n;          *result = n;
# Line 371  disk_read(HANDLE handle, uint8 * data, u Line 586  disk_read(HANDLE handle, uint8 * data, u
586          return STATUS_SUCCESS;          return STATUS_SUCCESS;
587  }  }
588    
589  NTSTATUS  static NTSTATUS
590  disk_write(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_write(NTHANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
591  {  {
592          int n;          int n;
593    
594          if (offset)          lseek(handle, offset, SEEK_SET);
                 lseek(handle, offset, SEEK_SET);  
595    
596          n = write(handle, data, length);          n = write(handle, data, length);
597    
# Line 385  disk_write(HANDLE handle, uint8 * data, Line 599  disk_write(HANDLE handle, uint8 * data,
599          {          {
600                  perror("write");                  perror("write");
601                  *result = 0;                  *result = 0;
602                  return STATUS_ACCESS_DENIED;                  switch (errno)
603                    {
604                            case ENOSPC:
605                                    return STATUS_DISK_FULL;
606                            default:
607                                    return STATUS_ACCESS_DENIED;
608                    }
609          }          }
610    
611          *result = n;          *result = n;
# Line 394  disk_write(HANDLE handle, uint8 * data, Line 614  disk_write(HANDLE handle, uint8 * data,
614  }  }
615    
616  NTSTATUS  NTSTATUS
617  disk_query_information(HANDLE handle, uint32 info_class, STREAM out)  disk_query_information(NTHANDLE handle, uint32 info_class, STREAM out)
618  {  {
619          uint32 file_attributes, ft_high, ft_low;          uint32 file_attributes, ft_high, ft_low;
620          struct stat filestat;          struct stat filestat;
# Line 402  disk_query_information(HANDLE handle, ui Line 622  disk_query_information(HANDLE handle, ui
622    
623          path = g_fileinfo[handle].path;          path = g_fileinfo[handle].path;
624    
625          // Get information about file          /* Get information about file */
626          if (fstat(handle, &filestat) != 0)          if (fstat(handle, &filestat) != 0)
627          {          {
628                  perror("stat");                  perror("stat");
# Line 410  disk_query_information(HANDLE handle, ui Line 630  disk_query_information(HANDLE handle, ui
630                  return STATUS_ACCESS_DENIED;                  return STATUS_ACCESS_DENIED;
631          }          }
632    
633          // Set file attributes          /* Set file attributes */
634          file_attributes = 0;          file_attributes = 0;
635          if (S_ISDIR(filestat.st_mode))          if (S_ISDIR(filestat.st_mode))
         {  
636                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
637          }  
638          filename = 1 + strrchr(path, '/');          filename = 1 + strrchr(path, '/');
639          if (filename && filename[0] == '.')          if (filename && filename[0] == '.')
         {  
640                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
         }  
641    
642          // Return requested data          if (!file_attributes)
643                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
644    
645            if (!(filestat.st_mode & S_IWUSR))
646                    file_attributes |= FILE_ATTRIBUTE_READONLY;
647    
648            /* Return requested data */
649          switch (info_class)          switch (info_class)
650          {          {
651                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
652                            seconds_since_1970_to_filetime(get_create_time(&filestat), &ft_high,
653                          out_uint8s(out, 8);     //create_time not available;                                                         &ft_low);
654                            out_uint32_le(out, ft_low);     /* create_access_time */
655                            out_uint32_le(out, ft_high);
656    
657                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);
658                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     /* last_access_time */
659                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
660    
661                          seconds_since_1970_to_filetime(filestat.st_mtime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(filestat.st_mtime, &ft_high, &ft_low);
662                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     /* last_write_time */
663                            out_uint32_le(out, ft_high);
664    
665                            seconds_since_1970_to_filetime(filestat.st_ctime, &ft_high, &ft_low);
666                            out_uint32_le(out, ft_low);     /* last_change_time */
667                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
668    
                         out_uint8s(out, 8);     //unknown zero  
669                          out_uint32_le(out, file_attributes);                          out_uint32_le(out, file_attributes);
670                          break;                          break;
671    
672                  case 5: /* FileStandardInformation */                  case FileStandardInformation:
673    
674                          out_uint32_le(out, filestat.st_size);   //Allocation size                          out_uint32_le(out, filestat.st_size);   /* Allocation size */
675                          out_uint32_le(out, 0);                          out_uint32_le(out, 0);
676                          out_uint32_le(out, filestat.st_size);   //End of file                          out_uint32_le(out, filestat.st_size);   /* End of file */
677                          out_uint32_le(out, 0);                          out_uint32_le(out, 0);
678                          out_uint32_le(out, filestat.st_nlink);  //Number of links                          out_uint32_le(out, filestat.st_nlink);  /* Number of links */
679                          out_uint8(out, 0);      //Delete pending                          out_uint8(out, 0);      /* Delete pending */
680                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      /* Directory */
681                          break;                          break;
682    
683                  case 35:        /* FileObjectIdInformation */                  case FileObjectIdInformation:
684    
685                          out_uint32_le(out, file_attributes);    /* File Attributes */                          out_uint32_le(out, file_attributes);    /* File Attributes */
686                          out_uint32_le(out, 0);  /* Reparse Tag */                          out_uint32_le(out, 0);  /* Reparse Tag */
# Line 467  disk_query_information(HANDLE handle, ui Line 695  disk_query_information(HANDLE handle, ui
695  }  }
696    
697  NTSTATUS  NTSTATUS
698  disk_set_information(HANDLE handle, uint32 info_class, STREAM in, STREAM out)  disk_set_information(NTHANDLE handle, uint32 info_class, STREAM in, STREAM out)
699  {  {
700          uint32 device_id, length, file_attributes, ft_high, ft_low;          uint32 length, file_attributes, ft_high, ft_low, delete_on_close;
701          char newname[256], fullpath[256];          char newname[PATH_MAX], fullpath[PATH_MAX];
702          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
703            int mode;
704            struct stat filestat;
705            time_t write_time, change_time, access_time, mod_time;
706            struct utimbuf tvs;
707            struct STATFS_T stat_fs;
708    
709          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
710            g_notify_stamp = True;
711    
712          switch (info_class)          switch (info_class)
713          {          {
714                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
715                            write_time = change_time = access_time = 0;
716    
717                            in_uint8s(in, 4);       /* Handle of root dir? */
718                            in_uint8s(in, 24);      /* unknown */
719    
720                            /* CreationTime */
721                            in_uint32_le(in, ft_low);
722                            in_uint32_le(in, ft_high);
723    
724                            /* AccessTime */
725                            in_uint32_le(in, ft_low);
726                            in_uint32_le(in, ft_high);
727                            if (ft_low || ft_high)
728                                    access_time = convert_1970_to_filetime(ft_high, ft_low);
729    
730                            /* WriteTime */
731                            in_uint32_le(in, ft_low);
732                            in_uint32_le(in, ft_high);
733                            if (ft_low || ft_high)
734                                    write_time = convert_1970_to_filetime(ft_high, ft_low);
735    
736                            /* ChangeTime */
737                            in_uint32_le(in, ft_low);
738                            in_uint32_le(in, ft_high);
739                            if (ft_low || ft_high)
740                                    change_time = convert_1970_to_filetime(ft_high, ft_low);
741    
742                            in_uint32_le(in, file_attributes);
743    
744                            if (fstat(handle, &filestat))
745                                    return STATUS_ACCESS_DENIED;
746    
747                            tvs.modtime = filestat.st_mtime;
748                            tvs.actime = filestat.st_atime;
749                            if (access_time)
750                                    tvs.actime = access_time;
751    
752    
753                            if (write_time || change_time)
754                                    mod_time = MIN(write_time, change_time);
755                            else
756                                    mod_time = write_time ? write_time : change_time;
757    
758                            if (mod_time)
759                                    tvs.modtime = mod_time;
760    
761    
762                            if (access_time || write_time || change_time)
763                            {
764    #if WITH_DEBUG_RDP5
765                                    printf("FileBasicInformation access       time %s",
766                                           ctime(&tvs.actime));
767                                    printf("FileBasicInformation modification time %s",
768                                           ctime(&tvs.modtime));
769    #endif
770                                    if (utime(pfinfo->path, &tvs) && errno != EPERM)
771                                            return STATUS_ACCESS_DENIED;
772                            }
773    
774                            if (!file_attributes)
775                                    break;  /* not valid */
776    
777                            mode = filestat.st_mode;
778    
779                            if (file_attributes & FILE_ATTRIBUTE_READONLY)
780                                    mode &= ~(S_IWUSR | S_IWGRP | S_IWOTH);
781                            else
782                                    mode |= S_IWUSR;
783    
784                            mode &= 0777;
785    #if WITH_DEBUG_RDP5
786                            printf("FileBasicInformation set access mode 0%o", mode);
787    #endif
788    
789                            if (fchmod(handle, mode))
790                                    return STATUS_ACCESS_DENIED;
791    
                         // Probably safe to ignore  
792                          break;                          break;
793    
794                  case 10:        /* FileRenameInformation */                  case FileRenameInformation:
795    
796                          in_uint8s(in, 4);       /* Handle of root dir? */                          in_uint8s(in, 4);       /* Handle of root dir? */
797                          in_uint8s(in, 0x1a);    /* unknown */                          in_uint8s(in, 0x1a);    /* unknown */
# Line 508  disk_set_information(HANDLE handle, uint Line 817  disk_set_information(HANDLE handle, uint
817                          }                          }
818                          break;                          break;
819    
820                  case 13:        /* FileDispositionInformation */                  case FileDispositionInformation:
821                            /* As far as I understand it, the correct
822                               thing to do here is to *schedule* a delete,
823                               so it will be deleted when the file is
824                               closed. Subsequent
825                               FileDispositionInformation requests with
826                               DeleteFile set to FALSE should unschedule
827                               the delete. See
828                               http://www.osronline.com/article.cfm?article=245. */
829    
830                            in_uint32_le(in, delete_on_close);
831    
832                            if (delete_on_close ||
833                                (pfinfo->
834                                 accessmask & (FILE_DELETE_ON_CLOSE | FILE_COMPLETE_IF_OPLOCKED)))
835                            {
836                                    pfinfo->delete_on_close = True;
837                            }
838    
                         //unimpl("IRP Set File Information class: FileDispositionInformation\n");  
                         // in_uint32_le(in, delete_on_close);  
                         // disk_close(handle);  
                         unlink(pfinfo->path);  
839                          break;                          break;
840    
841                  case 19:        /* FileAllocationInformation */                  case FileAllocationInformation:
842                            /* Fall through to FileEndOfFileInformation,
843                          unimpl("IRP Set File Information class: FileAllocationInformation\n");                             which uses ftrunc. This is like Samba with
844                          break;                             "strict allocation = false", and means that
845                               we won't detect out-of-quota errors, for
846                               example. */
847    
848                    case FileEndOfFileInformation:
849                            in_uint8s(in, 28);      /* unknown */
850                            in_uint32_le(in, length);       /* file size */
851    
852                            /* prevents start of writing if not enough space left on device */
853                            if (STATFS_FN(g_rdpdr_device[pfinfo->device_id].local_path, &stat_fs) == 0)
854                                    if (stat_fs.f_bfree * stat_fs.f_bsize < length)
855                                            return STATUS_DISK_FULL;
856    
857                  case 20:        /* FileEndOfFileInformation */                          if (ftruncate_growable(handle, length) != 0)
858                            {
859                                    return STATUS_DISK_FULL;
860                            }
861    
                         unimpl("IRP Set File Information class: FileEndOfFileInformation\n");  
862                          break;                          break;
   
863                  default:                  default:
864    
865                          unimpl("IRP Set File Information class: 0x%x\n", info_class);                          unimpl("IRP Set File Information class: 0x%x\n", info_class);
# Line 535  disk_set_information(HANDLE handle, uint Line 869  disk_set_information(HANDLE handle, uint
869  }  }
870    
871  NTSTATUS  NTSTATUS
872  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)  disk_check_notify(NTHANDLE handle)
873    {
874            struct fileinfo *pfinfo;
875            NTSTATUS status = STATUS_PENDING;
876    
877            NOTIFY notify;
878    
879            pfinfo = &(g_fileinfo[handle]);
880            if (!pfinfo->pdir)
881                    return STATUS_INVALID_DEVICE_REQUEST;
882    
883    
884    
885            status = NotifyInfo(handle, pfinfo->info_class, &notify);
886    
887            if (status != STATUS_PENDING)
888                    return status;
889    
890            if (memcmp(&pfinfo->notify, &notify, sizeof(NOTIFY)))
891            {
892                    /*printf("disk_check_notify found changed event\n"); */
893                    memcpy(&pfinfo->notify, &notify, sizeof(NOTIFY));
894                    status = STATUS_NOTIFY_ENUM_DIR;
895            }
896    
897            return status;
898    
899    
900    }
901    
902    NTSTATUS
903    disk_create_notify(NTHANDLE handle, uint32 info_class)
904    {
905    
906            struct fileinfo *pfinfo;
907            NTSTATUS ret = STATUS_PENDING;
908    
909            /* printf("start disk_create_notify info_class %X\n", info_class); */
910    
911            pfinfo = &(g_fileinfo[handle]);
912            pfinfo->info_class = info_class;
913    
914            ret = NotifyInfo(handle, info_class, &pfinfo->notify);
915    
916            if (info_class & 0x1000)
917            {                       /* ???? */
918                    if (ret == STATUS_PENDING)
919                            return STATUS_SUCCESS;
920            }
921    
922            /* printf("disk_create_notify: num_entries %d\n", pfinfo->notify.num_entries); */
923    
924    
925            return ret;
926    
927    }
928    
929    static NTSTATUS
930    NotifyInfo(NTHANDLE handle, uint32 info_class, NOTIFY * p)
931    {
932            struct fileinfo *pfinfo;
933            struct stat buf;
934            struct dirent *dp;
935            char *fullname;
936            DIR *dpr;
937    
938            pfinfo = &(g_fileinfo[handle]);
939            if (fstat(handle, &buf) < 0)
940            {
941                    perror("NotifyInfo");
942                    return STATUS_ACCESS_DENIED;
943            }
944            p->modify_time = buf.st_mtime;
945            p->status_time = buf.st_ctime;
946            p->num_entries = 0;
947            p->total_time = 0;
948    
949    
950            dpr = opendir(pfinfo->path);
951            if (!dpr)
952            {
953                    perror("NotifyInfo");
954                    return STATUS_ACCESS_DENIED;
955            }
956    
957    
958            while ((dp = readdir(dpr)))
959            {
960                    if (!strcmp(dp->d_name, ".") || !strcmp(dp->d_name, ".."))
961                            continue;
962                    p->num_entries++;
963                    fullname = xmalloc(strlen(pfinfo->path) + strlen(dp->d_name) + 2);
964                    sprintf(fullname, "%s/%s", pfinfo->path, dp->d_name);
965    
966                    if (!stat(fullname, &buf))
967                    {
968                            p->total_time += (buf.st_mtime + buf.st_ctime);
969                    }
970    
971                    xfree(fullname);
972            }
973            closedir(dpr);
974    
975            return STATUS_PENDING;
976    }
977    
978    static FsInfoType *
979    FsVolumeInfo(char *fpath)
980    {
981    
982            static FsInfoType info;
983    #ifdef USE_SETMNTENT
984            FILE *fdfs;
985            struct mntent *e;
986    #endif
987    
988            /* initialize */
989            memset(&info, 0, sizeof(info));
990            strcpy(info.label, "RDESKTOP");
991            strcpy(info.type, "RDPFS");
992    
993    #ifdef USE_SETMNTENT
994            fdfs = setmntent(MNTENT_PATH, "r");
995            if (!fdfs)
996                    return &info;
997    
998            while ((e = getmntent(fdfs)))
999            {
1000                    if (strncmp(fpath, e->mnt_dir, strlen(fpath)) == 0)
1001                    {
1002                            strcpy(info.type, e->mnt_type);
1003                            strcpy(info.name, e->mnt_fsname);
1004                            if (strstr(e->mnt_opts, "vfat") || strstr(e->mnt_opts, "iso9660"))
1005                            {
1006                                    int fd = open(e->mnt_fsname, O_RDONLY);
1007                                    if (fd >= 0)
1008                                    {
1009                                            unsigned char buf[512];
1010                                            memset(buf, 0, sizeof(buf));
1011                                            if (strstr(e->mnt_opts, "vfat"))
1012                                                     /*FAT*/
1013                                            {
1014                                                    strcpy(info.type, "vfat");
1015                                                    read(fd, buf, sizeof(buf));
1016                                                    info.serial =
1017                                                            (buf[42] << 24) + (buf[41] << 16) +
1018                                                            (buf[40] << 8) + buf[39];
1019                                                    strncpy(info.label, buf + 43, 10);
1020                                                    info.label[10] = '\0';
1021                                            }
1022                                            else if (lseek(fd, 32767, SEEK_SET) >= 0)       /* ISO9660 */
1023                                            {
1024                                                    read(fd, buf, sizeof(buf));
1025                                                    strncpy(info.label, buf + 41, 32);
1026                                                    info.label[32] = '\0';
1027                                                    /* info.Serial = (buf[128]<<24)+(buf[127]<<16)+(buf[126]<<8)+buf[125]; */
1028                                            }
1029                                            close(fd);
1030                                    }
1031                            }
1032                    }
1033            }
1034            endmntent(fdfs);
1035    #else
1036            /* initialize */
1037            memset(&info, 0, sizeof(info));
1038            strcpy(info.label, "RDESKTOP");
1039            strcpy(info.type, "RDPFS");
1040    
1041    #endif
1042            return &info;
1043    }
1044    
1045    
1046    NTSTATUS
1047    disk_query_volume_information(NTHANDLE handle, uint32 info_class, STREAM out)
1048  {  {
         char *volume, *fs_type;  
1049          struct STATFS_T stat_fs;          struct STATFS_T stat_fs;
1050          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
1051            FsInfoType *fsinfo;
1052    
1053          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
         volume = "RDESKTOP";  
         fs_type = "RDPFS";  
1054    
1055          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)     /* FIXME: statfs is not portable */          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)
1056          {          {
1057                  perror("statfs");                  perror("statfs");
1058                  return STATUS_ACCESS_DENIED;                  return STATUS_ACCESS_DENIED;
1059          }          }
1060    
1061            fsinfo = FsVolumeInfo(pfinfo->path);
1062    
1063          switch (info_class)          switch (info_class)
1064          {          {
1065                  case 1: /* FileFsVolumeInformation */                  case FileFsVolumeInformation:
1066    
1067                          out_uint32_le(out, 0);  /* volume creation time low */                          out_uint32_le(out, 0);  /* volume creation time low */
1068                          out_uint32_le(out, 0);  /* volume creation time high */                          out_uint32_le(out, 0);  /* volume creation time high */
1069                          out_uint32_le(out, 0);  /* serial */                          out_uint32_le(out, fsinfo->serial);     /* serial */
1070                          out_uint32_le(out, 2 * strlen(volume)); /* length of string */  
1071                            out_uint32_le(out, 2 * strlen(fsinfo->label));  /* length of string */
1072    
1073                          out_uint8(out, 0);      /* support objects? */                          out_uint8(out, 0);      /* support objects? */
1074                          rdp_out_unistr(out, volume, 2 * strlen(volume) - 2);                          rdp_out_unistr(out, fsinfo->label, 2 * strlen(fsinfo->label) - 2);
1075                          break;                          break;
1076    
1077                  case 3: /* FileFsSizeInformation */                  case FileFsSizeInformation:
1078    
1079                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */
1080                          out_uint32_le(out, 0);  /* Total allocation high units */                          out_uint32_le(out, 0);  /* Total allocation high units */
# Line 573  disk_query_volume_information(HANDLE han Line 1084  disk_query_volume_information(HANDLE han
1084                          out_uint32_le(out, 0x200);      /* Bytes per sector */                          out_uint32_le(out, 0x200);      /* Bytes per sector */
1085                          break;                          break;
1086    
1087                  case 5: /* FileFsAttributeInformation */                  case FileFsAttributeInformation:
1088    
1089                          out_uint32_le(out, FS_CASE_SENSITIVE | FS_CASE_IS_PRESERVED);   /* fs attributes */                          out_uint32_le(out, FS_CASE_SENSITIVE | FS_CASE_IS_PRESERVED);   /* fs attributes */
1090                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */
1091                          out_uint32_le(out, 2 * strlen(fs_type));        /* length of fs_type */  
1092                          rdp_out_unistr(out, fs_type, 2 * strlen(fs_type) - 2);                          out_uint32_le(out, 2 * strlen(fsinfo->type));   /* length of fs_type */
1093                            rdp_out_unistr(out, fsinfo->type, 2 * strlen(fsinfo->type) - 2);
1094                          break;                          break;
1095    
1096                  case 2: /* FileFsLabelInformation */                  case FileFsLabelInformation:
1097                  case 4: /* FileFsDeviceInformation */                  case FileFsDeviceInformation:
1098                  case 6: /* FileFsControlInformation */                  case FileFsControlInformation:
1099                  case 7: /* FileFsFullSizeInformation */                  case FileFsFullSizeInformation:
1100                  case 8: /* FileFsObjectIdInformation */                  case FileFsObjectIdInformation:
1101                  case 9: /* FileFsMaximumInformation */                  case FileFsMaximumInformation:
1102    
1103                  default:                  default:
1104    
1105                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);
# Line 596  disk_query_volume_information(HANDLE han Line 1109  disk_query_volume_information(HANDLE han
1109  }  }
1110    
1111  NTSTATUS  NTSTATUS
1112  disk_query_directory(HANDLE handle, uint32 info_class, char *pattern, STREAM out)  disk_query_directory(NTHANDLE handle, uint32 info_class, char *pattern, STREAM out)
1113  {  {
1114          uint32 file_attributes, ft_low, ft_high;          uint32 file_attributes, ft_low, ft_high;
1115          char *dirname, fullpath[256];          char *dirname, fullpath[PATH_MAX];
1116          DIR *pdir;          DIR *pdir;
1117          struct dirent *pdirent;          struct dirent *pdirent;
1118          struct stat fstat;          struct stat fstat;
# Line 612  disk_query_directory(HANDLE handle, uint Line 1125  disk_query_directory(HANDLE handle, uint
1125    
1126          switch (info_class)          switch (info_class)
1127          {          {
1128                  case 3: //FIXME: Why 3?                  case FileBothDirectoryInformation:
1129    
1130                          // If a search pattern is received, remember this pattern, and restart search                          /* If a search pattern is received, remember this pattern, and restart search */
1131                          if (pattern[0] != 0)                          if (pattern[0] != 0)
1132                          {                          {
1133                                  strncpy(pfinfo->pattern, 1 + strrchr(pattern, '/'), 64);                                  strncpy(pfinfo->pattern, 1 + strrchr(pattern, '/'), PATH_MAX - 1);
1134                                  rewinddir(pdir);                                  rewinddir(pdir);
1135                          }                          }
1136    
1137                          // find next dirent matching pattern                          /* find next dirent matching pattern */
1138                          pdirent = readdir(pdir);                          pdirent = readdir(pdir);
1139                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)
                         {  
1140                                  pdirent = readdir(pdir);                                  pdirent = readdir(pdir);
                         }  
1141    
1142                          if (pdirent == NULL)                          if (pdirent == NULL)
                         {  
1143                                  return STATUS_NO_MORE_FILES;                                  return STATUS_NO_MORE_FILES;
                         }  
1144    
1145                          // Get information for directory entry                          /* Get information for directory entry */
1146                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);
1147                          /* JIF  
                            printf("Stat: %s\n", fullpath); */  
1148                          if (stat(fullpath, &fstat))                          if (stat(fullpath, &fstat))
1149                          {                          {
1150                                  perror("stat");                                  switch (errno)
1151                                  out_uint8(out, 0);                                  {
1152                                  return STATUS_ACCESS_DENIED;                                          case ENOENT:
1153                                            case ELOOP:
1154                                            case EACCES:
1155                                                    /* These are non-fatal errors. */
1156                                                    memset(&fstat, 0, sizeof(fstat));
1157                                                    break;
1158                                            default:
1159                                                    /* Fatal error. By returning STATUS_NO_SUCH_FILE,
1160                                                       the directory list operation will be aborted */
1161                                                    perror(fullpath);
1162                                                    out_uint8(out, 0);
1163                                                    return STATUS_NO_SUCH_FILE;
1164                                    }
1165                          }                          }
1166    
1167                          if (S_ISDIR(fstat.st_mode))                          if (S_ISDIR(fstat.st_mode))
1168                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
1169                          if (pdirent->d_name[0] == '.')                          if (pdirent->d_name[0] == '.')
1170                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
1171                            if (!file_attributes)
1172                                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
1173                            if (!(fstat.st_mode & S_IWUSR))
1174                                    file_attributes |= FILE_ATTRIBUTE_READONLY;
1175    
1176                          // Return requested information                          /* Return requested information */
1177                          out_uint8s(out, 8);     //unknown zero                          out_uint8s(out, 8);     /* unknown zero */
1178                          out_uint8s(out, 8);     //create_time not available in posix;  
1179                            seconds_since_1970_to_filetime(get_create_time(&fstat), &ft_high, &ft_low);
1180                            out_uint32_le(out, ft_low);     /* create time */
1181                            out_uint32_le(out, ft_high);
1182    
1183                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);
1184                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     /* last_access_time */
1185                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
1186    
1187                          seconds_since_1970_to_filetime(fstat.st_mtime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(fstat.st_mtime, &ft_high, &ft_low);
1188                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     /* last_write_time */
1189                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
1190    
1191                          out_uint8s(out, 8);     //unknown zero                          seconds_since_1970_to_filetime(fstat.st_ctime, &ft_high, &ft_low);
1192                          out_uint32_le(out, fstat.st_size);      //filesize low                          out_uint32_le(out, ft_low);     /* change_write_time */
1193                          out_uint32_le(out, 0);  //filesize high                          out_uint32_le(out, ft_high);
1194                          out_uint32_le(out, fstat.st_size);      //filesize low  
1195                          out_uint32_le(out, 0);  //filesize high                          out_uint32_le(out, fstat.st_size);      /* filesize low */
1196                            out_uint32_le(out, 0);  /* filesize high */
1197                            out_uint32_le(out, fstat.st_size);      /* filesize low */
1198                            out_uint32_le(out, 0);  /* filesize high */
1199                          out_uint32_le(out, file_attributes);                          out_uint32_le(out, file_attributes);
1200                          out_uint8(out, 2 * strlen(pdirent->d_name) + 2);        //unicode length                          out_uint8(out, 2 * strlen(pdirent->d_name) + 2);        /* unicode length */
1201                          out_uint8s(out, 7);     //pad?                          out_uint8s(out, 7);     /* pad? */
1202                          out_uint8(out, 0);      //8.3 file length                          out_uint8(out, 0);      /* 8.3 file length */
1203                          out_uint8s(out, 2 * 12);        //8.3 unicode length                          out_uint8s(out, 2 * 12);        /* 8.3 unicode length */
1204                          rdp_out_unistr(out, pdirent->d_name, 2 * strlen(pdirent->d_name));                          rdp_out_unistr(out, pdirent->d_name, 2 * strlen(pdirent->d_name));
1205                          break;                          break;
1206    
1207                  default:                  default:
1208                            /* FIXME: Support FileDirectoryInformation,
1209                               FileFullDirectoryInformation, and
1210                               FileNamesInformation */
1211    
1212                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);
1213                          return STATUS_INVALID_PARAMETER;                          return STATUS_INVALID_PARAMETER;
# Line 683  disk_query_directory(HANDLE handle, uint Line 1216  disk_query_directory(HANDLE handle, uint
1216          return STATUS_SUCCESS;          return STATUS_SUCCESS;
1217  }  }
1218    
1219    
1220    
1221    static NTSTATUS
1222    disk_device_control(NTHANDLE handle, uint32 request, STREAM in, STREAM out)
1223    {
1224            if (((request >> 16) != 20) || ((request >> 16) != 9))
1225                    return STATUS_INVALID_PARAMETER;
1226    
1227            /* extract operation */
1228            request >>= 2;
1229            request &= 0xfff;
1230    
1231            printf("DISK IOCTL %d\n", request);
1232    
1233            switch (request)
1234            {
1235                    case 25:        /* ? */
1236                    case 42:        /* ? */
1237                    default:
1238                            unimpl("DISK IOCTL %d\n", request);
1239                            return STATUS_INVALID_PARAMETER;
1240            }
1241    
1242            return STATUS_SUCCESS;
1243    }
1244    
1245  DEVICE_FNS disk_fns = {  DEVICE_FNS disk_fns = {
1246          disk_create,          disk_create,
1247          disk_close,          disk_close,
1248          disk_read,          disk_read,
1249          disk_write,          disk_write,
1250          NULL                    /* device_control */          disk_device_control     /* device_control */
1251  };  };

Legend:
Removed from v.596  
changed lines
  Added in v.986

  ViewVC Help
Powered by ViewVC 1.1.26