/[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 572 by stargo, Wed Jan 21 21:51:59 2004 UTC revision 745 by stargo, Mon Aug 9 11:40:41 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  #ifdef SOLARIS  #include <utime.h>
32    #include <time.h>               /* ctime */
33    
34    #if defined(HAVE_DIRFD)
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  #define HAVE_STATVFS  /* 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))
52    #define STATFS_T statvfs
53    #define F_NAMELEN(buf) ((buf).f_namemax)
54    
55    #elif (defined(__OpenBSD__) || defined(__NetBSD__) || defined(__FreeBSD__))
56    #include <sys/param.h>
57    #include <sys/mount.h>
58    #define STATFS_FN(path, buf) (statfs(path,buf))
59    #define STATFS_T statfs
60    #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  #define HAVE_STATFS  #include <mntent.h>
72    #define HAVE_MNTENT_H
73    #define MNTENT_PATH "/etc/mtab"
74    #define STATFS_FN(path, buf) (statfs(path,buf))
75    #define STATFS_T statfs
76    #define F_NAMELEN(buf) ((buf).f_namelen)
77  #endif  #endif
78    
79  #include "rdesktop.h"  #include "rdesktop.h"
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;
         BOOL delete_on_close;  
 }  
 g_fileinfo[MAX_OPEN_FILES];  
92    
93  struct fsinfo  
94    static time_t
95    get_create_time(struct stat *st)
96  {  {
97          uint32 f_blocks, f_bfree, f_bsize, f_namelen;          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    }
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 121  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    
136  /* Enumeration of devices from rdesktop.c        */  /* Enumeration of devices from rdesktop.c        */
137  /* returns numer of units found and initialized. */  /* returns numer of units found and initialized. */
138  /* optarg looks like ':h:=/mnt/floppy,b:=/mnt/usbdevice1' */  /* optarg looks like ':h=/mnt/floppy,b=/mnt/usbdevice1' */
139  /* when it arrives to this function.             */  /* when it arrives to this function.             */
140  int  int
141  disk_enum_devices(int *id, char *optarg)  disk_enum_devices(uint32 * id, char *optarg)
142  {  {
143          char *pos = optarg;          char *pos = optarg;
144          char *pos2;          char *pos2;
# Line 156  disk_enum_devices(int *id, char *optarg) Line 168  disk_enum_devices(int *id, char *optarg)
168          return count;          return count;
169  }  }
170    
171  /* Opens of creates a file or directory */  /* Opens or creates a file or directory */
172  NTSTATUS  static NTSTATUS
173  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,  disk_create(uint32 device_id, uint32 accessmask, uint32 sharemode, uint32 create_disposition,
174              uint32 flags_and_attributes, char *filename, HANDLE * phandle)              uint32 flags_and_attributes, char *filename, HANDLE * phandle)
175  {  {
# Line 165  disk_create(uint32 device_id, uint32 acc Line 177  disk_create(uint32 device_id, uint32 acc
177          DIR *dirp;          DIR *dirp;
178          int flags, mode;          int flags, mode;
179          char path[256];          char path[256];
180            struct stat filestat;
181    
182          handle = 0;          handle = 0;
183          dirp = NULL;          dirp = NULL;
184          flags = 0;          flags = 0;
185          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;          mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
186    
187    
188          if (filename[strlen(filename) - 1] == '/')          if (filename[strlen(filename) - 1] == '/')
189                  filename[strlen(filename) - 1] = 0;                  filename[strlen(filename) - 1] = 0;
190          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);  
191    
192          switch (create_disposition)          switch (create_disposition)
193          {          {
# Line 209  disk_create(uint32 device_id, uint32 acc Line 222  disk_create(uint32 device_id, uint32 acc
222                          break;                          break;
223          }          }
224    
225            //printf("Open: \"%s\"  flags: %u, accessmask: %u sharemode: %u create disp: %u\n", path, flags_and_attributes, accessmask, sharemode, create_disposition);
226    
227            // Get information about file and set that flag ourselfs
228            if ((stat(path, &filestat) == 0) && (S_ISDIR(filestat.st_mode)))
229            {
230                    if (flags_and_attributes & FILE_NON_DIRECTORY_FILE)
231                            return STATUS_FILE_IS_A_DIRECTORY;
232                    else
233                            flags_and_attributes |= FILE_DIRECTORY_FILE;
234            }
235    
236          if (flags_and_attributes & FILE_DIRECTORY_FILE)          if (flags_and_attributes & FILE_DIRECTORY_FILE)
237          {          {
238                  if (flags & O_CREAT)                  if (flags & O_CREAT)
# Line 239  disk_create(uint32 device_id, uint32 acc Line 263  disk_create(uint32 device_id, uint32 acc
263          }          }
264          else          else
265          {          {
266    
267                  if (accessmask & GENERIC_ALL                  if (accessmask & GENERIC_ALL
268                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))                      || (accessmask & GENERIC_READ && accessmask & GENERIC_WRITE))
269                  {                  {
# Line 258  disk_create(uint32 device_id, uint32 acc Line 283  disk_create(uint32 device_id, uint32 acc
283                  {                  {
284                          switch (errno)                          switch (errno)
285                          {                          {
286                                    case EISDIR:
287    
288                                            return STATUS_FILE_IS_A_DIRECTORY;
289    
290                                  case EACCES:                                  case EACCES:
291    
292                                          return STATUS_ACCESS_DENIED;                                          return STATUS_ACCESS_DENIED;
# Line 265  disk_create(uint32 device_id, uint32 acc Line 294  disk_create(uint32 device_id, uint32 acc
294                                  case ENOENT:                                  case ENOENT:
295    
296                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
297                                    case EEXIST:
298    
299                                            return STATUS_OBJECT_NAME_COLLISION;
300                                  default:                                  default:
301    
302                                          perror("open");                                          perror("open");
303                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
304                          }                          }
305                  }                  }
306    
307                    /* all read and writes of files should be non blocking */
308                    if (fcntl(handle, F_SETFL, O_NONBLOCK) == -1)
309                            perror("fcntl");
310          }          }
311    
312          if (handle >= MAX_OPEN_FILES)          if (handle >= MAX_OPEN_FILES)
# Line 291  disk_create(uint32 device_id, uint32 acc Line 326  disk_create(uint32 device_id, uint32 acc
326          return STATUS_SUCCESS;          return STATUS_SUCCESS;
327  }  }
328    
329  NTSTATUS  static NTSTATUS
330  disk_close(HANDLE handle)  disk_close(HANDLE handle)
331  {  {
332          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
# Line 311  disk_close(HANDLE handle) Line 346  disk_close(HANDLE handle)
346          return STATUS_SUCCESS;          return STATUS_SUCCESS;
347  }  }
348    
349  NTSTATUS  static NTSTATUS
350  disk_read(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_read(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
351  {  {
352          int n;          int n;
353    
354          if (offset)  #if 0
355                  lseek(handle, offset, SEEK_SET);          /* browsing dir ????        */
356            /* each request is 24 bytes */
357            if (g_fileinfo[handle].flags_and_attributes & FILE_DIRECTORY_FILE)
358            {
359                    *result = 0;
360                    return STATUS_SUCCESS;
361            }
362    #endif
363    
364            lseek(handle, offset, SEEK_SET);
365    
366          n = read(handle, data, length);          n = read(handle, data, length);
367    
368          if (n < 0)          if (n < 0)
369          {          {
                 perror("read");  
370                  *result = 0;                  *result = 0;
371                  return STATUS_INVALID_PARAMETER;                  switch (errno)
372                    {
373                            case EISDIR:
374                                    return STATUS_FILE_IS_A_DIRECTORY;
375                            default:
376                                    perror("read");
377                                    return STATUS_INVALID_PARAMETER;
378                    }
379          }          }
380    
381          *result = n;          *result = n;
# Line 332  disk_read(HANDLE handle, uint8 * data, u Line 383  disk_read(HANDLE handle, uint8 * data, u
383          return STATUS_SUCCESS;          return STATUS_SUCCESS;
384  }  }
385    
386  NTSTATUS  static NTSTATUS
387  disk_write(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)  disk_write(HANDLE handle, uint8 * data, uint32 length, uint32 offset, uint32 * result)
388  {  {
389          int n;          int n;
390    
391          if (offset)          lseek(handle, offset, SEEK_SET);
                 lseek(handle, offset, SEEK_SET);  
392    
393          n = write(handle, data, length);          n = write(handle, data, length);
394    
# Line 346  disk_write(HANDLE handle, uint8 * data, Line 396  disk_write(HANDLE handle, uint8 * data,
396          {          {
397                  perror("write");                  perror("write");
398                  *result = 0;                  *result = 0;
399                  return STATUS_ACCESS_DENIED;                  switch (errno)
400                    {
401                            case ENOSPC:
402                                    return STATUS_DISK_FULL;
403                            default:
404                                    return STATUS_ACCESS_DENIED;
405                    }
406          }          }
407    
408          *result = n;          *result = n;
# Line 374  disk_query_information(HANDLE handle, ui Line 430  disk_query_information(HANDLE handle, ui
430          // Set file attributes          // Set file attributes
431          file_attributes = 0;          file_attributes = 0;
432          if (S_ISDIR(filestat.st_mode))          if (S_ISDIR(filestat.st_mode))
         {  
433                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
434          }  
435          filename = 1 + strrchr(path, '/');          filename = 1 + strrchr(path, '/');
436          if (filename && filename[0] == '.')          if (filename && filename[0] == '.')
         {  
437                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
438          }  
439            if (!file_attributes)
440                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
441    
442            if (!(filestat.st_mode & S_IWUSR))
443                    file_attributes |= FILE_ATTRIBUTE_READONLY;
444    
445          // Return requested data          // Return requested data
446          switch (info_class)          switch (info_class)
447          {          {
448                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
449                            seconds_since_1970_to_filetime(get_create_time(&filestat), &ft_high,
450                          out_uint8s(out, 8);     //create_time not available;                                                         &ft_low);
451                            out_uint32_le(out, ft_low);     //create_access_time
452                            out_uint32_le(out, ft_high);
453    
454                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(filestat.st_atime, &ft_high, &ft_low);
455                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     //last_access_time
# Line 398  disk_query_information(HANDLE handle, ui Line 459  disk_query_information(HANDLE handle, ui
459                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     //last_write_time
460                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
461    
462                          out_uint8s(out, 8);     //unknown zero                          seconds_since_1970_to_filetime(filestat.st_ctime, &ft_high, &ft_low);
463                            out_uint32_le(out, ft_low);     //last_change_time
464                            out_uint32_le(out, ft_high);
465    
466                          out_uint32_le(out, file_attributes);                          out_uint32_le(out, file_attributes);
467                          break;                          break;
468    
469                  case 5: /* FileStandardInformation */                  case FileStandardInformation:
470    
471                          out_uint32_le(out, filestat.st_size);   //Allocation size                          out_uint32_le(out, filestat.st_size);   //Allocation size
472                          out_uint32_le(out, 0);                          out_uint32_le(out, 0);
# Line 413  disk_query_information(HANDLE handle, ui Line 477  disk_query_information(HANDLE handle, ui
477                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory
478                          break;                          break;
479    
480                  case 35:        /* FileObjectIdInformation */                  case FileObjectIdInformation:
481    
482                          out_uint32_le(out, file_attributes);    /* File Attributes */                          out_uint32_le(out, file_attributes);    /* File Attributes */
483                          out_uint32_le(out, 0);  /* Reparse Tag */                          out_uint32_le(out, 0);  /* Reparse Tag */
# Line 434  disk_set_information(HANDLE handle, uint Line 498  disk_set_information(HANDLE handle, uint
498          char newname[256], fullpath[256];          char newname[256], fullpath[256];
499          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
500    
501            int mode;
502            struct stat filestat;
503            time_t write_time, change_time, access_time, mod_time;
504            struct utimbuf tvs;
505            struct STATFS_T stat_fs;
506    
507          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
508    
509          switch (info_class)          switch (info_class)
510          {          {
511                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
512                            write_time = change_time = access_time = 0;
513    
514                            in_uint8s(in, 4);       /* Handle of root dir? */
515                            in_uint8s(in, 24);      /* unknown */
516    
517                            // CreationTime
518                            in_uint32_le(in, ft_low);
519                            in_uint32_le(in, ft_high);
520    
521                            // AccessTime
522                            in_uint32_le(in, ft_low);
523                            in_uint32_le(in, ft_high);
524                            if (ft_low || ft_high)
525                                    access_time = convert_1970_to_filetime(ft_high, ft_low);
526    
527                            // WriteTime
528                            in_uint32_le(in, ft_low);
529                            in_uint32_le(in, ft_high);
530                            if (ft_low || ft_high)
531                                    write_time = convert_1970_to_filetime(ft_high, ft_low);
532    
533                            // ChangeTime
534                            in_uint32_le(in, ft_low);
535                            in_uint32_le(in, ft_high);
536                            if (ft_low || ft_high)
537                                    change_time = convert_1970_to_filetime(ft_high, ft_low);
538    
539                            in_uint32_le(in, file_attributes);
540    
541                            if (fstat(handle, &filestat))
542                                    return STATUS_ACCESS_DENIED;
543    
544                            tvs.modtime = filestat.st_mtime;
545                            tvs.actime = filestat.st_atime;
546                            if (access_time)
547                                    tvs.actime = access_time;
548    
549    
550                            if (write_time || change_time)
551                                    mod_time = MIN(write_time, change_time);
552                            else
553                                    mod_time = write_time ? write_time : change_time;
554    
555                            if (mod_time)
556                                    tvs.modtime = mod_time;
557    
558    
559                            if (access_time || write_time || change_time)
560                            {
561    #if WITH_DEBUG_RDP5
562                                    printf("FileBasicInformation access       time %s",
563                                           ctime(&tvs.actime));
564                                    printf("FileBasicInformation modification time %s",
565                                           ctime(&tvs.modtime));
566    #endif
567                                    if (utime(pfinfo->path, &tvs))
568                                            return STATUS_ACCESS_DENIED;
569                            }
570    
571                            if (!file_attributes)
572                                    break;  // not valid
573    
574                            mode = filestat.st_mode;
575    
576                            if (file_attributes & FILE_ATTRIBUTE_READONLY)
577                                    mode &= ~(S_IWUSR | S_IWGRP | S_IWOTH);
578                            else
579                                    mode |= S_IWUSR;
580    
581                            mode &= 0777;
582    #if WITH_DEBUG_RDP5
583                            printf("FileBasicInformation set access mode 0%o", mode);
584    #endif
585    
586                            if (fchmod(handle, mode))
587                                    return STATUS_ACCESS_DENIED;
588    
                         // Probably safe to ignore  
589                          break;                          break;
590    
591                  case 10:        /* FileRenameInformation */                  case FileRenameInformation:
592    
593                          in_uint8s(in, 4);       /* Handle of root dir? */                          in_uint8s(in, 4);       /* Handle of root dir? */
594                          in_uint8s(in, 0x1a);    /* unknown */                          in_uint8s(in, 0x1a);    /* unknown */
# Line 469  disk_set_information(HANDLE handle, uint Line 614  disk_set_information(HANDLE handle, uint
614                          }                          }
615                          break;                          break;
616    
617                  case 13:        /* FileDispositionInformation */                  case FileDispositionInformation:
618                            /* As far as I understand it, the correct
619                          //unimpl("IRP Set File Information class: FileDispositionInformation\n");                             thing to do here is to *schedule* a delete,
620                          // in_uint32_le(in, delete_on_close);                             so it will be deleted when the file is
621                          // disk_close(handle);                             closed. Subsequent
622                          unlink(pfinfo->path);                             FileDispositionInformation requests with
623                          break;                             DeleteFile set to FALSE should unschedule
624                               the delete. See
625                               http://www.osronline.com/article.cfm?article=245. Currently,
626                               we are deleting the file immediately. I
627                               guess this is a FIXME. */
628    
629                            //in_uint32_le(in, delete_on_close);
630    
631                            /* Make sure we close the file before
632                               unlinking it. Not doing so would trigger
633                               silly-delete if using NFS, which might fail
634                               on FAT floppies, for example. */
635                            disk_close(handle);
636    
637                  case 19:        /* FileAllocationInformation */                          if ((pfinfo->flags_and_attributes & FILE_DIRECTORY_FILE))       // remove a directory
638                            {
639                                    if (rmdir(pfinfo->path) < 0)
640                                            return STATUS_ACCESS_DENIED;
641                            }
642                            else if (unlink(pfinfo->path) < 0)      // unlink a file
643                                    return STATUS_ACCESS_DENIED;
644    
                         unimpl("IRP Set File Information class: FileAllocationInformation\n");  
645                          break;                          break;
646    
647                  case 20:        /* FileEndOfFileInformation */                  case FileAllocationInformation:
648                            /* Fall through to FileEndOfFileInformation,
649                               which uses ftrunc. This is like Samba with
650                               "strict allocation = false", and means that
651                               we won't detect out-of-quota errors, for
652                               example. */
653    
654                    case FileEndOfFileInformation:
655                            in_uint8s(in, 28);      /* unknown */
656                            in_uint32_le(in, length);       /* file size */
657    
658                            /* prevents start of writing if not enough space left on device */
659                            if (STATFS_FN(g_rdpdr_device[pfinfo->device_id].local_path, &stat_fs) == 0)
660                                    if (stat_fs.f_bsize * stat_fs.f_bfree < length)
661                                            return STATUS_DISK_FULL;
662    
663                            /* FIXME: Growing file with ftruncate doesn't
664                               work with Linux FAT fs */
665                            if (ftruncate(handle, length) != 0)
666                            {
667                                    perror("ftruncate");
668                                    return STATUS_DISK_FULL;
669                            }
670    
                         unimpl("IRP Set File Information class: FileEndOfFileInformation\n");  
671                          break;                          break;
   
672                  default:                  default:
673    
674                          unimpl("IRP Set File Information class: 0x%x\n", info_class);                          unimpl("IRP Set File Information class: 0x%x\n", info_class);
# Line 495  disk_set_information(HANDLE handle, uint Line 677  disk_set_information(HANDLE handle, uint
677          return STATUS_SUCCESS;          return STATUS_SUCCESS;
678  }  }
679    
680  int fsstat(const char *path, struct fsinfo *buf)  static FsInfoType *
681    FsVolumeInfo(char *fpath)
682  {  {
         int ret;  
 #if defined(HAVE_STATFS)  
         struct statfs statbuf;  
 #elif defined(HAVE_STATVFS)  
         struct statvfs statbuf;  
 #endif  
683    
684  #if defined(HAVE_STATFS)  #ifdef HAVE_MNTENT_H
685          ret = statfs(path, &statbuf);          FILE *fdfs;
686          buf->f_namelen = statbuf.f_namelen;          struct mntent *e;
687  #elif defined(HAVE_STATVFS)          static FsInfoType info;
688          ret = statvfs(path, &statbuf);  
689          buf->f_namelen = statbuf.f_namemax;          /* initialize */
690            memset(&info, 0, sizeof(info));
691            strcpy(info.label, "RDESKTOP");
692            strcpy(info.type, "RDPFS");
693    
694            fdfs = setmntent(MNTENT_PATH, "r");
695            if (!fdfs)
696                    return &info;
697    
698            while ((e = getmntent(fdfs)))
699            {
700                    if (strncmp(fpath, e->mnt_dir, strlen(fpath)) == 0)
701                    {
702                            strcpy(info.type, e->mnt_type);
703                            strcpy(info.name, e->mnt_fsname);
704                            if (strstr(e->mnt_opts, "vfat") || strstr(e->mnt_opts, "iso9660"))
705                            {
706                                    int fd = open(e->mnt_fsname, O_RDONLY);
707                                    if (fd >= 0)
708                                    {
709                                            unsigned char buf[512];
710                                            memset(buf, 0, sizeof(buf));
711                                            if (strstr(e->mnt_opts, "vfat"))
712                                                     /*FAT*/
713                                            {
714                                                    strcpy(info.type, "vfat");
715                                                    read(fd, buf, sizeof(buf));
716                                                    info.serial =
717                                                            (buf[42] << 24) + (buf[41] << 16) +
718                                                            (buf[40] << 8) + buf[39];
719                                                    strncpy(info.label, buf + 43, 10);
720                                                    info.label[10] = '\0';
721                                            }
722                                            else if (lseek(fd, 32767, SEEK_SET) >= 0)       /* ISO9660 */
723                                            {
724                                                    read(fd, buf, sizeof(buf));
725                                                    strncpy(info.label, buf + 41, 32);
726                                                    info.label[32] = '\0';
727                                                    //info.Serial = (buf[128]<<24)+(buf[127]<<16)+(buf[126]<<8)+buf[125];
728                                            }
729                                            close(fd);
730                                    }
731                            }
732                    }
733            }
734            endmntent(fdfs);
735  #else  #else
736          ret=-1;          static FsInfoType info;
 #endif  
737    
738          buf->f_blocks = statbuf.f_blocks;          /* initialize */
739          buf->f_bfree = statbuf.f_bfree;          memset(&info, 0, sizeof(info));
740          buf->f_bsize = statbuf.f_bsize;          strcpy(info.label, "RDESKTOP");
741            strcpy(info.type, "RDPFS");
742    
743          return ret;  #endif
744            return &info;
745  }  }
746    
747    
748  NTSTATUS  NTSTATUS
749  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)
750  {  {
751          char *volume, *fs_type;          struct STATFS_T stat_fs;
         struct fsinfo stat_fs;  
752          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
753            FsInfoType *fsinfo;
754    
755          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
         volume = "RDESKTOP";  
         fs_type = "RDPFS";  
756    
757          if (fsstat(pfinfo->path, &stat_fs) != 0)        /* FIXME: statfs is not portable */          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)
758          {          {
759                  perror("statfs");                  perror("statfs");
760                  return STATUS_ACCESS_DENIED;                  return STATUS_ACCESS_DENIED;
761          }          }
762    
763            fsinfo = FsVolumeInfo(pfinfo->path);
764    
765          switch (info_class)          switch (info_class)
766          {          {
767                  case 1: /* FileFsVolumeInformation */                  case FileFsVolumeInformation:
768    
769                          out_uint32_le(out, 0);  /* volume creation time low */                          out_uint32_le(out, 0);  /* volume creation time low */
770                          out_uint32_le(out, 0);  /* volume creation time high */                          out_uint32_le(out, 0);  /* volume creation time high */
771                          out_uint32_le(out, 0);  /* serial */                          out_uint32_le(out, fsinfo->serial);     /* serial */
772                          out_uint32_le(out, 2 * strlen(volume)); /* length of string */  
773                            out_uint32_le(out, 2 * strlen(fsinfo->label));  /* length of string */
774    
775                          out_uint8(out, 0);      /* support objects? */                          out_uint8(out, 0);      /* support objects? */
776                          rdp_out_unistr(out, volume, 2 * strlen(volume) - 2);                          rdp_out_unistr(out, fsinfo->label, 2 * strlen(fsinfo->label) - 2);
777                          break;                          break;
778    
779                  case 3: /* FileFsSizeInformation */                  case FileFsSizeInformation:
780    
781                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */
782                          out_uint32_le(out, 0);  /* Total allocation high units */                          out_uint32_le(out, 0);  /* Total allocation high units */
# Line 560  disk_query_volume_information(HANDLE han Line 786  disk_query_volume_information(HANDLE han
786                          out_uint32_le(out, 0x200);      /* Bytes per sector */                          out_uint32_le(out, 0x200);      /* Bytes per sector */
787                          break;                          break;
788    
789                  case 5: /* FileFsAttributeInformation */                  case FileFsAttributeInformation:
790    
791                          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 */
792                          out_uint32_le(out, stat_fs.f_namelen);  /* max length of filename */                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */
793                          out_uint32_le(out, 2 * strlen(fs_type));        /* length of fs_type */  
794                          rdp_out_unistr(out, fs_type, 2 * strlen(fs_type) - 2);                          out_uint32_le(out, 2 * strlen(fsinfo->type));   /* length of fs_type */
795                            rdp_out_unistr(out, fsinfo->type, 2 * strlen(fsinfo->type) - 2);
796                          break;                          break;
797    
798                  case 2: /* FileFsLabelInformation */                  case FileFsLabelInformation:
799                  case 4: /* FileFsDeviceInformation */                  case FileFsDeviceInformation:
800                  case 6: /* FileFsControlInformation */                  case FileFsControlInformation:
801                  case 7: /* FileFsFullSizeInformation */                  case FileFsFullSizeInformation:
802                  case 8: /* FileFsObjectIdInformation */                  case FileFsObjectIdInformation:
803                  case 9: /* FileFsMaximumInformation */                  case FileFsMaximumInformation:
804    
805                  default:                  default:
806    
807                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);
# Line 599  disk_query_directory(HANDLE handle, uint Line 827  disk_query_directory(HANDLE handle, uint
827    
828          switch (info_class)          switch (info_class)
829          {          {
830                  case 3: //FIXME: Why 3?                  case FileBothDirectoryInformation:
831    
832                          // If a search pattern is received, remember this pattern, and restart search                          // If a search pattern is received, remember this pattern, and restart search
833                          if (pattern[0] != 0)                          if (pattern[0] != 0)
# Line 611  disk_query_directory(HANDLE handle, uint Line 839  disk_query_directory(HANDLE handle, uint
839                          // find next dirent matching pattern                          // find next dirent matching pattern
840                          pdirent = readdir(pdir);                          pdirent = readdir(pdir);
841                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)                          while (pdirent && fnmatch(pfinfo->pattern, pdirent->d_name, 0) != 0)
                         {  
842                                  pdirent = readdir(pdir);                                  pdirent = readdir(pdir);
                         }  
843    
844                          if (pdirent == NULL)                          if (pdirent == NULL)
                         {  
845                                  return STATUS_NO_MORE_FILES;                                  return STATUS_NO_MORE_FILES;
                         }  
846    
847                          // Get information for directory entry                          // Get information for directory entry
848                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);                          sprintf(fullpath, "%s/%s", dirname, pdirent->d_name);
849                          /* JIF  
                            printf("Stat: %s\n", fullpath); */  
850                          if (stat(fullpath, &fstat))                          if (stat(fullpath, &fstat))
851                          {                          {
852                                  perror("stat");                                  switch (errno)
853                                  out_uint8(out, 0);                                  {
854                                  return STATUS_ACCESS_DENIED;                                          case ENOENT:
855                                            case ELOOP:
856                                            case EACCES:
857                                                    /* These are non-fatal errors. */
858                                                    memset(&fstat, 0, sizeof(fstat));
859                                                    break;
860                                            default:
861                                                    /* Fatal error. By returning STATUS_NO_SUCH_FILE,
862                                                       the directory list operation will be aborted */
863                                                    perror(fullpath);
864                                                    out_uint8(out, 0);
865                                                    return STATUS_NO_SUCH_FILE;
866                                    }
867                          }                          }
868    
869                          if (S_ISDIR(fstat.st_mode))                          if (S_ISDIR(fstat.st_mode))
870                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;                                  file_attributes |= FILE_ATTRIBUTE_DIRECTORY;
871                          if (pdirent->d_name[0] == '.')                          if (pdirent->d_name[0] == '.')
872                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;                                  file_attributes |= FILE_ATTRIBUTE_HIDDEN;
873                            if (!file_attributes)
874                                    file_attributes |= FILE_ATTRIBUTE_NORMAL;
875                            if (!(fstat.st_mode & S_IWUSR))
876                                    file_attributes |= FILE_ATTRIBUTE_READONLY;
877    
878                          // Return requested information                          // Return requested information
879                          out_uint8s(out, 8);     //unknown zero                          out_uint8s(out, 8);     //unknown zero
880                          out_uint8s(out, 8);     //create_time not available in posix;  
881                            seconds_since_1970_to_filetime(get_create_time(&fstat), &ft_high, &ft_low);
882                            out_uint32_le(out, ft_low);     // create time
883                            out_uint32_le(out, ft_high);
884    
885                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);                          seconds_since_1970_to_filetime(fstat.st_atime, &ft_high, &ft_low);
886                          out_uint32_le(out, ft_low);     //last_access_time                          out_uint32_le(out, ft_low);     //last_access_time
# Line 648  disk_query_directory(HANDLE handle, uint Line 890  disk_query_directory(HANDLE handle, uint
890                          out_uint32_le(out, ft_low);     //last_write_time                          out_uint32_le(out, ft_low);     //last_write_time
891                          out_uint32_le(out, ft_high);                          out_uint32_le(out, ft_high);
892    
893                          out_uint8s(out, 8);     //unknown zero                          seconds_since_1970_to_filetime(fstat.st_ctime, &ft_high, &ft_low);
894                            out_uint32_le(out, ft_low);     //change_write_time
895                            out_uint32_le(out, ft_high);
896    
897                          out_uint32_le(out, fstat.st_size);      //filesize low                          out_uint32_le(out, fstat.st_size);      //filesize low
898                          out_uint32_le(out, 0);  //filesize high                          out_uint32_le(out, 0);  //filesize high
899                          out_uint32_le(out, fstat.st_size);      //filesize low                          out_uint32_le(out, fstat.st_size);      //filesize low
# Line 662  disk_query_directory(HANDLE handle, uint Line 907  disk_query_directory(HANDLE handle, uint
907                          break;                          break;
908    
909                  default:                  default:
910                            /* FIXME: Support FileDirectoryInformation,
911                               FileFullDirectoryInformation, and
912                               FileNamesInformation */
913    
914                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);                          unimpl("IRP Query Directory sub: 0x%x\n", info_class);
915                          return STATUS_INVALID_PARAMETER;                          return STATUS_INVALID_PARAMETER;
# Line 670  disk_query_directory(HANDLE handle, uint Line 918  disk_query_directory(HANDLE handle, uint
918          return STATUS_SUCCESS;          return STATUS_SUCCESS;
919  }  }
920    
921    
922    
923    static NTSTATUS
924    disk_device_control(HANDLE handle, uint32 request, STREAM in, STREAM out)
925    {
926            uint32 result;
927    
928            if (((request >> 16) != 20) || ((request >> 16) != 9))
929                    return STATUS_INVALID_PARAMETER;
930    
931            /* extract operation */
932            request >>= 2;
933            request &= 0xfff;
934    
935            printf("DISK IOCTL %d\n", request);
936    
937            switch (request)
938            {
939                    case 25:        // ?
940                    case 42:        // ?
941                    default:
942                            unimpl("DISK IOCTL %d\n", request);
943                            return STATUS_INVALID_PARAMETER;
944            }
945    
946            return STATUS_SUCCESS;
947    }
948    
949  DEVICE_FNS disk_fns = {  DEVICE_FNS disk_fns = {
950          disk_create,          disk_create,
951          disk_close,          disk_close,
952          disk_read,          disk_read,
953          disk_write,          disk_write,
954          NULL                    /* device_control */          disk_device_control     /* device_control */
955  };  };

Legend:
Removed from v.572  
changed lines
  Added in v.745

  ViewVC Help
Powered by ViewVC 1.1.26