@@ -59,6 +59,16 @@ static struct cached_fid *find_or_create_cached_dir(struct cached_fids *cfids,
59
59
list_add (& cfid -> entry , & cfids -> entries );
60
60
cfid -> on_list = true;
61
61
kref_get (& cfid -> refcount );
62
+ /*
63
+ * Set @cfid->has_lease to true during construction so that the lease
64
+ * reference can be put in cached_dir_lease_break() due to a potential
65
+ * lease break right after the request is sent or while @cfid is still
66
+ * being cached, or if a reconnection is triggered during construction.
67
+ * Concurrent processes won't be to use it yet due to @cfid->time being
68
+ * zero.
69
+ */
70
+ cfid -> has_lease = true;
71
+
62
72
spin_unlock (& cfids -> cfid_list_lock );
63
73
return cfid ;
64
74
}
@@ -176,12 +186,12 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
176
186
return - ENOENT ;
177
187
}
178
188
/*
179
- * Return cached fid if it has a lease. Otherwise, it is either a new
180
- * entry or laundromat worker removed it from @cfids->entries. Caller
181
- * will put last reference if the latter.
189
+ * Return cached fid if it is valid ( has a lease and has a time).
190
+ * Otherwise, it is either a new entry or laundromat worker removed it
191
+ * from @cfids->entries. Caller will put last reference if the latter.
182
192
*/
183
193
spin_lock (& cfids -> cfid_list_lock );
184
- if (cfid -> has_lease ) {
194
+ if (cfid -> has_lease && cfid -> time ) {
185
195
spin_unlock (& cfids -> cfid_list_lock );
186
196
* ret_cfid = cfid ;
187
197
kfree (utf16_path );
@@ -267,15 +277,6 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
267
277
268
278
smb2_set_related (& rqst [1 ]);
269
279
270
- /*
271
- * Set @cfid->has_lease to true before sending out compounded request so
272
- * its lease reference can be put in cached_dir_lease_break() due to a
273
- * potential lease break right after the request is sent or while @cfid
274
- * is still being cached. Concurrent processes won't be to use it yet
275
- * due to @cfid->time being zero.
276
- */
277
- cfid -> has_lease = true;
278
-
279
280
if (retries ) {
280
281
smb2_set_replay (server , & rqst [0 ]);
281
282
smb2_set_replay (server , & rqst [1 ]);
@@ -347,6 +348,7 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
347
348
SMB2_query_info_free (& rqst [1 ]);
348
349
free_rsp_buf (resp_buftype [0 ], rsp_iov [0 ].iov_base );
349
350
free_rsp_buf (resp_buftype [1 ], rsp_iov [1 ].iov_base );
351
+ out :
350
352
if (rc ) {
351
353
spin_lock (& cfids -> cfid_list_lock );
352
354
if (cfid -> on_list ) {
@@ -358,23 +360,14 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
358
360
/*
359
361
* We are guaranteed to have two references at this
360
362
* point. One for the caller and one for a potential
361
- * lease. Release the Lease-ref so that the directory
362
- * will be closed when the caller closes the cached
363
- * handle.
363
+ * lease. Release one here, and the second below.
364
364
*/
365
365
cfid -> has_lease = false;
366
- spin_unlock (& cfids -> cfid_list_lock );
367
366
kref_put (& cfid -> refcount , smb2_close_cached_fid );
368
- goto out ;
369
367
}
370
368
spin_unlock (& cfids -> cfid_list_lock );
371
- }
372
- out :
373
- if (rc ) {
374
- if (cfid -> is_open )
375
- SMB2_close (0 , cfid -> tcon , cfid -> fid .persistent_fid ,
376
- cfid -> fid .volatile_fid );
377
- free_cached_dir (cfid );
369
+
370
+ kref_put (& cfid -> refcount , smb2_close_cached_fid );
378
371
} else {
379
372
* ret_cfid = cfid ;
380
373
atomic_inc (& tcon -> num_remote_opens );
@@ -401,7 +394,7 @@ int open_cached_dir_by_dentry(struct cifs_tcon *tcon,
401
394
spin_lock (& cfids -> cfid_list_lock );
402
395
list_for_each_entry (cfid , & cfids -> entries , entry ) {
403
396
if (dentry && cfid -> dentry == dentry ) {
404
- cifs_dbg (FYI , "found a cached root file handle by dentry\n" );
397
+ cifs_dbg (FYI , "found a cached file handle by dentry\n" );
405
398
kref_get (& cfid -> refcount );
406
399
* ret_cfid = cfid ;
407
400
spin_unlock (& cfids -> cfid_list_lock );
@@ -512,25 +505,24 @@ void invalidate_all_cached_dirs(struct cifs_tcon *tcon)
512
505
cfids -> num_entries -- ;
513
506
cfid -> is_open = false;
514
507
cfid -> on_list = false;
515
- /* To prevent race with smb2_cached_lease_break() */
516
- kref_get (& cfid -> refcount );
508
+ if (cfid -> has_lease ) {
509
+ /*
510
+ * The lease was never cancelled from the server,
511
+ * so steal that reference.
512
+ */
513
+ cfid -> has_lease = false;
514
+ } else
515
+ kref_get (& cfid -> refcount );
517
516
}
518
517
spin_unlock (& cfids -> cfid_list_lock );
519
518
520
519
list_for_each_entry_safe (cfid , q , & entry , entry ) {
521
520
list_del (& cfid -> entry );
522
521
cancel_work_sync (& cfid -> lease_break );
523
- if (cfid -> has_lease ) {
524
- /*
525
- * We lease was never cancelled from the server so we
526
- * need to drop the reference.
527
- */
528
- spin_lock (& cfids -> cfid_list_lock );
529
- cfid -> has_lease = false;
530
- spin_unlock (& cfids -> cfid_list_lock );
531
- kref_put (& cfid -> refcount , smb2_close_cached_fid );
532
- }
533
- /* Drop the extra reference opened above*/
522
+ /*
523
+ * Drop the ref-count from above, either the lease-ref (if there
524
+ * was one) or the extra one acquired.
525
+ */
534
526
kref_put (& cfid -> refcount , smb2_close_cached_fid );
535
527
}
536
528
}
@@ -541,9 +533,6 @@ smb2_cached_lease_break(struct work_struct *work)
541
533
struct cached_fid * cfid = container_of (work ,
542
534
struct cached_fid , lease_break );
543
535
544
- spin_lock (& cfid -> cfids -> cfid_list_lock );
545
- cfid -> has_lease = false;
546
- spin_unlock (& cfid -> cfids -> cfid_list_lock );
547
536
kref_put (& cfid -> refcount , smb2_close_cached_fid );
548
537
}
549
538
@@ -561,6 +550,7 @@ int cached_dir_lease_break(struct cifs_tcon *tcon, __u8 lease_key[16])
561
550
!memcmp (lease_key ,
562
551
cfid -> fid .lease_key ,
563
552
SMB2_LEASE_KEY_SIZE )) {
553
+ cfid -> has_lease = false;
564
554
cfid -> time = 0 ;
565
555
/*
566
556
* We found a lease remove it from the list
@@ -638,8 +628,14 @@ static void cfids_laundromat_worker(struct work_struct *work)
638
628
cfid -> on_list = false;
639
629
list_move (& cfid -> entry , & entry );
640
630
cfids -> num_entries -- ;
641
- /* To prevent race with smb2_cached_lease_break() */
642
- kref_get (& cfid -> refcount );
631
+ if (cfid -> has_lease ) {
632
+ /*
633
+ * Our lease has not yet been cancelled from the
634
+ * server. Steal that reference.
635
+ */
636
+ cfid -> has_lease = false;
637
+ } else
638
+ kref_get (& cfid -> refcount );
643
639
}
644
640
}
645
641
spin_unlock (& cfids -> cfid_list_lock );
@@ -651,17 +647,10 @@ static void cfids_laundromat_worker(struct work_struct *work)
651
647
* with it.
652
648
*/
653
649
cancel_work_sync (& cfid -> lease_break );
654
- if (cfid -> has_lease ) {
655
- /*
656
- * Our lease has not yet been cancelled from the server
657
- * so we need to drop the reference.
658
- */
659
- spin_lock (& cfids -> cfid_list_lock );
660
- cfid -> has_lease = false;
661
- spin_unlock (& cfids -> cfid_list_lock );
662
- kref_put (& cfid -> refcount , smb2_close_cached_fid );
663
- }
664
- /* Drop the extra reference opened above */
650
+ /*
651
+ * Drop the ref-count from above, either the lease-ref (if there
652
+ * was one) or the extra one acquired.
653
+ */
665
654
kref_put (& cfid -> refcount , smb2_close_cached_fid );
666
655
}
667
656
queue_delayed_work (cifsiod_wq , & cfids -> laundromat_work ,
0 commit comments