/[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 576 by stargo, Thu Jan 22 20:31:59 2004 UTC revision 790 by astrand, Thu Oct 28 07:46:39 2004 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 82  Line 28 
28  #include <fnmatch.h>  #include <fnmatch.h>
29  #include <errno.h>              /* errno */  #include <errno.h>              /* errno */
30    
31  #if defined(SOLARIS)  #include <utime.h>
32    #include <time.h>               /* ctime */
33    
34    #if (defined(HAVE_DIRFD) || (HAVE_DECL_DIRFD == 1))
35    #define DIRFD(a) (dirfd(a))
36    #else
37    #define DIRFD(a) ((a)->DIR_FD_MEMBER_NAME)
38    #endif
39    
40    /* TODO: let autoconf figure out everything below... */
41    #if (defined(sun) && (defined(__svr4__) || defined(__SVR4)))
42    #define SOLARIS
43    #endif
44    
45    #if (defined(SOLARIS) || defined (__hpux) || defined(__BEOS__))
46  #include <sys/statvfs.h>        /* solaris statvfs */  #include <sys/statvfs.h>        /* solaris statvfs */
47    /* TODO: Fix mntent-handling for solaris/hpux
48     * #include <sys/mntent.h> */
49    #undef HAVE_MNTENT_H
50    #define MNTENT_PATH "/etc/mnttab"
51  #define STATFS_FN(path, buf) (statvfs(path,buf))  #define STATFS_FN(path, buf) (statvfs(path,buf))
52  #define STATFS_T statvfs  #define STATFS_T statvfs
53  #define F_NAMELEN(buf) ((buf).f_namemax)  #define F_NAMELEN(buf) ((buf).f_namemax)
54    
55  #elif (defined(__OpenBSD__) || defined(__NetBSD__) || defined(__FreeBSD__))  #elif (defined(__OpenBSD__) || defined(__NetBSD__) || defined(__FreeBSD__) || defined(__APPLE__))
56  #include <sys/param.h>  #include <sys/param.h>
57  #include <sys/mount.h>  #include <sys/mount.h>
58  #define STATFS_FN(path, buf) (statfs(path,buf))  #define STATFS_FN(path, buf) (statfs(path,buf))
59  #define STATFS_T statfs  #define STATFS_T statfs
60  #define F_NAMELEN(buf) (NAME_MAX)  #define F_NAMELEN(buf) (NAME_MAX)
61    
62    #elif (defined(__SGI_IRIX__))
63    #include <sys/types.h>
64    #include <sys/statvfs.h>
65    #define STATFS_FN(path, buf) (statvfs(path,buf))
66    #define STATFS_T statvfs
67    #define F_NAMELEN(buf) ((buf).f_namemax)
68    
69  #else  #else
70  #include <sys/vfs.h>            /* linux statfs */  #include <sys/vfs.h>            /* linux statfs */
71    #include <mntent.h>
72    #define HAVE_MNTENT_H
73    #define MNTENT_PATH "/etc/mtab"
74  #define STATFS_FN(path, buf) (statfs(path,buf))  #define STATFS_FN(path, buf) (statfs(path,buf))
75  #define STATFS_T statfs  #define STATFS_T statfs
76  #define F_NAMELEN(buf) ((buf).f_namelen)  #define F_NAMELEN(buf) ((buf).f_namelen)
# Line 106  Line 80 
80    
81  extern RDPDR_DEVICE g_rdpdr_device[];  extern RDPDR_DEVICE g_rdpdr_device[];
82    
83  struct fileinfo  FILEINFO g_fileinfo[MAX_OPEN_FILES];
84    
85    typedef struct
86  {  {
87          uint32 device_id, flags_and_attributes;          char name[256];
88          char path[256];          char label[256];
89          DIR *pdir;          unsigned long serial;
90          struct dirent *pdirent;          char type[256];
91          char pattern[64];  } FsInfoType;
92          BOOL delete_on_close;  
93    
94    static time_t
95    get_create_time(struct stat *st)
96    {
97            time_t ret, ret1;
98    
99            ret = MIN(st->st_ctime, st->st_mtime);
100            ret1 = MIN(ret, st->st_atime);
101    
102            if (ret1 != (time_t) 0)
103                    return ret1;
104    
105            return ret;
106  }  }
 g_fileinfo[MAX_OPEN_FILES];  
107    
108  /* Convert seconds since 1970 to a filetime */  /* Convert seconds since 1970 to a filetime */
109  void  static void
110  seconds_since_1970_to_filetime(time_t seconds, uint32 * high, uint32 * low)  seconds_since_1970_to_filetime(time_t seconds, uint32 * high, uint32 * low)
111  {  {
112          unsigned long long ticks;          unsigned long long ticks;
# Line 128  seconds_since_1970_to_filetime(time_t se Line 116  seconds_since_1970_to_filetime(time_t se
116          *high = (uint32) (ticks >> 32);          *high = (uint32) (ticks >> 32);
117  }  }
118    
119    /* Convert seconds since 1970 back to filetime */
120    static time_t
121    convert_1970_to_filetime(uint32 high, uint32 low)
122    {
123            unsigned long long ticks;
124            time_t val;
125    
126            ticks = low + (((unsigned long long) high) << 32);
127            ticks /= 10000000;
128            ticks -= 11644473600LL;
129    
130            val = (time_t) ticks;
131            return (val);
132    
133    }
134    
135    /* A wrapper for ftruncate which supports growing files, even if the
136       native ftruncate doesn't. This is needed on Linux FAT filesystems,
137       for example. */
138    static int
139    ftruncate_growable(int fd, off_t length)
140    {
141            int ret;
142            off_t pos;
143            static const char zero;
144    
145            /* Try the simple method first */
146            if ((ret = ftruncate(fd, length)) != -1)
147            {
148                    return ret;
149            }
150    
151            /*
152             * Some kind of error. Perhaps we were trying to grow. Retry
153             * in a safe way.
154             */
155    
156            /* Get current position */
157            if ((pos = lseek(fd, 0, SEEK_CUR)) == -1)
158            {
159                    perror("lseek");
160                    return -1;
161            }
162    
163            /* Seek to new size */
164            if (lseek(fd, length, SEEK_SET) == -1)
165            {
166                    perror("lseek");
167                    return -1;
168            }
169    
170            /* Write a zero */
171            if (write(fd, &zero, 1) == -1)
172            {
173                    perror("write");
174                    return -1;
175            }
176    
177            /* Truncate. This shouldn't fail. */
178            if (ftruncate(fd, length) == -1)
179            {
180                    perror("ftruncate");
181                    return -1;
182            }
183    
184            /* Restore position */
185            if (lseek(fd, pos, SEEK_SET) == -1)
186            {
187                    perror("lseek");
188                    return -1;
189            }
190    
191            return 0;
192    }
193    
194    /* Just like open(2), but if a open with O_EXCL fails, retry with
195       GUARDED semantics. This might be necessary because some filesystems
196       (such as NFS filesystems mounted from a unfsd server) doesn't
197       support O_EXCL. GUARDED semantics are subject to race conditions,
198       but we can live with that.
199    */
200    static int
201    open_weak_exclusive(const char *pathname, int flags, mode_t mode)
202    {
203            int ret;
204            struct stat statbuf;
205    
206            ret = open(pathname, flags, mode);
207            if (ret != -1 || !(flags & O_EXCL))
208            {
209                    /* Success, or not using O_EXCL */
210                    return ret;
211            }
212    
213            /* An error occured, and we are using O_EXCL. In case the FS
214               doesn't support O_EXCL, some kind of error will be
215               returned. Unfortunately, we don't know which one. Linux
216               2.6.8 seems to return 524, but I cannot find a documented
217               #define for this case. So, we'll return only on errors that
218               we know aren't related to O_EXCL. */
219            switch (errno)
220            {
221                    case EACCES:
222                    case EEXIST:
223                    case EINTR:
224                    case EISDIR:
225                    case ELOOP:
226                    case ENAMETOOLONG:
227                    case ENOENT:
228                    case ENOTDIR:
229                            return ret;
230            }
231    
232            /* Retry with GUARDED semantics */
233            if (stat(pathname, &statbuf) != -1)
234            {
235                    /* File exists */
236                    errno = EEXIST;
237                    return -1;
238            }
239            else
240            {
241                    return open(pathname, flags & ~O_EXCL, mode);
242            }
243    }
244    
245  /* Enumeration of devices from rdesktop.c        */  /* Enumeration of devices from rdesktop.c        */
246  /* returns numer of units found and initialized. */  /* returns numer of units found and initialized. */
247  /* optarg looks like ':h:=/mnt/floppy,b:=/mnt/usbdevice1' */  /* optarg looks like ':h=/mnt/floppy,b=/mnt/usbdevice1' */
248  /* when it arrives to this function.             */  /* when it arrives to this function.             */
249  int  int
250  disk_enum_devices(int *id, char *optarg)  disk_enum_devices(uint32 * id, char *optarg)
251  {  {
252          char *pos = optarg;          char *pos = optarg;
253          char *pos2;          char *pos2;
# Line 144  disk_enum_devices(int *id, char *optarg) Line 258  disk_enum_devices(int *id, char *optarg)
258          while ((pos = next_arg(optarg, ',')) && *id < RDPDR_MAX_DEVICES)          while ((pos = next_arg(optarg, ',')) && *id < RDPDR_MAX_DEVICES)
259          {          {
260                  pos2 = next_arg(optarg, '=');                  pos2 = next_arg(optarg, '=');
                 strcpy(g_rdpdr_device[*id].name, optarg);  
   
                 toupper_str(g_rdpdr_device[*id].name);  
261    
262                  /* add trailing colon to name. */                  strncpy(g_rdpdr_device[*id].name, optarg, sizeof(g_rdpdr_device[*id].name));
263                  strcat(g_rdpdr_device[*id].name, ":");                  if (strlen(optarg) > 8)
264                            fprintf(stderr, "share name %s truncated to %s\n", optarg,
265                                    g_rdpdr_device[*id].name);
266    
267                  g_rdpdr_device[*id].local_path = xmalloc(strlen(pos2) + 1);                  g_rdpdr_device[*id].local_path = xmalloc(strlen(pos2) + 1);
268                  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);  
269                  g_rdpdr_device[*id].device_type = DEVICE_TYPE_DISK;                  g_rdpdr_device[*id].device_type = DEVICE_TYPE_DISK;
270                  count++;                  count++;
271                  (*id)++;                  (*id)++;
# Line 163  disk_enum_devices(int *id, char *optarg) Line 275  disk_enum_devices(int *id, char *optarg)
275          return count;          return count;
276  }  }
277    
278  /* Opens of creates a file or directory */  /* Opens or creates a file or directory */
279  NTSTATUS  static NTSTATUS
280  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,
281              uint32 flags_and_attributes, char *filename, HANDLE * phandle)              uint32 flags_and_attributes, char *filename, NTHANDLE * phandle)
282  {  {
283          HANDLE handle;          NTHANDLE handle;
284          DIR *dirp;          DIR *dirp;
285          int flags, mode;          int flags, mode;
286          char path[256];          char path[256];
287            struct stat filestat;
288    
289          handle = 0;          handle = 0;
290          dirp = NULL;          dirp = NULL;
291          flags = 0;          flags = 0;
292          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
293    
294          if (filename[strlen(filename) - 1] == '/')          if (*filename && filename[strlen(filename) - 1] == '/')
295                  filename[strlen(filename) - 1] = 0;                  filename[strlen(filename) - 1] = 0;
296          sprintf(path, "%s%s", g_rdpdr_device[device_id].local_path, filename);          sprintf(path, "%s%s", g_rdpdr_device[device_id].local_path, filename);
         //printf("Open: %s\n", path);  
297    
298          switch (create_disposition)          switch (create_disposition)
299          {          {
# Line 216  disk_create(uint32 device_id, uint32 acc Line 328  disk_create(uint32 device_id, uint32 acc
328                          break;                          break;
329          }          }
330    
331            //printf("Open: \"%s\"  flags: %u, accessmask: %u sharemode: %u create disp: %u\n", path, flags_and_attributes, accessmask, sharemode, create_disposition);
332    
333            // Get information about file and set that flag ourselfs
334            if ((stat(path, &filestat) == 0) && (S_ISDIR(filestat.st_mode)))
335            {
336                    if (flags_and_attributes & FILE_NON_DIRECTORY_FILE)
337                            return STATUS_FILE_IS_A_DIRECTORY;
338                    else
339                            flags_and_attributes |= FILE_DIRECTORY_FILE;
340            }
341    
342          if (flags_and_attributes & FILE_DIRECTORY_FILE)          if (flags_and_attributes & FILE_DIRECTORY_FILE)
343          {          {
344                  if (flags & O_CREAT)                  if (flags & O_CREAT)
# Line 246  disk_create(uint32 device_id, uint32 acc Line 369  disk_create(uint32 device_id, uint32 acc
369          }          }
370          else          else
371          {          {
372    
373                  if (accessmask & GENERIC_ALL                  if (accessmask & GENERIC_ALL
374                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))
375                  {                  {
# Line 260  disk_create(uint32 device_id, uint32 acc Line 384  disk_create(uint32 device_id, uint32 acc
384                          flags |= O_RDONLY;                          flags |= O_RDONLY;
385                  }                  }
386    
387                  handle = open(path, flags, mode);                  handle = open_weak_exclusive(path, flags, mode);
388                  if (handle == -1)                  if (handle == -1)
389                  {                  {
390                          switch (errno)                          switch (errno)
391                          {                          {
392                                    case EISDIR:
393    
394                                            return STATUS_FILE_IS_A_DIRECTORY;
395    
396                                  case EACCES:                                  case EACCES:
397    
398                                          return STATUS_ACCESS_DENIED;                                          return STATUS_ACCESS_DENIED;
# Line 272  disk_create(uint32 device_id, uint32 acc Line 400  disk_create(uint32 device_id, uint32 acc
400                                  case ENOENT:                                  case ENOENT:
401    
402                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
403                                    case EEXIST:
404    
405                                            return STATUS_OBJECT_NAME_COLLISION;
406                                  default:                                  default:
407    
408                                          perror("open");                                          perror("open");
409                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
410                          }                          }
411                  }                  }
412    
413                    /* all read and writes of files should be non blocking */
414                    if (fcntl(handle, F_SETFL, O_NONBLOCK) == -1)
415                            perror("fcntl");
416          }          }
417    
418          if (handle >= MAX_OPEN_FILES)          if (handle >= MAX_OPEN_FILES)
# Line 298  disk_create(uint32 device_id, uint32 acc Line 432  disk_create(uint32 device_id, uint32 acc
432          return STATUS_SUCCESS;          return STATUS_SUCCESS;
433  }  }
434    
435  NTSTATUS  static NTSTATUS
436  disk_close(HANDLE handle)  disk_close(NTHANDLE handle)
437  {  {
438          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
439    
# Line 318  disk_close(HANDLE handle) Line 452  disk_close(HANDLE handle)
452          return STATUS_SUCCESS;          return STATUS_SUCCESS;
453  }  }
454    
455  NTSTATUS  static NTSTATUS
456  disk_read(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_read(NTHANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
457  {  {
458          int n;          int n;
459    
460          if (offset)  #if 0
461                  lseek(handle, offset, SEEK_SET);          /* browsing dir ????        */
462            /* each request is 24 bytes */
463            if (g_fileinfo[handle].flags_and_attributes & FILE_DIRECTORY_FILE)
464            {
465                    *result = 0;
466                    return STATUS_SUCCESS;
467            }
468    #endif
469    
470            lseek(handle, offset, SEEK_SET);
471    
472          n = read(handle, data, length);          n = read(handle, data, length);
473    
474          if (n < 0)          if (n < 0)
475          {          {
                 perror("read");  
476                  *result = 0;                  *result = 0;
477                  return STATUS_INVALID_PARAMETER;                  switch (errno)
478                    {
479                            case EISDIR:
480                                    return STATUS_FILE_IS_A_DIRECTORY;
481                            default:
482                                    perror("read");
483                                    return STATUS_INVALID_PARAMETER;
484                    }
485          }          }
486    
487          *result = n;          *result = n;
# Line 339  disk_read(HANDLE handle, uint8 * data, u Line 489  disk_read(HANDLE handle, uint8 * data, u
489          return STATUS_SUCCESS;          return STATUS_SUCCESS;
490  }  }
491    
492  NTSTATUS  static NTSTATUS
493  disk_write(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_write(NTHANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
494  {  {
495          int n;          int n;
496    
497          if (offset)          lseek(handle, offset, SEEK_SET);
                 lseek(handle, offset, SEEK_SET);  
498    
499          n = write(handle, data, length);          n = write(handle, data, length);
500    
# Line 353  disk_write(HANDLE handle, uint8 * data, Line 502  disk_write(HANDLE handle, uint8 * data,
502          {          {
503                  perror("write");                  perror("write");
504                  *result = 0;                  *result = 0;
505                  return STATUS_ACCESS_DENIED;                  switch (errno)
506                    {
507                            case ENOSPC:
508                                    return STATUS_DISK_FULL;
509                            default:
510                                    return STATUS_ACCESS_DENIED;
511                    }
512          }          }
513    
514          *result = n;          *result = n;
# Line 362  disk_write(HANDLE handle, uint8 * data, Line 517  disk_write(HANDLE handle, uint8 * data,
517  }  }
518    
519  NTSTATUS  NTSTATUS
520  disk_query_information(HANDLE handle, uint32 info_class, STREAM out)  disk_query_information(NTHANDLE handle, uint32 info_class, STREAM out)
521  {  {
522          uint32 file_attributes, ft_high, ft_low;          uint32 file_attributes, ft_high, ft_low;
523          struct stat filestat;          struct stat filestat;
# Line 381  disk_query_information(HANDLE handle, ui Line 536  disk_query_information(HANDLE handle, ui
536          // Set file attributes          // Set file attributes
537          file_attributes = 0;          file_attributes = 0;
538          if (S_ISDIR(filestat.st_mode))          if (S_ISDIR(filestat.st_mode))
         {  
539                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
540          }  
541          filename = 1 + strrchr(path, '/');          filename = 1 + strrchr(path, '/');
542          if (filename && filename[0] == '.')          if (filename && filename[0] == '.')
         {  
543                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
544          }  
545            if (!file_attributes)
546                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
547    
548            if (!(filestat.st_mode & S_IWUSR))
549                    file_attributes |= FILE_ATTRIBUTE_READONLY;
550    
551          // Return requested data          // Return requested data
552          switch (info_class)          switch (info_class)
553          {          {
554                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
555                            seconds_since_1970_to_filetime(get_create_time(&filestat), &ft_high,
556                          out_uint8s(out, 8);     //create_time not available;                                                         &ft_low);
557                            out_uint32_le(out, ft_low);     //create_access_time
558                            out_uint32_le(out, ft_high);
559    
560                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);
561                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     //last_access_time
# Line 405  disk_query_information(HANDLE handle, ui Line 565  disk_query_information(HANDLE handle, ui
565                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     //last_write_time
566                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
567    
568                          out_uint8s(out, 8);     //unknown zero                          seconds_since_1970_to_filetime(filestat.st_ctime, &ft_high, &ft_low);
569                            out_uint32_le(out, ft_low);     //last_change_time
570                            out_uint32_le(out, ft_high);
571    
572                          out_uint32_le(out, file_attributes);                          out_uint32_le(out, file_attributes);
573                          break;                          break;
574    
575                  case 5: /* FileStandardInformation */                  case FileStandardInformation:
576    
577                          out_uint32_le(out, filestat.st_size);   //Allocation size                          out_uint32_le(out, filestat.st_size);   //Allocation size
578                          out_uint32_le(out, 0);                          out_uint32_le(out, 0);
# Line 420  disk_query_information(HANDLE handle, ui Line 583  disk_query_information(HANDLE handle, ui
583                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory
584                          break;                          break;
585    
586                  case 35:        /* FileObjectIdInformation */                  case FileObjectIdInformation:
587    
588                          out_uint32_le(out, file_attributes);    /* File Attributes */                          out_uint32_le(out, file_attributes);    /* File Attributes */
589                          out_uint32_le(out, 0);  /* Reparse Tag */                          out_uint32_le(out, 0);  /* Reparse Tag */
# Line 435  disk_query_information(HANDLE handle, ui Line 598  disk_query_information(HANDLE handle, ui
598  }  }
599    
600  NTSTATUS  NTSTATUS
601  disk_set_information(HANDLE handle, uint32 info_class, STREAM in, STREAM out)  disk_set_information(NTHANDLE handle, uint32 info_class, STREAM in, STREAM out)
602  {  {
603          uint32 device_id, length, file_attributes, ft_high, ft_low;          uint32 length, file_attributes, ft_high, ft_low;
604          char newname[256], fullpath[256];          char newname[256], fullpath[256];
605          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
606    
607            int mode;
608            struct stat filestat;
609            time_t write_time, change_time, access_time, mod_time;
610            struct utimbuf tvs;
611            struct STATFS_T stat_fs;
612    
613          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
614    
615          switch (info_class)          switch (info_class)
616          {          {
617                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
618                            write_time = change_time = access_time = 0;
619    
620                            in_uint8s(in, 4);       /* Handle of root dir? */
621                            in_uint8s(in, 24);      /* unknown */
622    
623                            // CreationTime
624                            in_uint32_le(in, ft_low);
625                            in_uint32_le(in, ft_high);
626    
627                            // AccessTime
628                            in_uint32_le(in, ft_low);
629                            in_uint32_le(in, ft_high);
630                            if (ft_low || ft_high)
631                                    access_time = convert_1970_to_filetime(ft_high, ft_low);
632    
633                            // WriteTime
634                            in_uint32_le(in, ft_low);
635                            in_uint32_le(in, ft_high);
636                            if (ft_low || ft_high)
637                                    write_time = convert_1970_to_filetime(ft_high, ft_low);
638    
639                            // ChangeTime
640                            in_uint32_le(in, ft_low);
641                            in_uint32_le(in, ft_high);
642                            if (ft_low || ft_high)
643                                    change_time = convert_1970_to_filetime(ft_high, ft_low);
644    
645                            in_uint32_le(in, file_attributes);
646    
647                            if (fstat(handle, &filestat))
648                                    return STATUS_ACCESS_DENIED;
649    
650                            tvs.modtime = filestat.st_mtime;
651                            tvs.actime = filestat.st_atime;
652                            if (access_time)
653                                    tvs.actime = access_time;
654    
655    
656                            if (write_time || change_time)
657                                    mod_time = MIN(write_time, change_time);
658                            else
659                                    mod_time = write_time ? write_time : change_time;
660    
661                            if (mod_time)
662                                    tvs.modtime = mod_time;
663    
664    
665                            if (access_time || write_time || change_time)
666                            {
667    #if WITH_DEBUG_RDP5
668                                    printf("FileBasicInformation access       time %s",
669                                           ctime(&tvs.actime));
670                                    printf("FileBasicInformation modification time %s",
671                                           ctime(&tvs.modtime));
672    #endif
673                                    if (utime(pfinfo->path, &tvs))
674                                            return STATUS_ACCESS_DENIED;
675                            }
676    
677                            if (!file_attributes)
678                                    break;  // not valid
679    
680                            mode = filestat.st_mode;
681    
682                            if (file_attributes & FILE_ATTRIBUTE_READONLY)
683                                    mode &= ~(S_IWUSR | S_IWGRP | S_IWOTH);
684                            else
685                                    mode |= S_IWUSR;
686    
687                            mode &= 0777;
688    #if WITH_DEBUG_RDP5
689                            printf("FileBasicInformation set access mode 0%o", mode);
690    #endif
691    
692                            if (fchmod(handle, mode))
693                                    return STATUS_ACCESS_DENIED;
694    
                         // Probably safe to ignore  
695                          break;                          break;
696    
697                  case 10:        /* FileRenameInformation */                  case FileRenameInformation:
698    
699                          in_uint8s(in, 4);       /* Handle of root dir? */                          in_uint8s(in, 4);       /* Handle of root dir? */
700                          in_uint8s(in, 0x1a);    /* unknown */                          in_uint8s(in, 0x1a);    /* unknown */
# Line 476  disk_set_information(HANDLE handle, uint Line 720  disk_set_information(HANDLE handle, uint
720                          }                          }
721                          break;                          break;
722    
723                  case 13:        /* FileDispositionInformation */                  case FileDispositionInformation:
724                            /* As far as I understand it, the correct
725                               thing to do here is to *schedule* a delete,
726                               so it will be deleted when the file is
727                               closed. Subsequent
728                               FileDispositionInformation requests with
729                               DeleteFile set to FALSE should unschedule
730                               the delete. See
731                               http://www.osronline.com/article.cfm?article=245. Currently,
732                               we are deleting the file immediately. I
733                               guess this is a FIXME. */
734    
735                            //in_uint32_le(in, delete_on_close);
736    
737                            /* Make sure we close the file before
738                               unlinking it. Not doing so would trigger
739                               silly-delete if using NFS, which might fail
740                               on FAT floppies, for example. */
741                            disk_close(handle);
742    
743                          //unimpl("IRP Set File Information class: FileDispositionInformation\n");                          if ((pfinfo->flags_and_attributes & FILE_DIRECTORY_FILE))       // remove a directory
744                          // in_uint32_le(in, delete_on_close);                          {
745                          // disk_close(handle);                                  if (rmdir(pfinfo->path) < 0)
746                          unlink(pfinfo->path);                                          return STATUS_ACCESS_DENIED;
747                          break;                          }
748                            else if (unlink(pfinfo->path) < 0)      // unlink a file
749                  case 19:        /* FileAllocationInformation */                                  return STATUS_ACCESS_DENIED;
750    
                         unimpl("IRP Set File Information class: FileAllocationInformation\n");  
751                          break;                          break;
752    
753                  case 20:        /* FileEndOfFileInformation */                  case FileAllocationInformation:
754                            /* Fall through to FileEndOfFileInformation,
755                               which uses ftrunc. This is like Samba with
756                               "strict allocation = false", and means that
757                               we won't detect out-of-quota errors, for
758                               example. */
759    
760                    case FileEndOfFileInformation:
761                            in_uint8s(in, 28);      /* unknown */
762                            in_uint32_le(in, length);       /* file size */
763    
764                            /* prevents start of writing if not enough space left on device */
765                            if (STATFS_FN(g_rdpdr_device[pfinfo->device_id].local_path, &stat_fs) == 0)
766                                    if (stat_fs.f_bsize * stat_fs.f_bfree < length)
767                                            return STATUS_DISK_FULL;
768    
769                          unimpl("IRP Set File Information class: FileEndOfFileInformation\n");                          if (ftruncate_growable(handle, length) != 0)
770                          break;                          {
771                                    return STATUS_DISK_FULL;
772                            }
773    
774                            break;
775                  default:                  default:
776    
777                          unimpl("IRP Set File Information class: 0x%x\n", info_class);                          unimpl("IRP Set File Information class: 0x%x\n", info_class);
# Line 502  disk_set_information(HANDLE handle, uint Line 780  disk_set_information(HANDLE handle, uint
780          return STATUS_SUCCESS;          return STATUS_SUCCESS;
781  }  }
782    
783    static FsInfoType *
784    FsVolumeInfo(char *fpath)
785    {
786    
787    #ifdef HAVE_MNTENT_H
788            FILE *fdfs;
789            struct mntent *e;
790            static FsInfoType info;
791    
792            /* initialize */
793            memset(&info, 0, sizeof(info));
794            strcpy(info.label, "RDESKTOP");
795            strcpy(info.type, "RDPFS");
796    
797            fdfs = setmntent(MNTENT_PATH, "r");
798            if (!fdfs)
799                    return &info;
800    
801            while ((e = getmntent(fdfs)))
802            {
803                    if (strncmp(fpath, e->mnt_dir, strlen(fpath)) == 0)
804                    {
805                            strcpy(info.type, e->mnt_type);
806                            strcpy(info.name, e->mnt_fsname);
807                            if (strstr(e->mnt_opts, "vfat") || strstr(e->mnt_opts, "iso9660"))
808                            {
809                                    int fd = open(e->mnt_fsname, O_RDONLY);
810                                    if (fd >= 0)
811                                    {
812                                            unsigned char buf[512];
813                                            memset(buf, 0, sizeof(buf));
814                                            if (strstr(e->mnt_opts, "vfat"))
815                                                     /*FAT*/
816                                            {
817                                                    strcpy(info.type, "vfat");
818                                                    read(fd, buf, sizeof(buf));
819                                                    info.serial =
820                                                            (buf[42] << 24) + (buf[41] << 16) +
821                                                            (buf[40] << 8) + buf[39];
822                                                    strncpy(info.label, buf + 43, 10);
823                                                    info.label[10] = '\0';
824                                            }
825                                            else if (lseek(fd, 32767, SEEK_SET) >= 0)       /* ISO9660 */
826                                            {
827                                                    read(fd, buf, sizeof(buf));
828                                                    strncpy(info.label, buf + 41, 32);
829                                                    info.label[32] = '\0';
830                                                    //info.Serial = (buf[128]<<24)+(buf[127]<<16)+(buf[126]<<8)+buf[125];
831                                            }
832                                            close(fd);
833                                    }
834                            }
835                    }
836            }
837            endmntent(fdfs);
838    #else
839            static FsInfoType info;
840    
841            /* initialize */
842            memset(&info, 0, sizeof(info));
843            strcpy(info.label, "RDESKTOP");
844            strcpy(info.type, "RDPFS");
845    
846    #endif
847            return &info;
848    }
849    
850    
851  NTSTATUS  NTSTATUS
852  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)  disk_query_volume_information(NTHANDLE handle, uint32 info_class, STREAM out)
853  {  {
         char *volume, *fs_type;  
854          struct STATFS_T stat_fs;          struct STATFS_T stat_fs;
855          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
856            FsInfoType *fsinfo;
857    
858          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
         volume = "RDESKTOP";  
         fs_type = "RDPFS";  
859    
860          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)     /* FIXME: statfs is not portable */          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)
861          {          {
862                  perror("statfs");                  perror("statfs");
863                  return STATUS_ACCESS_DENIED;                  return STATUS_ACCESS_DENIED;
864          }          }
865    
866            fsinfo = FsVolumeInfo(pfinfo->path);
867    
868          switch (info_class)          switch (info_class)
869          {          {
870                  case 1: /* FileFsVolumeInformation */                  case FileFsVolumeInformation:
871    
872                          out_uint32_le(out, 0);  /* volume creation time low */                          out_uint32_le(out, 0);  /* volume creation time low */
873                          out_uint32_le(out, 0);  /* volume creation time high */                          out_uint32_le(out, 0);  /* volume creation time high */
874                          out_uint32_le(out, 0);  /* serial */                          out_uint32_le(out, fsinfo->serial);     /* serial */
875                          out_uint32_le(out, 2 * strlen(volume)); /* length of string */  
876                            out_uint32_le(out, 2 * strlen(fsinfo->label));  /* length of string */
877    
878                          out_uint8(out, 0);      /* support objects? */                          out_uint8(out, 0);      /* support objects? */
879                          rdp_out_unistr(out, volume, 2 * strlen(volume) - 2);                          rdp_out_unistr(out, fsinfo->label, 2 * strlen(fsinfo->label) - 2);
880                          break;                          break;
881    
882                  case 3: /* FileFsSizeInformation */                  case FileFsSizeInformation:
883    
884                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */
885                          out_uint32_le(out, 0);  /* Total allocation high units */                          out_uint32_le(out, 0);  /* Total allocation high units */
# Line 541  disk_query_volume_information(HANDLE han Line 889  disk_query_volume_information(HANDLE han
889                          out_uint32_le(out, 0x200);      /* Bytes per sector */                          out_uint32_le(out, 0x200);      /* Bytes per sector */
890                          break;                          break;
891    
892                  case 5: /* FileFsAttributeInformation */                  case FileFsAttributeInformation:
893    
894                          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 */
895                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */
896                          out_uint32_le(out, 2 * strlen(fs_type));        /* length of fs_type */  
897                          rdp_out_unistr(out, fs_type, 2 * strlen(fs_type) - 2);                          out_uint32_le(out, 2 * strlen(fsinfo->type));   /* length of fs_type */
898                            rdp_out_unistr(out, fsinfo->type, 2 * strlen(fsinfo->type) - 2);
899                          break;                          break;
900    
901                  case 2: /* FileFsLabelInformation */                  case FileFsLabelInformation:
902                  case 4: /* FileFsDeviceInformation */                  case FileFsDeviceInformation:
903                  case 6: /* FileFsControlInformation */                  case FileFsControlInformation:
904                  case 7: /* FileFsFullSizeInformation */                  case FileFsFullSizeInformation:
905                  case 8: /* FileFsObjectIdInformation */                  case FileFsObjectIdInformation:
906                  case 9: /* FileFsMaximumInformation */                  case FileFsMaximumInformation:
907    
908                  default:                  default:
909    
910                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);
# Line 564  disk_query_volume_information(HANDLE han Line 914  disk_query_volume_information(HANDLE han
914  }  }
915    
916  NTSTATUS  NTSTATUS
917  disk_query_directory(HANDLE handle, uint32 info_class, char *pattern, STREAM out)  disk_query_directory(NTHANDLE handle, uint32 info_class, char *pattern, STREAM out)
918  {  {
919          uint32 file_attributes, ft_low, ft_high;          uint32 file_attributes, ft_low, ft_high;
920          char *dirname, fullpath[256];          char *dirname, fullpath[256];
# Line 580  disk_query_directory(HANDLE handle, uint Line 930  disk_query_directory(HANDLE handle, uint
930    
931          switch (info_class)          switch (info_class)
932          {          {
933                  case 3: //FIXME: Why 3?                  case FileBothDirectoryInformation:
934    
935                          // If a search pattern is received, remember this pattern, and restart search                          // If a search pattern is received, remember this pattern, and restart search
936                          if (pattern[0] != 0)                          if (pattern[0] != 0)
# Line 592  disk_query_directory(HANDLE handle, uint Line 942  disk_query_directory(HANDLE handle, uint
942                          // find next dirent matching pattern                          // find next dirent matching pattern
943                          pdirent = readdir(pdir);                          pdirent = readdir(pdir);
944                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)
                         {  
945                                  pdirent = readdir(pdir);                                  pdirent = readdir(pdir);
                         }  
946    
947                          if (pdirent == NULL)                          if (pdirent == NULL)
                         {  
948                                  return STATUS_NO_MORE_FILES;                                  return STATUS_NO_MORE_FILES;
                         }  
949    
950                          // Get information for directory entry                          // Get information for directory entry
951                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);
952                          /* JIF  
                            printf("Stat: %s\n", fullpath); */  
953                          if (stat(fullpath, &fstat))                          if (stat(fullpath, &fstat))
954                          {                          {
955                                  perror("stat");                                  switch (errno)
956                                  out_uint8(out, 0);                                  {
957                                  return STATUS_ACCESS_DENIED;                                          case ENOENT:
958                                            case ELOOP:
959                                            case EACCES:
960                                                    /* These are non-fatal errors. */
961                                                    memset(&fstat, 0, sizeof(fstat));
962                                                    break;
963                                            default:
964                                                    /* Fatal error. By returning STATUS_NO_SUCH_FILE,
965                                                       the directory list operation will be aborted */
966                                                    perror(fullpath);
967                                                    out_uint8(out, 0);
968                                                    return STATUS_NO_SUCH_FILE;
969                                    }
970                          }                          }
971    
972                          if (S_ISDIR(fstat.st_mode))                          if (S_ISDIR(fstat.st_mode))
973                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
974                          if (pdirent->d_name[0] == '.')                          if (pdirent->d_name[0] == '.')
975                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
976                            if (!file_attributes)
977                                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
978                            if (!(fstat.st_mode & S_IWUSR))
979                                    file_attributes |= FILE_ATTRIBUTE_READONLY;
980    
981                          // Return requested information                          // Return requested information
982                          out_uint8s(out, 8);     //unknown zero                          out_uint8s(out, 8);     //unknown zero
983                          out_uint8s(out, 8);     //create_time not available in posix;  
984                            seconds_since_1970_to_filetime(get_create_time(&fstat), &ft_high, &ft_low);
985                            out_uint32_le(out, ft_low);     // create time
986                            out_uint32_le(out, ft_high);
987    
988                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);
989                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     //last_access_time
# Line 629  disk_query_directory(HANDLE handle, uint Line 993  disk_query_directory(HANDLE handle, uint
993                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     //last_write_time
994                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
995    
996                          out_uint8s(out, 8);     //unknown zero                          seconds_since_1970_to_filetime(fstat.st_ctime, &ft_high, &ft_low);
997                            out_uint32_le(out, ft_low);     //change_write_time
998                            out_uint32_le(out, ft_high);
999    
1000                          out_uint32_le(out, fstat.st_size);      //filesize low                          out_uint32_le(out, fstat.st_size);      //filesize low
1001                          out_uint32_le(out, 0);  //filesize high                          out_uint32_le(out, 0);  //filesize high
1002                          out_uint32_le(out, fstat.st_size);      //filesize low                          out_uint32_le(out, fstat.st_size);      //filesize low
# Line 643  disk_query_directory(HANDLE handle, uint Line 1010  disk_query_directory(HANDLE handle, uint
1010                          break;                          break;
1011    
1012                  default:                  default:
1013                            /* FIXME: Support FileDirectoryInformation,
1014                               FileFullDirectoryInformation, and
1015                               FileNamesInformation */
1016    
1017                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);
1018                          return STATUS_INVALID_PARAMETER;                          return STATUS_INVALID_PARAMETER;
# Line 651  disk_query_directory(HANDLE handle, uint Line 1021  disk_query_directory(HANDLE handle, uint
1021          return STATUS_SUCCESS;          return STATUS_SUCCESS;
1022  }  }
1023    
1024    
1025    
1026    static NTSTATUS
1027    disk_device_control(NTHANDLE handle, uint32 request, STREAM in, STREAM out)
1028    {
1029            if (((request >> 16) != 20) || ((request >> 16) != 9))
1030                    return STATUS_INVALID_PARAMETER;
1031    
1032            /* extract operation */
1033            request >>= 2;
1034            request &= 0xfff;
1035    
1036            printf("DISK IOCTL %d\n", request);
1037    
1038            switch (request)
1039            {
1040                    case 25:        // ?
1041                    case 42:        // ?
1042                    default:
1043                            unimpl("DISK IOCTL %d\n", request);
1044                            return STATUS_INVALID_PARAMETER;
1045            }
1046    
1047            return STATUS_SUCCESS;
1048    }
1049    
1050  DEVICE_FNS disk_fns = {  DEVICE_FNS disk_fns = {
1051          disk_create,          disk_create,
1052          disk_close,          disk_close,
1053          disk_read,          disk_read,
1054          disk_write,          disk_write,
1055          NULL                    /* device_control */          disk_device_control     /* device_control */
1056  };  };

Legend:
Removed from v.576  
changed lines
  Added in v.790

  ViewVC Help
Powered by ViewVC 1.1.26