1 /*++
2
3 Copyright (c) 1989-2000 Microsoft Corporation
4
5 Module Name:
6
7 FilObSup.c
8
9 Abstract:
10
11 This module implements the Fat File object support routines.
12
13
14 --*/
15
16 #include "fatprocs.h"
17
18 //
19 // The Bug check file id for this module
20 //
21
22 #define BugCheckFileId (FAT_BUG_CHECK_FILOBSUP)
23
24 //
25 // The debug trace level
26 //
27
28 #define Dbg (DEBUG_TRACE_FILOBSUP)
29
30 #ifdef ALLOC_PRAGMA
31 #pragma alloc_text(PAGE, FatForceCacheMiss)
32 #pragma alloc_text(PAGE, FatPurgeReferencedFileObjects)
33 #pragma alloc_text(PAGE, FatSetFileObject)
34 #pragma alloc_text(PAGE, FatDecodeFileObject)
35 #endif
36
37
38 VOID
FatSetFileObject(IN PFILE_OBJECT FileObject OPTIONAL,IN TYPE_OF_OPEN TypeOfOpen,IN PVOID VcbOrFcbOrDcb,IN PCCB Ccb OPTIONAL)39 FatSetFileObject (
40 IN PFILE_OBJECT FileObject OPTIONAL,
41 IN TYPE_OF_OPEN TypeOfOpen,
42 IN PVOID VcbOrFcbOrDcb,
43 IN PCCB Ccb OPTIONAL
44 )
45
46 /*++
47
48 Routine Description:
49
50 This routine sets the file system pointers within the file object
51
52 Arguments:
53
54 FileObject - Supplies a pointer to the file object being modified, and
55 can optionally be null.
56
57 TypeOfOpen - Supplies the type of open denoted by the file object.
58 This is only used by this procedure for sanity checking.
59
60 VcbOrFcbOrDcb - Supplies a pointer to either a vcb, fcb, or dcb
61
62 Ccb - Optionally supplies a pointer to a ccb
63
64 Return Value:
65
66 None.
67
68 --*/
69
70 {
71 PAGED_CODE();
72
73 DebugTrace(+1, Dbg, "FatSetFileObject, FileObject = %p\n", FileObject );
74
75 NT_ASSERT((Ccb == NULL) || (NodeType(Ccb) == FAT_NTC_CCB));
76
77
78 NT_ASSERT(((TypeOfOpen == UnopenedFileObject))
79
80 ||
81
82 ((TypeOfOpen == UserFileOpen) &&
83 (NodeType(VcbOrFcbOrDcb) == FAT_NTC_FCB) &&
84 (Ccb != NULL))
85
86 ||
87
88 ((TypeOfOpen == EaFile) &&
89 (NodeType(VcbOrFcbOrDcb) == FAT_NTC_FCB) &&
90 (Ccb == NULL))
91
92 ||
93
94 ((TypeOfOpen == UserDirectoryOpen) &&
95 ((NodeType(VcbOrFcbOrDcb) == FAT_NTC_DCB) || (NodeType(VcbOrFcbOrDcb) == FAT_NTC_ROOT_DCB)) &&
96 (Ccb != NULL))
97
98 ||
99
100 ((TypeOfOpen == UserVolumeOpen) &&
101 (NodeType(VcbOrFcbOrDcb) == FAT_NTC_VCB) &&
102 (Ccb != NULL))
103
104 ||
105
106 ((TypeOfOpen == VirtualVolumeFile) &&
107 (NodeType(VcbOrFcbOrDcb) == FAT_NTC_VCB) &&
108 (Ccb == NULL))
109
110 ||
111
112 ((TypeOfOpen == DirectoryFile) &&
113 ((NodeType(VcbOrFcbOrDcb) == FAT_NTC_DCB) || (NodeType(VcbOrFcbOrDcb) == FAT_NTC_ROOT_DCB)) &&
114 (Ccb == NULL))
115 );
116
117
118 UNREFERENCED_PARAMETER( TypeOfOpen );
119
120 //
121 // If we were given an Fcb, Dcb, or Vcb, we have some processing to do.
122 //
123
124 NT_ASSERT((Ccb == NULL) || (NodeType(Ccb) == FAT_NTC_CCB));
125
126 if ( VcbOrFcbOrDcb != NULL ) {
127
128 //
129 // Set the Vpb field in the file object, and if we were given an
130 // Fcb or Dcb move the field over to point to the nonpaged Fcb/Dcb
131 //
132
133 if (NodeType(VcbOrFcbOrDcb) == FAT_NTC_VCB) {
134
135 FileObject->Vpb = ((PVCB)VcbOrFcbOrDcb)->Vpb;
136
137 } else {
138
139 FileObject->Vpb = ((PFCB)VcbOrFcbOrDcb)->Vcb->Vpb;
140
141 //
142 // If this is a temporary file, note it in the FcbState
143 //
144
145 if (FlagOn(((PFCB)VcbOrFcbOrDcb)->FcbState, FCB_STATE_TEMPORARY)) {
146
147 SetFlag(FileObject->Flags, FO_TEMPORARY_FILE);
148 }
149 }
150 }
151
152 NT_ASSERT((Ccb == NULL) || (NodeType(Ccb) == FAT_NTC_CCB));
153
154 //
155 // Now set the fscontext fields of the file object
156 //
157
158 if (ARGUMENT_PRESENT( FileObject )) {
159
160 FileObject->FsContext = VcbOrFcbOrDcb;
161 FileObject->FsContext2 = Ccb;
162 }
163
164 NT_ASSERT((Ccb == NULL) || (NodeType(Ccb) == FAT_NTC_CCB));
165
166 //
167 // And return to our caller
168 //
169
170 DebugTrace(-1, Dbg, "FatSetFileObject -> VOID\n", 0);
171
172 return;
173 }
174
175 TYPE_OF_OPEN
FatDecodeFileObject(_In_ PFILE_OBJECT FileObject,_Outptr_ PVCB * Vcb,_Outptr_ PFCB * FcbOrDcb,_Outptr_ PCCB * Ccb)176 FatDecodeFileObject (
177 _In_ PFILE_OBJECT FileObject,
178 _Outptr_ PVCB *Vcb,
179 _Outptr_ PFCB *FcbOrDcb,
180 _Outptr_ PCCB *Ccb
181 )
182
183 /*++
184
185 Routine Description:
186
187 This procedure takes a pointer to a file object, that has already been
188 opened by the Fat file system and figures out what really is opened.
189
190 Arguments:
191
192 FileObject - Supplies the file object pointer being interrogated
193
194 Vcb - Receives a pointer to the Vcb for the file object.
195
196 FcbOrDcb - Receives a pointer to the Fcb/Dcb for the file object, if
197 one exists.
198
199 Ccb - Receives a pointer to the Ccb for the file object, if one exists.
200
201 Return Value:
202
203 TYPE_OF_OPEN - returns the type of file denoted by the input file object.
204
205 UserFileOpen - The FO represents a user's opened data file.
206 Ccb, FcbOrDcb, and Vcb are set. FcbOrDcb points to an Fcb.
207
208 UserDirectoryOpen - The FO represents a user's opened directory.
209 Ccb, FcbOrDcb, and Vcb are set. FcbOrDcb points to a Dcb/RootDcb
210
211 UserVolumeOpen - The FO represents a user's opened volume.
212 Ccb and Vcb are set. FcbOrDcb is null.
213
214 VirtualVolumeFile - The FO represents the special virtual volume file.
215 Vcb is set, and Ccb and FcbOrDcb are null.
216
217 DirectoryFile - The FO represents a special directory file.
218 Vcb and FcbOrDcb are set. Ccb is null. FcbOrDcb points to a
219 Dcb/RootDcb.
220
221 EaFile - The FO represents an Ea Io stream file.
222 FcbOrDcb, and Vcb are set. FcbOrDcb points to an Fcb, and Ccb is
223 null.
224
225 --*/
226
227 {
228 TYPE_OF_OPEN TypeOfOpen;
229 PVOID FsContext;
230 PVOID FsContext2;
231
232 PAGED_CODE();
233
234 DebugTrace(+1, Dbg, "FatDecodeFileObject, FileObject = %p\n", FileObject);
235
236 //
237 // Reference the fs context fields of the file object, and zero out
238 // the out pointer parameters.
239 //
240
241 FsContext = FileObject->FsContext;
242 FsContext2 = FileObject->FsContext2;
243
244 //
245 // Special case the situation where FsContext is null
246 //
247
248 if (FsContext == NULL) {
249
250 *Ccb = NULL;
251 *FcbOrDcb = NULL;
252 *Vcb = NULL;
253
254 TypeOfOpen = UnopenedFileObject;
255
256 } else {
257
258 //
259 // Now we can case on the node type code of the fscontext pointer
260 // and set the appropriate out pointers
261 //
262
263 switch (NodeType(FsContext)) {
264
265 case FAT_NTC_VCB:
266
267 *Ccb = FsContext2;
268 *FcbOrDcb = NULL;
269 *Vcb = FsContext;
270
271 TypeOfOpen = ( *Ccb == NULL ? VirtualVolumeFile : UserVolumeOpen );
272
273 break;
274
275 case FAT_NTC_ROOT_DCB:
276 case FAT_NTC_DCB:
277
278 *Ccb = FsContext2;
279 *FcbOrDcb = FsContext;
280 *Vcb = (*FcbOrDcb)->Vcb;
281
282 TypeOfOpen = ( *Ccb == NULL ? DirectoryFile : UserDirectoryOpen );
283
284 DebugTrace(0, Dbg, "Referencing directory: %wZ\n", &(*FcbOrDcb)->FullFileName);
285
286 break;
287
288 case FAT_NTC_FCB:
289
290 *Ccb = FsContext2;
291 *FcbOrDcb = FsContext;
292 *Vcb = (*FcbOrDcb)->Vcb;
293
294 if (*Ccb != NULL ) {
295
296 TypeOfOpen = UserFileOpen;
297 DebugTrace(0, Dbg, "Referencing file: %wZ\n", &(*FcbOrDcb)->FullFileName);
298
299 } else {
300
301 //
302 // No Ccb means this is a special open.
303 //
304
305
306 if ( *FcbOrDcb == (*Vcb)->EaFcb ) {
307
308 TypeOfOpen = EaFile;
309 DebugTrace(0, Dbg, "Referencing EA file: %wZ\n", &(*FcbOrDcb)->FullFileName);
310
311 } else {
312
313 #ifdef _MSC_VER
314 #pragma prefast(suppress:28159, "things are seriously wrong if we get here")
315 #endif
316 FatBugCheck( NodeType(FsContext), 0, 0 );
317
318 }
319
320 }
321
322 break;
323
324 default:
325
326 #ifdef _MSC_VER
327 #pragma prefast( suppress:28159, "things are seriously wrong if we get here" )
328 #endif
329 FatBugCheck( NodeType(FsContext), 0, 0 );
330 }
331 }
332
333 //
334 // and return to our caller
335 //
336
337 DebugTrace(0, Dbg, "FatDecodeFileObject -> VCB(%p)\n", *Vcb);
338 DebugTrace(0, Dbg, "FatDecodeFileObject -> FCB(%p)\n", *FcbOrDcb);
339 DebugTrace(0, Dbg, "FatDecodeFileObject -> CCB(%p)\n", *Ccb);
340 DebugTrace(-1, Dbg, "FatDecodeFileObject -> TypeOfOpen = %08lx\n", TypeOfOpen);
341
342 return TypeOfOpen;
343 }
344
_Requires_lock_held_(_Global_critical_region_)345 _Requires_lock_held_(_Global_critical_region_)
346 VOID
347 FatPurgeReferencedFileObjects (
348 IN PIRP_CONTEXT IrpContext,
349 IN PFCB Fcb,
350 IN FAT_FLUSH_TYPE FlushType
351 )
352
353 /*++
354
355 Routine Description:
356
357 This routine non-recursively walks from the given FcbOrDcb and trys
358 to force Cc or Mm to close any sections it may be holding on to.
359
360 Arguments:
361
362 Fcb - Supplies a pointer to either an fcb or a dcb
363
364 FlushType - Specifies the kind of flushing to perform
365
366 Return Value:
367
368 None.
369
370 --*/
371
372 {
373 PFCB OriginalFcb = Fcb;
374 PFCB NextFcb;
375
376 PAGED_CODE();
377
378 DebugTrace(+1, Dbg, "FatPurgeReferencedFileObjects, Fcb = %p\n", Fcb );
379
380 NT_ASSERT( FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) );
381
382 //
383 // First, if we have a delayed close, force it closed.
384 //
385
386 FatFspClose(Fcb->Vcb);
387
388 //
389 // Walk the directory tree forcing sections closed.
390 //
391 // Note that it very important to get the next node to visit before
392 // acting on the current node. This is because acting on a node may
393 // make it, and an arbitrary number of direct ancestors, vanish.
394 // Since we never visit ancestors in our top-down enumeration scheme, we
395 // can safely continue the enumeration even when the tree is vanishing
396 // beneath us. This is way cool.
397 //
398
399 while ( Fcb != NULL ) {
400
401 NextFcb = FatGetNextFcbTopDown(IrpContext, Fcb, OriginalFcb);
402
403 //
404 // Check for the EA file fcb
405 //
406
407 if ( !FlagOn(Fcb->DirentFatFlags, FAT_DIRENT_ATTR_VOLUME_ID) ) {
408
409 FatForceCacheMiss( IrpContext, Fcb, FlushType );
410 }
411
412 Fcb = NextFcb;
413 }
414
415 DebugTrace(-1, Dbg, "FatPurgeReferencedFileObjects (VOID)\n", 0 );
416
417 return;
418 }
419
420
_Requires_lock_held_(_Global_critical_region_)421 _Requires_lock_held_(_Global_critical_region_)
422 VOID
423 FatForceCacheMiss (
424 IN PIRP_CONTEXT IrpContext,
425 IN PFCB Fcb,
426 IN FAT_FLUSH_TYPE FlushType
427 )
428
429 /*++
430
431 Routine Description:
432
433 The following routine asks either Cc or Mm to get rid of any cached
434 pages on a file. Note that this will fail if a user has mapped a file.
435
436 If there is a shared cache map, purge the cache section. Otherwise
437 we have to go and ask Mm to blow away the section.
438
439 NOTE: This caller MUST own the Vcb exclusive.
440
441 Arguments:
442
443 Fcb - Supplies a pointer to an fcb
444
445 FlushType - Specifies the kind of flushing to perform
446
447 Return Value:
448
449 None.
450
451 --*/
452
453 {
454 PVCB Vcb;
455 BOOLEAN ChildrenAcquired = FALSE;
456
457 PAGED_CODE();
458
459 //
460 // If we can't wait, bail.
461 //
462
463 NT_ASSERT( FatVcbAcquiredExclusive( IrpContext, Fcb->Vcb ) ||
464 FlagOn( Fcb->Vcb->VcbState, VCB_STATE_FLAG_LOCKED ) );
465
466 if (!FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT)) {
467
468 FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
469 }
470
471 //
472 // If we are purging a directory file object, we must acquire all the
473 // FCBs exclusive so that the parent directory is not being pinned.
474 // Careful, we can collide with something acquiring up the tree like
475 // an unpin repinned flush (FsRtlAcquireFileForCcFlush ...) of a parent
476 // dir on extending writethrough of a child file (oops). So get things
477 // going up the tree, not down.
478 //
479
480 if ((NodeType(Fcb) != FAT_NTC_FCB) &&
481 !IsListEmpty(&Fcb->Specific.Dcb.ParentDcbQueue)) {
482
483 PLIST_ENTRY Links;
484 PFCB TempFcb;
485
486 ChildrenAcquired = TRUE;
487
488 for (Links = Fcb->Specific.Dcb.ParentDcbQueue.Flink;
489 Links != &Fcb->Specific.Dcb.ParentDcbQueue;
490 Links = Links->Flink) {
491
492 TempFcb = CONTAINING_RECORD( Links, FCB, ParentDcbLinks );
493
494 (VOID)FatAcquireExclusiveFcb( IrpContext, TempFcb );
495 }
496 }
497
498 (VOID)FatAcquireExclusiveFcb( IrpContext, Fcb );
499
500 //
501 // We use this flag to indicate to a close beneath us that
502 // the Fcb resource should be freed before deleting the Fcb.
503 //
504
505 Vcb = Fcb->Vcb;
506
507 SetFlag( Fcb->FcbState, FCB_STATE_FORCE_MISS_IN_PROGRESS );
508
509 ClearFlag( Vcb->VcbState, VCB_STATE_FLAG_DELETED_FCB );
510
511 _SEH2_TRY {
512
513 BOOLEAN DataSectionExists;
514 BOOLEAN ImageSectionExists;
515
516 PSECTION_OBJECT_POINTERS Section;
517
518 if ( FlushType ) {
519
520 (VOID)FatFlushFile( IrpContext, Fcb, FlushType );
521 }
522
523 //
524 // The Flush may have made the Fcb go away
525 //
526
527 if (!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_DELETED_FCB)) {
528
529 Section = &Fcb->NonPaged->SectionObjectPointers;
530
531 DataSectionExists = (BOOLEAN)(Section->DataSectionObject != NULL);
532 ImageSectionExists = (BOOLEAN)(Section->ImageSectionObject != NULL);
533
534 //
535 // Note, it is critical to do the Image section first as the
536 // purge of the data section may cause the image section to go
537 // away, but the opposite is not true.
538 //
539
540 if (ImageSectionExists) {
541
542 (VOID)MmFlushImageSection( Section, MmFlushForWrite );
543 }
544
545 if (DataSectionExists) {
546
547 CcPurgeCacheSection( Section, NULL, 0, FALSE );
548 }
549 }
550
551 } _SEH2_FINALLY {
552
553 //
554 // If we purging a directory file object, release all the Fcb
555 // resources that we acquired above. The Dcb cannot have vanished
556 // if there were Fcbs underneath it, and the Fcbs couldn't have gone
557 // away since I own the Vcb.
558 //
559
560 if (ChildrenAcquired) {
561
562 PLIST_ENTRY Links;
563 PFCB TempFcb;
564
565 for (Links = Fcb->Specific.Dcb.ParentDcbQueue.Flink;
566 Links != &Fcb->Specific.Dcb.ParentDcbQueue;
567 Links = Links->Flink) {
568
569 TempFcb = CONTAINING_RECORD( Links, FCB, ParentDcbLinks );
570
571 FatReleaseFcb( IrpContext, TempFcb );
572 }
573 }
574
575 //
576 // Since we have the Vcb exclusive we know that if any closes
577 // come in it is because the CcPurgeCacheSection caused the
578 // Fcb to go away. Also in close, the Fcb was released
579 // before being freed.
580 //
581
582 if ( !FlagOn(Vcb->VcbState, VCB_STATE_FLAG_DELETED_FCB) ) {
583
584 ClearFlag( Fcb->FcbState, FCB_STATE_FORCE_MISS_IN_PROGRESS );
585
586 FatReleaseFcb( (IRPCONTEXT), Fcb );
587 }
588 } _SEH2_END;
589 }
590
591
592