15331Samw /* 25331Samw * CDDL HEADER START 35331Samw * 45331Samw * The contents of this file are subject to the terms of the 55331Samw * Common Development and Distribution License (the "License"). 65331Samw * You may not use this file except in compliance with the License. 75331Samw * 85331Samw * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 95331Samw * or http://www.opensolaris.org/os/licensing. 105331Samw * See the License for the specific language governing permissions 115331Samw * and limitations under the License. 125331Samw * 135331Samw * When distributing Covered Code, include this CDDL HEADER in each 145331Samw * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 155331Samw * If applicable, add the following below this CDDL HEADER, with the 165331Samw * fields enclosed by brackets "[]" replaced with your own identifying 175331Samw * information: Portions Copyright [yyyy] [name of copyright owner] 185331Samw * 195331Samw * CDDL HEADER END 205331Samw */ 215331Samw /* 22*8934SJose.Borrego@Sun.COM * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 235331Samw * Use is subject to license terms. 245331Samw */ 255331Samw 265331Samw /* 275331Samw * The flush SMB is sent to ensure all data and allocation information 285331Samw * for the corresponding file has been written to stable storage. This 295331Samw * is a synchronous request. The response should not be sent until the 305331Samw * writes are complete. 315331Samw * 325331Samw * The SmbFlush request is described in CIFS/1.0 1996 Section 3.9.14. 335331Samw * 345331Samw * CIFS/1.0 June 13, 1996 355331Samw * Heizer, et al 365331Samw * draft-heizer-cifs-v1-spec-00.txt 375331Samw */ 385331Samw 395331Samw #include <smbsrv/smb_incl.h> 405331Samw #include <smbsrv/smb_fsops.h> 415331Samw 425331Samw 435331Samw static void smb_flush_file(struct smb_request *sr, struct smb_ofile *ofile); 445331Samw 455331Samw /* 465331Samw * smb_com_flush 475331Samw * 485331Samw * Flush any cached data for a specified file, or for all files that 495331Samw * this client has open, to stable storage. If the fid is valid (i.e. 505331Samw * not 0xFFFF), we flush only that file. Otherwise we flush all files 515331Samw * associated with this client. 525331Samw * 535331Samw * We need to protect the list because there's a good chance we'll 545331Samw * block during the flush operation. 555331Samw */ 566030Sjb150015 smb_sdrc_t 576139Sjb150015 smb_pre_flush(smb_request_t *sr) 586139Sjb150015 { 596139Sjb150015 int rc; 606139Sjb150015 616139Sjb150015 rc = smbsr_decode_vwv(sr, "w", &sr->smb_fid); 626139Sjb150015 636139Sjb150015 DTRACE_SMB_1(op__Flush__start, smb_request_t *, sr); 646139Sjb150015 656139Sjb150015 return ((rc == 0) ? SDRC_SUCCESS : SDRC_ERROR); 666139Sjb150015 } 676139Sjb150015 686139Sjb150015 void 696139Sjb150015 smb_post_flush(smb_request_t *sr) 706139Sjb150015 { 716139Sjb150015 DTRACE_SMB_1(op__Flush__done, smb_request_t *, sr); 726139Sjb150015 } 736139Sjb150015 746139Sjb150015 smb_sdrc_t 755331Samw smb_com_flush(smb_request_t *sr) 765331Samw { 775331Samw smb_ofile_t *file; 785331Samw smb_llist_t *flist; 796030Sjb150015 int rc; 805331Samw 815331Samw if (smb_flush_required == 0) { 826030Sjb150015 rc = smbsr_encode_empty_result(sr); 836139Sjb150015 return ((rc == 0) ? SDRC_SUCCESS : SDRC_ERROR); 845331Samw } 855331Samw 865331Samw if (sr->smb_fid != 0xffff) { 87*8934SJose.Borrego@Sun.COM smbsr_lookup_file(sr); 885331Samw if (sr->fid_ofile == NULL) { 895772Sas200622 smbsr_error(sr, NT_STATUS_INVALID_HANDLE, 905331Samw ERRDOS, ERRbadfid); 916139Sjb150015 return (SDRC_ERROR); 925331Samw } 935331Samw 945331Samw smb_flush_file(sr, sr->fid_ofile); 955331Samw } else { 965331Samw flist = &sr->tid_tree->t_ofile_list; 975331Samw smb_llist_enter(flist, RW_READER); 985331Samw file = smb_llist_head(flist); 995331Samw while (file) { 1005331Samw mutex_enter(&file->f_mutex); 1015331Samw smb_flush_file(sr, file); 1025331Samw mutex_exit(&file->f_mutex); 1035331Samw file = smb_llist_next(flist, file); 1045331Samw } 1055331Samw smb_llist_exit(flist); 1065331Samw } 1076030Sjb150015 1086030Sjb150015 rc = smbsr_encode_empty_result(sr); 1096139Sjb150015 return ((rc == 0) ? SDRC_SUCCESS : SDRC_ERROR); 1105331Samw } 1115331Samw 1125331Samw 1135331Samw /* 1145331Samw * smb_flush_file 1155331Samw * 1165331Samw * If writes on this file are not synchronous, flush it using the NFSv3 1175331Samw * commit interface. 1185331Samw */ 1196030Sjb150015 static void 1206030Sjb150015 smb_flush_file(struct smb_request *sr, struct smb_ofile *ofile) 1215331Samw { 1227961SNatalie.Li@Sun.COM sr->user_cr = smb_ofile_getcred(ofile); 1237961SNatalie.Li@Sun.COM 1245331Samw if ((ofile->f_node->flags & NODE_FLAGS_WRITE_THROUGH) == 0) 1255331Samw (void) smb_fsop_commit(sr, sr->user_cr, ofile->f_node); 1265331Samw } 127