ensure that we honor SMB2 read min_count properly
[kai/samba.git] / source4 / ntvfs / posix / pvfs_read.c
index e0c7e0b1f201279e609a76024cd2f87a221f1f50..a01a8a57e3b800595ebe434d9aed4a54ba76cab4 100644 (file)
@@ -7,7 +7,7 @@
 
    This program is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
-   the Free Software Foundation; either version 2 of the License, or
+   the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.
    
    This program is distributed in the hope that it will be useful,
    GNU General Public License for more details.
    
    You should have received a copy of the GNU General Public License
-   along with this program; if not, write to the Free Software
-   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+   along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
 
-#include "include/includes.h"
+#include "includes.h"
 #include "vfs_posix.h"
+#include "lib/events/events.h"
 
 /*
   read from a file
 */
 NTSTATUS pvfs_read(struct ntvfs_module_context *ntvfs,
-                  struct smbsrv_request *req, union smb_read *rd)
+                  struct ntvfs_request *req, union smb_read *rd)
 {
        struct pvfs_state *pvfs = ntvfs->private_data;
        ssize_t ret;
        struct pvfs_file *f;
+       NTSTATUS status;
+       uint32_t maxcnt;
+       uint32_t mask;
 
        if (rd->generic.level != RAW_READ_READX) {
-               return NT_STATUS_NOT_SUPPORTED;
+               return ntvfs_map_read(ntvfs, req, rd);
        }
 
-
-       f = pvfs_find_fd(pvfs, req, rd->readx.in.fnum);
+       f = pvfs_find_fd(pvfs, req, rd->readx.in.file.ntvfs);
        if (!f) {
                return NT_STATUS_INVALID_HANDLE;
        }
 
-       ret = pread(f->fd, 
-                   rd->readx.out.data, 
-                   rd->readx.in.maxcnt,
-                   rd->readx.in.offset);
+       if (f->handle->fd == -1) {
+               return NT_STATUS_FILE_IS_A_DIRECTORY;
+       }
+
+       mask = SEC_FILE_READ_DATA;
+       if (rd->readx.in.read_for_execute) {
+               mask |= SEC_FILE_EXECUTE;
+       }
+       if (!(f->access_mask & mask)) {
+               return NT_STATUS_ACCESS_DENIED;
+       }
+
+       maxcnt = rd->readx.in.maxcnt;
+       if (maxcnt > UINT16_MAX && req->ctx->protocol < PROTOCOL_SMB2) {
+               maxcnt = 0;
+       }
+
+       status = pvfs_check_lock(pvfs, f, req->smbpid, 
+                                rd->readx.in.offset,
+                                maxcnt,
+                                READ_LOCK);
+       if (!NT_STATUS_IS_OK(status)) {
+               return status;
+       }
+
+       if (f->handle->name->stream_name) {
+               ret = pvfs_stream_read(pvfs, f->handle, 
+                                      rd->readx.out.data, maxcnt, rd->readx.in.offset);
+       } else {
+#if HAVE_LINUX_AIO
+               /* possibly try an aio read */
+               if ((req->async_states->state & NTVFS_ASYNC_STATE_MAY_ASYNC) &&
+                   (pvfs->flags & PVFS_FLAG_LINUX_AIO)) {
+                       status = pvfs_aio_pread(req, rd, f, maxcnt);
+                       if (NT_STATUS_IS_OK(status)) {
+                               return NT_STATUS_OK;
+                       }
+               }
+#endif
+               ret = pread(f->handle->fd, 
+                           rd->readx.out.data, 
+                           maxcnt,
+                           rd->readx.in.offset);
+       }
        if (ret == -1) {
                return pvfs_map_errno(pvfs, errno);
        }
 
+       /* only SMB2 honors mincnt */
+       if (req->ctx->protocol == PROTOCOL_SMB2) {
+               if (rd->readx.in.mincnt > ret ||
+                   (ret == 0 && maxcnt > 0)) {
+                       return NT_STATUS_END_OF_FILE;
+               }
+       }
+
+       f->handle->position = f->handle->seek_offset = rd->readx.in.offset + ret;
+
        rd->readx.out.nread = ret;
-       rd->readx.out.remaining = 0; /* should fill this in? */
+       rd->readx.out.remaining = 0xFFFF;
        rd->readx.out.compaction_mode = 0; 
 
        return NT_STATUS_OK;