/[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 601 by n-ki, Fri Feb 6 12:57:26 2004 UTC revision 659 by astrand, Fri Apr 16 11:37:24 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_UNKNOWNXXX0              0x00000060      /* ??? ACTION i.e. 0x860 == compress this file ? */  
 #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_FLAG_OPEN_NO_RECALL                0x00100000  
 #define FILE_FLAG_OPEN_REPARSE_POINT            0x00200000  
 #define FILE_FLAG_POSIX_SEMANTICS               0x01000000  
 #define FILE_FLAG_BACKUP_SEMANTICS              0x02000000      /* sometimes used to create a directory */  
 #define FILE_FLAG_DELETE_ON_CLOSE               0x04000000  
 #define FILE_FLAG_SEQUENTIAL_SCAN               0x08000000  
 #define FILE_FLAG_RANDOM_ACCESS                 0x10000000  
 #define FILE_FLAG_NO_BUFFERING                  0x20000000  
 #define FILE_FLAG_OVERLAPPED                    0x40000000  
 #define FILE_FLAG_WRITE_THROUGH                 0x80000000  
   
 #define FILE_SHARE_READ                         0x01  
 #define FILE_SHARE_WRITE                        0x02  
 #define FILE_SHARE_DELETE                       0x04  
   
 #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  
22    
23  #if (defined(sun) && (defined(__svr4__) || defined(__SVR4)))  #if (defined(sun) && (defined(__svr4__) || defined(__SVR4)))
24  #define SOLARIS  #define SOLARIS
25  #endif  #endif
26    
27  #ifdef SOLARIS  #if (defined(SOLARIS) || defined(__hpux))
28  #define DIRFD(a) ((a)->dd_fd)  #define DIRFD(a) ((a)->dd_fd)
29  #else  #else
30  #define DIRFD(a) (dirfd(a))  #define DIRFD(a) (dirfd(a))
# Line 102  Line 42 
42  #include <time.h>               /* ctime */  #include <time.h>               /* ctime */
43    
44    
45  #if defined(SOLARIS)  #if (defined(SOLARIS) || defined (__hpux) || defined(__BEOS__))
46  #include <sys/statvfs.h>        /* solaris statvfs */  #include <sys/statvfs.h>        /* solaris statvfs */
47    #include <sys/mntent.h>
48    /* TODO: Fix mntent-handling for solaris */
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)
# Line 117  Line 61 
61    
62  #else  #else
63  #include <sys/vfs.h>            /* linux statfs */  #include <sys/vfs.h>            /* linux statfs */
64    #include <mntent.h>
65    #define HAVE_MNTENT_H
66    #define MNTENT_PATH "/etc/mtab"
67  #define STATFS_FN(path, buf) (statfs(path,buf))  #define STATFS_FN(path, buf) (statfs(path,buf))
68  #define STATFS_T statfs  #define STATFS_T statfs
69  #define F_NAMELEN(buf) ((buf).f_namelen)  #define F_NAMELEN(buf) ((buf).f_namelen)
# Line 126  Line 73 
73    
74  extern RDPDR_DEVICE g_rdpdr_device[];  extern RDPDR_DEVICE g_rdpdr_device[];
75    
76  struct fileinfo  FILEINFO g_fileinfo[MAX_OPEN_FILES];
77    
78    typedef struct
79  {  {
80          uint32 device_id, flags_and_attributes;          char name[256];
81          char path[256];          char label[256];
82          DIR *pdir;          unsigned long serial;
83          struct dirent *pdirent;          char type[256];
84          char pattern[64];  } FsInfoType;
         BOOL delete_on_close;  
 }  
 g_fileinfo[MAX_OPEN_FILES];  
85    
86    
87  time_t  time_t
# Line 182  convert_1970_to_filetime(uint32 high, ui Line 128  convert_1970_to_filetime(uint32 high, ui
128    
129  /* Enumeration of devices from rdesktop.c        */  /* Enumeration of devices from rdesktop.c        */
130  /* returns numer of units found and initialized. */  /* returns numer of units found and initialized. */
131  /* optarg looks like ':h:=/mnt/floppy,b:=/mnt/usbdevice1' */  /* optarg looks like ':h=/mnt/floppy,b=/mnt/usbdevice1' */
132  /* when it arrives to this function.             */  /* when it arrives to this function.             */
133  int  int
134  disk_enum_devices(int *id, char *optarg)  disk_enum_devices(uint32 * id, char *optarg)
135  {  {
136          char *pos = optarg;          char *pos = optarg;
137          char *pos2;          char *pos2;
# Line 341  disk_create(uint32 device_id, uint32 acc Line 287  disk_create(uint32 device_id, uint32 acc
287                                  case ENOENT:                                  case ENOENT:
288    
289                                          return STATUS_NO_SUCH_FILE;                                          return STATUS_NO_SUCH_FILE;
290                                    case EEXIST:
291    
292                                            return STATUS_OBJECT_NAME_COLLISION;
293                                  default:                                  default:
294    
295                                          perror("open");                                          perror("open");
# Line 405  disk_read(HANDLE handle, uint8 * data, u Line 354  disk_read(HANDLE handle, uint8 * data, u
354          }          }
355  #endif  #endif
356    
357          if (offset)          lseek(handle, offset, SEEK_SET);
358                  lseek(handle, offset, SEEK_SET);  
359          n = read(handle, data, length);          n = read(handle, data, length);
360    
361          if (n < 0)          if (n < 0)
# Line 432  disk_write(HANDLE handle, uint8 * data, Line 381  disk_write(HANDLE handle, uint8 * data,
381  {  {
382          int n;          int n;
383    
384          if (offset)          lseek(handle, offset, SEEK_SET);
                 lseek(handle, offset, SEEK_SET);  
385    
386          n = write(handle, data, length);          n = write(handle, data, length);
387    
# Line 490  disk_query_information(HANDLE handle, ui Line 438  disk_query_information(HANDLE handle, ui
438          // Return requested data          // Return requested data
439          switch (info_class)          switch (info_class)
440          {          {
441                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
442                          seconds_since_1970_to_filetime(get_create_time(&filestat), &ft_high,                          seconds_since_1970_to_filetime(get_create_time(&filestat), &ft_high,
443                                                         &ft_low);                                                         &ft_low);
444                          out_uint32_le(out, ft_low);     //create_access_time                          out_uint32_le(out, ft_low);     //create_access_time
# Line 511  disk_query_information(HANDLE handle, ui Line 459  disk_query_information(HANDLE handle, ui
459                          out_uint32_le(out, file_attributes);                          out_uint32_le(out, file_attributes);
460                          break;                          break;
461    
462                  case 5: /* FileStandardInformation */                  case FileStandardInformation:
463    
464                          out_uint32_le(out, filestat.st_size);   //Allocation size                          out_uint32_le(out, filestat.st_size);   //Allocation size
465                          out_uint32_le(out, 0);                          out_uint32_le(out, 0);
# Line 522  disk_query_information(HANDLE handle, ui Line 470  disk_query_information(HANDLE handle, ui
470                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory                          out_uint8(out, S_ISDIR(filestat.st_mode) ? 1 : 0);      //Directory
471                          break;                          break;
472    
473                  case 35:        /* FileObjectIdInformation */                  case FileObjectIdInformation:
474    
475                          out_uint32_le(out, file_attributes);    /* File Attributes */                          out_uint32_le(out, file_attributes);    /* File Attributes */
476                          out_uint32_le(out, 0);  /* Reparse Tag */                          out_uint32_le(out, 0);  /* Reparse Tag */
# Line 547  disk_set_information(HANDLE handle, uint Line 495  disk_set_information(HANDLE handle, uint
495          struct stat filestat;          struct stat filestat;
496          time_t write_time, change_time, access_time, mod_time;          time_t write_time, change_time, access_time, mod_time;
497          struct utimbuf tvs;          struct utimbuf tvs;
498            struct STATFS_T stat_fs;
499    
500          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
501    
502          switch (info_class)          switch (info_class)
503          {          {
504                  case 4: /* FileBasicInformation */                  case FileBasicInformation:
505                          write_time = change_time = access_time = 0;                          write_time = change_time = access_time = 0;
506    
507                          in_uint8s(in, 4);       /* Handle of root dir? */                          in_uint8s(in, 4);       /* Handle of root dir? */
# Line 629  disk_set_information(HANDLE handle, uint Line 578  disk_set_information(HANDLE handle, uint
578    
579                          if (fchmod(handle, mode))                          if (fchmod(handle, mode))
580                                  return STATUS_ACCESS_DENIED;                                  return STATUS_ACCESS_DENIED;
581    
582                          break;                          break;
583    
584                  case 10:        /* FileRenameInformation */                  case FileRenameInformation:
585    
586                          in_uint8s(in, 4);       /* Handle of root dir? */                          in_uint8s(in, 4);       /* Handle of root dir? */
587                          in_uint8s(in, 0x1a);    /* unknown */                          in_uint8s(in, 0x1a);    /* unknown */
# Line 657  disk_set_information(HANDLE handle, uint Line 607  disk_set_information(HANDLE handle, uint
607                          }                          }
608                          break;                          break;
609    
610                  case 13:        /* FileDispositionInformation */                  case FileDispositionInformation:
611                            /* As far as I understand it, the correct
612                          //unimpl("IRP Set File Information class: FileDispositionInformation\n");                             thing to do here is to *schedule* a delete,
613                               so it will be deleted when the file is
614                               closed. Subsequent
615                               FileDispositionInformation requests with
616                               DeleteFile set to FALSE should unschedule
617                               the delete. See
618                               http://www.osronline.com/article.cfm?article=245. Currently,
619                               we are deleting the file immediately. I
620                               guess this is a FIXME. */
621    
622                          //in_uint32_le(in, delete_on_close);                          //in_uint32_le(in, delete_on_close);
623                          // disk_close(handle);  
624                            /* Make sure we close the file before
625                               unlinking it. Not doing so would trigger
626                               silly-delete if using NFS, which might fail
627                               on FAT floppies, for example. */
628                            disk_close(handle);
629    
630                          if ((pfinfo->flags_and_attributes & FILE_DIRECTORY_FILE))       // remove a directory                          if ((pfinfo->flags_and_attributes & FILE_DIRECTORY_FILE))       // remove a directory
631                          {                          {
632                                  if (rmdir(pfinfo->path) < 0)                                  if (rmdir(pfinfo->path) < 0)
# Line 673  disk_set_information(HANDLE handle, uint Line 637  disk_set_information(HANDLE handle, uint
637    
638                          break;                          break;
639    
640                  case 19:        /* FileAllocationInformation */                  case FileAllocationInformation:
641    
642                          unimpl("IRP Set File Information class: FileAllocationInformation\n");                          unimpl("IRP Set File Information class: FileAllocationInformation\n");
643                          break;                          break;
644    
645                  case 20:        /* FileEndOfFileInformation */                  case FileEndOfFileInformation:
646                          in_uint8s(in, 28);      /* unknown */                          in_uint8s(in, 28);      /* unknown */
647                          in_uint32_le(in, length);       /* file size */                          in_uint32_le(in, length);       /* file size */
648    
649                          printf("FileEndOfFileInformation length = %d\n", length);                          /* prevents start of writing if not enough space left on device */
650                          // ????????????                          if (STATFS_FN(g_rdpdr_device[pfinfo->device_id].local_path, &stat_fs) == 0)
651                                    if (stat_fs.f_bsize * stat_fs.f_bfree < length)
652                                            return STATUS_DISK_FULL;
653    
654                          unimpl("IRP Set File Information class: FileEndOfFileInformation\n");                          //printf("FileEndOfFileInformation length = %d\n", length);
655                            // ????????????
656                            //unimpl("IRP Set File Information class: FileEndOfFileInformation\n");
657                          break;                          break;
658                  default:                  default:
659    
# Line 695  disk_set_information(HANDLE handle, uint Line 663  disk_set_information(HANDLE handle, uint
663          return STATUS_SUCCESS;          return STATUS_SUCCESS;
664  }  }
665    
666    FsInfoType *
667    FsVolumeInfo(char *fpath)
668    {
669    
670    #ifdef HAVE_MNTENT_H
671            FILE *fdfs;
672            struct mntent *e;
673            static FsInfoType info;
674    
675            /* initialize */
676            memset(&info, 0, sizeof(info));
677            strcpy(info.label, "RDESKTOP");
678            strcpy(info.type, "RDPFS");
679    
680            fdfs = setmntent(MNTENT_PATH, "r");
681            if (!fdfs)
682                    return &info;
683    
684            while ((e = getmntent(fdfs)))
685            {
686                    if (strncmp(fpath, e->mnt_dir, strlen(fpath)) == 0)
687                    {
688                            strcpy(info.type, e->mnt_type);
689                            strcpy(info.name, e->mnt_fsname);
690                            if (strstr(e->mnt_opts, "vfat") || strstr(e->mnt_opts, "iso9660"))
691                            {
692                                    int fd = open(e->mnt_fsname, O_RDONLY);
693                                    if (fd >= 0)
694                                    {
695                                            unsigned char buf[512];
696                                            memset(buf, 0, sizeof(buf));
697                                            if (strstr(e->mnt_opts, "vfat"))
698                                                     /*FAT*/
699                                            {
700                                                    strcpy(info.type, "vfat");
701                                                    read(fd, buf, sizeof(buf));
702                                                    info.serial =
703                                                            (buf[42] << 24) + (buf[41] << 16) +
704                                                            (buf[40] << 8) + buf[39];
705                                                    strncpy(info.label, buf + 43, 10);
706                                                    info.label[10] = '\0';
707                                            }
708                                            else if (lseek(fd, 32767, SEEK_SET) >= 0)       /* ISO9660 */
709                                            {
710                                                    read(fd, buf, sizeof(buf));
711                                                    strncpy(info.label, buf + 41, 32);
712                                                    info.label[32] = '\0';
713                                                    //info.Serial = (buf[128]<<24)+(buf[127]<<16)+(buf[126]<<8)+buf[125];
714                                            }
715                                            close(fd);
716                                    }
717                            }
718                    }
719            }
720            endmntent(fdfs);
721    #else
722            static FsInfoType info;
723    
724            /* initialize */
725            memset(&info, 0, sizeof(info));
726            strcpy(info.label, "RDESKTOP");
727            strcpy(info.type, "RDPFS");
728    
729    #endif
730            return &info;
731    }
732    
733    
734  NTSTATUS  NTSTATUS
735  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)  disk_query_volume_information(HANDLE handle, uint32 info_class, STREAM out)
736  {  {
         char *volume, *fs_type;  
737          struct STATFS_T stat_fs;          struct STATFS_T stat_fs;
738          struct fileinfo *pfinfo;          struct fileinfo *pfinfo;
739            FsInfoType *fsinfo;
740    
741          pfinfo = &(g_fileinfo[handle]);          pfinfo = &(g_fileinfo[handle]);
         volume = "RDESKTOP";  
         fs_type = "RDPFS";  
742    
743          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)          if (STATFS_FN(pfinfo->path, &stat_fs) != 0)
744          {          {
# Line 712  disk_query_volume_information(HANDLE han Line 746  disk_query_volume_information(HANDLE han
746                  return STATUS_ACCESS_DENIED;                  return STATUS_ACCESS_DENIED;
747          }          }
748    
749            fsinfo = FsVolumeInfo(pfinfo->path);
750    
751          switch (info_class)          switch (info_class)
752          {          {
753                  case 1: /* FileFsVolumeInformation */                  case FileFsVolumeInformation:
754    
755                          out_uint32_le(out, 0);  /* volume creation time low */                          out_uint32_le(out, 0);  /* volume creation time low */
756                          out_uint32_le(out, 0);  /* volume creation time high */                          out_uint32_le(out, 0);  /* volume creation time high */
757                          out_uint32_le(out, 0);  /* serial */                          out_uint32_le(out, fsinfo->serial);     /* serial */
758                          out_uint32_le(out, 2 * strlen(volume)); /* length of string */  
759                            out_uint32_le(out, 2 * strlen(fsinfo->label));  /* length of string */
760    
761                          out_uint8(out, 0);      /* support objects? */                          out_uint8(out, 0);      /* support objects? */
762                          rdp_out_unistr(out, volume, 2 * strlen(volume) - 2);                          rdp_out_unistr(out, fsinfo->label, 2 * strlen(fsinfo->label) - 2);
763                          break;                          break;
764    
765                  case 3: /* FileFsSizeInformation */                  case FileFsSizeInformation:
766    
767                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */                          out_uint32_le(out, stat_fs.f_blocks);   /* Total allocation units low */
768                          out_uint32_le(out, 0);  /* Total allocation high units */                          out_uint32_le(out, 0);  /* Total allocation high units */
# Line 734  disk_query_volume_information(HANDLE han Line 772  disk_query_volume_information(HANDLE han
772                          out_uint32_le(out, 0x200);      /* Bytes per sector */                          out_uint32_le(out, 0x200);      /* Bytes per sector */
773                          break;                          break;
774    
775                  case 5: /* FileFsAttributeInformation */                  case FileFsAttributeInformation:
776    
777                          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 */
778                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */                          out_uint32_le(out, F_NAMELEN(stat_fs)); /* max length of filename */
779                          out_uint32_le(out, 2 * strlen(fs_type));        /* length of fs_type */  
780                          rdp_out_unistr(out, fs_type, 2 * strlen(fs_type) - 2);                          out_uint32_le(out, 2 * strlen(fsinfo->type));   /* length of fs_type */
781                            rdp_out_unistr(out, fsinfo->type, 2 * strlen(fsinfo->type) - 2);
782                          break;                          break;
783    
784                  case 2: /* FileFsLabelInformation */                  case FileFsLabelInformation:
785                  case 4: /* FileFsDeviceInformation */                  case FileFsDeviceInformation:
786                  case 6: /* FileFsControlInformation */                  case FileFsControlInformation:
787                  case 7: /* FileFsFullSizeInformation */                  case FileFsFullSizeInformation:
788                  case 8: /* FileFsObjectIdInformation */                  case FileFsObjectIdInformation:
789                  case 9: /* FileFsMaximumInformation */                  case FileFsMaximumInformation:
790    
791                  default:                  default:
792    
793                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);                          unimpl("IRP Query Volume Information class: 0x%x\n", info_class);

Legend:
Removed from v.601  
changed lines
  Added in v.659

  ViewVC Help
Powered by ViewVC 1.1.26