⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 visopsys_io.c

📁 上一个上传的有问题,这个是好的。visopsys包括系统内核和GUI的全部SOURCE code ,还包括一些基本的docs文档。里面src子目录对应所有SOURCE code.对于想研究操作系统的朋
💻 C
📖 第 1 页 / 共 2 页
字号:
// //  Visopsys//  Copyright (C) 1998-2007 J. Andrew McLaughlin//  //  This library is free software; you can redistribute it and/or modify it//  under the terms of the GNU Lesser General Public License as published by//  the Free Software Foundation; either version 2.1 of the License, or (at//  your option) any later version.////  This library is distributed in the hope that it will be useful, but//  WITHOUT ANY WARRANTY; without even the implied warranty of//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser//  General Public License for more details.////  You should have received a copy of the GNU Lesser General Public License//  along with this library; if not, write to the Free Software Foundation,//  Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.////  visopsys_io.c////  A stdio-like disk I/O implementation for low-level disk access on//  Visopsys.  Can access an NTFS volume while it is mounted.  Heavily//  modified from win32_io.c in the Linux-NTFS project.////  Copyrights in original file://    Copyright (c) 2003-2004 Lode Leroy//    Copyright (c) 2003-2005 Anton Altaparmakov#include <errno.h>#include <fcntl.h>#include <stdlib.h>#include <sys/api.h>#include "device.h"typedef struct {  int isDisk;  file file;  disk disk;  s64 partLength;  s64 position;    // Logical current position in the file/disk} visopsys_fd;#define Vdebug(f, a...) do {                      \  ntfs_log_debug("VISOPSYS: %s: ", __FUNCTION__); \  ntfs_log_debug(f, ##a);                         \} while(0)static int ntfs_visopsys_errno(unsigned error){  // Convert a Visopsys error to a UNIX one  switch (error) {  case ERR_NOSUCHFUNCTION:    return EBADRQC;  case ERR_NOSUCHENTRY:  case ERR_NOSUCHFILE:  case ERR_NOSUCHDIR:    return ENOENT;  case ERR_PERMISSION:    return EACCES;  case ERR_MEMORY:    return ENOMEM;  case ERR_NOFREE:    return ENOSPC;  case ERR_NOMEDIA:    return ENODEV;  case ERR_NOWRITE:    return EROFS;  case ERR_BUSY:    return EBUSY;  case ERR_INVALID:  case ERR_NULLPARAMETER:    return EINVAL;  case ERR_NOSUCHDRIVER:  case ERR_NOTIMPLEMENTED:    return EOPNOTSUPP;  default:    // generic message    return ERR_ERROR;  }}static int ntfs_device_visopsys_open(struct ntfs_device *dev, int flags){  // Open a device.  dev->d_name must hold the device name, the rest is  // ignored.  Supported flags are O_RDONLY, O_WRONLY and O_RDWR.  // If the name is not a Visopsys disk name, treat it as a file.  int status = 0;  file f;  visopsys_fd *fd = NULL;  Vdebug("OPEN\n");  if (NDevOpen(dev))    {      errno = ntfs_visopsys_errno(ERR_BUSY);      return (status = -1);    }  // File name?  if (fileFind(dev->d_name, &f) >= 0)    {      ntfs_log_trace("Can't open regular files\n");      errno = ntfs_visopsys_errno(ERR_NOTIMPLEMENTED);      return (status = -1);    }  fd = malloc(sizeof(visopsys_fd));  if (fd == NULL)    {      ntfs_log_trace("Memory allocation failure\n");      errno = ntfs_visopsys_errno(ERR_MEMORY);      return (status = -1);    }  // Disk name.  No UNIX-style opening required.  // Try to get disk information  status = diskGet(dev->d_name, &(fd->disk));  if (status < 0)    {      ntfs_log_trace("Can't get disk information\n");      goto error_out;    }  // Make sure it's a logical disk (a partition, that is) rather than a  // physical one  if ((fd->disk.type & DISKTYPE_LOGICALPHYSICAL) != DISKTYPE_LOGICAL)    {      ntfs_log_trace("Can't open physical disks\n");      status = ERR_NOTIMPLEMENTED;      goto error_out;    }  // Make sure the sector size is set  if (fd->disk.sectorSize == 0)    {      ntfs_log_trace("Disk sector size is NULL\n");      status = ERR_BUG;      goto error_out;    }  fd->partLength = ((s64) fd->disk.numSectors * (s64) fd->disk.sectorSize);  dev->d_private = fd;  NDevSetBlock(dev);  NDevSetOpen(dev);  NDevClearDirty(dev);  // Setup our read-only flag.  if ((flags & O_RDWR) != O_RDWR)    NDevSetReadOnly(dev);  return (status = 0); error_out:  free(fd);  errno = ntfs_visopsys_errno(status);  return (status = -1);}static int ntfs_device_visopsys_close(struct ntfs_device *dev){  // Close an open ntfs deivce  // dev:	ntfs device obtained via ->open  // Return 0 if o.k.  //	 -1 if not, and errno set.  Note if error fd->vol_handle is trashed.  int status = 0;  visopsys_fd *fd = NULL;  Vdebug("CLOSE\n");  // Check params  if (dev == NULL)    {      ntfs_log_trace("NULL device parameter\n");      errno = ntfs_visopsys_errno(ERR_NULLPARAMETER);      return (status = -1);    }  if (!NDevOpen(dev))    {      ntfs_log_trace("Can't open device\n");      errno = ntfs_visopsys_errno(ERR_INVALID);      return (status = -1);    }  fd = (visopsys_fd *) dev->d_private;  if (NDevDirty(dev))    diskSync(fd->disk.name);  NDevClearOpen(dev);  dev->d_private = NULL;  free(fd);  return (status = 0);}static s64 ntfs_device_visopsys_seek(struct ntfs_device *dev, s64 offset,				     int whence){  // Change current logical file position  // dev:	ntfs device obtained via ->open  // offset:	required offset from the whence anchor  // whence:	whence anchor specifying what @offset is relative to  // Return the new position on the volume on success and -1 on error with  // errno set to the error code.  // whence may be one of the following:  //	SEEK_SET - Offset is relative to file start.  //	SEEK_CUR - Offset is relative to current position.  //	SEEK_END - Offset is relative to end of file.  s64 abs_ofs = 0;  visopsys_fd *fd = NULL;  //Vdebug("SEEK\n");  // Check params  if (dev == NULL)    {      ntfs_log_trace("NULL device parameter\n");      errno = ntfs_visopsys_errno(ERR_NULLPARAMETER);      return (abs_ofs = -1);    }  fd = (visopsys_fd *) dev->d_private;  switch (whence)    {    case SEEK_SET:      abs_ofs = offset;      break;    case SEEK_CUR:      abs_ofs = (fd->position + offset);      break;    case SEEK_END:      abs_ofs = (fd->partLength + offset);      break;    default:      ntfs_log_trace("Invalid 'whence' seek argument %d\n", whence);      errno = ntfs_visopsys_errno(ERR_INVALID);      return (abs_ofs = -1);    }  // abs_ofs should be a multiple of the block size  if (abs_ofs % (s64) fd->disk.sectorSize)    {      ntfs_log_trace("Seek address is not a multiple of sector size\n");      errno = ntfs_visopsys_errno(ERR_INVALID);      return (abs_ofs = -1);    }  if ((abs_ofs < 0) || (abs_ofs > fd->partLength))    {      ntfs_log_trace("Seek outside partition (sector %llu)",	     (abs_ofs / (s64) fd->disk.sectorSize));      if (abs_ofs < 0)	ntfs_log_trace("(abs_ofs (%llu) < 0)\n", abs_ofs);      else	ntfs_log_trace("(abs_ofs (%llu) > fd->partLength (%llu))\n", abs_ofs,		       fd->partLength);      errno = EINVAL;      return (abs_ofs = -1);    }  fd->position = abs_ofs;  return (abs_ofs);}static s64 ntfs_device_visopsys_read(struct ntfs_device *dev, void *buff,				     s64 count){  // Read bytes from an ntfs device  // dev:	ntfs device obtained via ->open  // buff:	pointer to where to put the contents  // count:	how many bytes should be read  // On success returns the number of bytes actually read (can be < count).  // On error returns -1 with errno set.  int status = 0;  visopsys_fd *fd = NULL;  s64 startSector = 0;  s64 sectorCount = 0;  void *saveBuff = NULL;  s64 br = 0;  //Vdebug("READ\n");  // Check params  if ((dev == NULL) || (buff == NULL) || (count <= 0))    {      ntfs_log_trace("NULL parameter\n");      errno = ntfs_visopsys_errno(ERR_NULLPARAMETER);      return (status = -1);    }  fd = (visopsys_fd *) dev->d_private;  startSector = (fd->position / (s64) fd->disk.sectorSize);  sectorCount = (count / (s64) fd->disk.sectorSize);  if ((fd->position % (s64) fd->disk.sectorSize) ||      (count % (s64) fd->disk.sectorSize))    {      Vdebug("Doing off-kilter read");

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -