Skip to content

Commit 0235da0

Browse files
committed
Merge tag '6.13-rc-part2-smb3-client-fixes' of git://git.samba.org/sfrench/cifs-2.6
Pull smb client updates from Steve French: - directory lease fixes - password rotation fixes - reconnect fix - fix for SMB3.02 mounts - DFS (global namespace) fixes - fixes for special file handling (most relating to better handling various types of symlinks) - two minor cleanups * tag '6.13-rc-part2-smb3-client-fixes' of git://git.samba.org/sfrench/cifs-2.6: (22 commits) cifs: update internal version number cifs: unlock on error in smb3_reconfigure() cifs: during remount, make sure passwords are in sync cifs: support mounting with alternate password to allow password rotation smb: Initialize cfid->tcon before performing network ops smb: During unmount, ensure all cached dir instances drop their dentry smb: client: fix noisy message when mounting shares smb: client: don't try following DFS links in cifs_tree_connect() smb: client: allow reconnect when sending ioctl smb: client: get rid of @NLSC param in cifs_tree_connect() smb: client: allow more DFS referrals to be cached cifs: Fix parsing reparse point with native symlink in SMB1 non-UNICODE session cifs: Validate content of WSL reparse point buffers cifs: Improve guard for excluding $LXDEV xattr cifs: Add support for parsing WSL-style symlinks cifs: Validate content of native symlink cifs: Fix parsing native symlinks relative to the export smb: client: fix NULL ptr deref in crypto_aead_setkey() Update misleading comment in cifs_chan_update_iface smb: client: change return value in open_cached_dir_by_dentry() if !cfids ...
2 parents 109daa2 + 8d7690b commit 0235da0

24 files changed

+523
-297
lines changed

fs/smb/client/cached_dir.c

Lines changed: 134 additions & 36 deletions
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,11 @@ static void free_cached_dir(struct cached_fid *cfid);
1717
static void smb2_close_cached_fid(struct kref *ref);
1818
static void cfids_laundromat_worker(struct work_struct *work);
1919

20+
struct cached_dir_dentry {
21+
struct list_head entry;
22+
struct dentry *dentry;
23+
};
24+
2025
static struct cached_fid *find_or_create_cached_dir(struct cached_fids *cfids,
2126
const char *path,
2227
bool lookup_only,
@@ -157,15 +162,17 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
157162
const char *npath;
158163
int retries = 0, cur_sleep = 1;
159164

160-
if (tcon == NULL || tcon->cfids == NULL || tcon->nohandlecache ||
161-
is_smb1_server(tcon->ses->server) || (dir_cache_timeout == 0))
165+
if (cifs_sb->root == NULL)
166+
return -ENOENT;
167+
168+
if (tcon == NULL)
162169
return -EOPNOTSUPP;
163170

164171
ses = tcon->ses;
165172
cfids = tcon->cfids;
166173

167-
if (cifs_sb->root == NULL)
168-
return -ENOENT;
174+
if (cfids == NULL)
175+
return -EOPNOTSUPP;
169176

170177
replay_again:
171178
/* reinitialize for possible replay */
@@ -222,6 +229,7 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
222229
}
223230
}
224231
cfid->dentry = dentry;
232+
cfid->tcon = tcon;
225233

226234
/*
227235
* We do not hold the lock for the open because in case
@@ -293,7 +301,6 @@ int open_cached_dir(unsigned int xid, struct cifs_tcon *tcon,
293301
}
294302
goto oshr_free;
295303
}
296-
cfid->tcon = tcon;
297304
cfid->is_open = true;
298305

299306
spin_lock(&cfids->cfid_list_lock);
@@ -389,7 +396,7 @@ int open_cached_dir_by_dentry(struct cifs_tcon *tcon,
389396
struct cached_fids *cfids = tcon->cfids;
390397

391398
if (cfids == NULL)
392-
return -ENOENT;
399+
return -EOPNOTSUPP;
393400

394401
spin_lock(&cfids->cfid_list_lock);
395402
list_for_each_entry(cfid, &cfids->entries, entry) {
@@ -470,7 +477,10 @@ void close_all_cached_dirs(struct cifs_sb_info *cifs_sb)
470477
struct cifs_tcon *tcon;
471478
struct tcon_link *tlink;
472479
struct cached_fids *cfids;
480+
struct cached_dir_dentry *tmp_list, *q;
481+
LIST_HEAD(entry);
473482

483+
spin_lock(&cifs_sb->tlink_tree_lock);
474484
for (node = rb_first(root); node; node = rb_next(node)) {
475485
tlink = rb_entry(node, struct tcon_link, tl_rbnode);
476486
tcon = tlink_tcon(tlink);
@@ -479,11 +489,30 @@ void close_all_cached_dirs(struct cifs_sb_info *cifs_sb)
479489
cfids = tcon->cfids;
480490
if (cfids == NULL)
481491
continue;
492+
spin_lock(&cfids->cfid_list_lock);
482493
list_for_each_entry(cfid, &cfids->entries, entry) {
483-
dput(cfid->dentry);
494+
tmp_list = kmalloc(sizeof(*tmp_list), GFP_ATOMIC);
495+
if (tmp_list == NULL)
496+
break;
497+
spin_lock(&cfid->fid_lock);
498+
tmp_list->dentry = cfid->dentry;
484499
cfid->dentry = NULL;
500+
spin_unlock(&cfid->fid_lock);
501+
502+
list_add_tail(&tmp_list->entry, &entry);
485503
}
504+
spin_unlock(&cfids->cfid_list_lock);
486505
}
506+
spin_unlock(&cifs_sb->tlink_tree_lock);
507+
508+
list_for_each_entry_safe(tmp_list, q, &entry, entry) {
509+
list_del(&tmp_list->entry);
510+
dput(tmp_list->dentry);
511+
kfree(tmp_list);
512+
}
513+
514+
/* Flush any pending work that will drop dentries */
515+
flush_workqueue(cfid_put_wq);
487516
}
488517

489518
/*
@@ -494,14 +523,18 @@ void invalidate_all_cached_dirs(struct cifs_tcon *tcon)
494523
{
495524
struct cached_fids *cfids = tcon->cfids;
496525
struct cached_fid *cfid, *q;
497-
LIST_HEAD(entry);
498526

499527
if (cfids == NULL)
500528
return;
501529

530+
/*
531+
* Mark all the cfids as closed, and move them to the cfids->dying list.
532+
* They'll be cleaned up later by cfids_invalidation_worker. Take
533+
* a reference to each cfid during this process.
534+
*/
502535
spin_lock(&cfids->cfid_list_lock);
503536
list_for_each_entry_safe(cfid, q, &cfids->entries, entry) {
504-
list_move(&cfid->entry, &entry);
537+
list_move(&cfid->entry, &cfids->dying);
505538
cfids->num_entries--;
506539
cfid->is_open = false;
507540
cfid->on_list = false;
@@ -514,26 +547,47 @@ void invalidate_all_cached_dirs(struct cifs_tcon *tcon)
514547
} else
515548
kref_get(&cfid->refcount);
516549
}
550+
/*
551+
* Queue dropping of the dentries once locks have been dropped
552+
*/
553+
if (!list_empty(&cfids->dying))
554+
queue_work(cfid_put_wq, &cfids->invalidation_work);
517555
spin_unlock(&cfids->cfid_list_lock);
518-
519-
list_for_each_entry_safe(cfid, q, &entry, entry) {
520-
list_del(&cfid->entry);
521-
cancel_work_sync(&cfid->lease_break);
522-
/*
523-
* Drop the ref-count from above, either the lease-ref (if there
524-
* was one) or the extra one acquired.
525-
*/
526-
kref_put(&cfid->refcount, smb2_close_cached_fid);
527-
}
528556
}
529557

530558
static void
531-
smb2_cached_lease_break(struct work_struct *work)
559+
cached_dir_offload_close(struct work_struct *work)
532560
{
533561
struct cached_fid *cfid = container_of(work,
534-
struct cached_fid, lease_break);
562+
struct cached_fid, close_work);
563+
struct cifs_tcon *tcon = cfid->tcon;
564+
565+
WARN_ON(cfid->on_list);
535566

536567
kref_put(&cfid->refcount, smb2_close_cached_fid);
568+
cifs_put_tcon(tcon, netfs_trace_tcon_ref_put_cached_close);
569+
}
570+
571+
/*
572+
* Release the cached directory's dentry, and then queue work to drop cached
573+
* directory itself (closing on server if needed).
574+
*
575+
* Must be called with a reference to the cached_fid and a reference to the
576+
* tcon.
577+
*/
578+
static void cached_dir_put_work(struct work_struct *work)
579+
{
580+
struct cached_fid *cfid = container_of(work, struct cached_fid,
581+
put_work);
582+
struct dentry *dentry;
583+
584+
spin_lock(&cfid->fid_lock);
585+
dentry = cfid->dentry;
586+
cfid->dentry = NULL;
587+
spin_unlock(&cfid->fid_lock);
588+
589+
dput(dentry);
590+
queue_work(serverclose_wq, &cfid->close_work);
537591
}
538592

539593
int cached_dir_lease_break(struct cifs_tcon *tcon, __u8 lease_key[16])
@@ -560,8 +614,10 @@ int cached_dir_lease_break(struct cifs_tcon *tcon, __u8 lease_key[16])
560614
cfid->on_list = false;
561615
cfids->num_entries--;
562616

563-
queue_work(cifsiod_wq,
564-
&cfid->lease_break);
617+
++tcon->tc_count;
618+
trace_smb3_tcon_ref(tcon->debug_id, tcon->tc_count,
619+
netfs_trace_tcon_ref_get_cached_lease_break);
620+
queue_work(cfid_put_wq, &cfid->put_work);
565621
spin_unlock(&cfids->cfid_list_lock);
566622
return true;
567623
}
@@ -583,7 +639,8 @@ static struct cached_fid *init_cached_dir(const char *path)
583639
return NULL;
584640
}
585641

586-
INIT_WORK(&cfid->lease_break, smb2_cached_lease_break);
642+
INIT_WORK(&cfid->close_work, cached_dir_offload_close);
643+
INIT_WORK(&cfid->put_work, cached_dir_put_work);
587644
INIT_LIST_HEAD(&cfid->entry);
588645
INIT_LIST_HEAD(&cfid->dirents.entries);
589646
mutex_init(&cfid->dirents.de_mutex);
@@ -596,6 +653,9 @@ static void free_cached_dir(struct cached_fid *cfid)
596653
{
597654
struct cached_dirent *dirent, *q;
598655

656+
WARN_ON(work_pending(&cfid->close_work));
657+
WARN_ON(work_pending(&cfid->put_work));
658+
599659
dput(cfid->dentry);
600660
cfid->dentry = NULL;
601661

@@ -613,10 +673,30 @@ static void free_cached_dir(struct cached_fid *cfid)
613673
kfree(cfid);
614674
}
615675

676+
static void cfids_invalidation_worker(struct work_struct *work)
677+
{
678+
struct cached_fids *cfids = container_of(work, struct cached_fids,
679+
invalidation_work);
680+
struct cached_fid *cfid, *q;
681+
LIST_HEAD(entry);
682+
683+
spin_lock(&cfids->cfid_list_lock);
684+
/* move cfids->dying to the local list */
685+
list_cut_before(&entry, &cfids->dying, &cfids->dying);
686+
spin_unlock(&cfids->cfid_list_lock);
687+
688+
list_for_each_entry_safe(cfid, q, &entry, entry) {
689+
list_del(&cfid->entry);
690+
/* Drop the ref-count acquired in invalidate_all_cached_dirs */
691+
kref_put(&cfid->refcount, smb2_close_cached_fid);
692+
}
693+
}
694+
616695
static void cfids_laundromat_worker(struct work_struct *work)
617696
{
618697
struct cached_fids *cfids;
619698
struct cached_fid *cfid, *q;
699+
struct dentry *dentry;
620700
LIST_HEAD(entry);
621701

622702
cfids = container_of(work, struct cached_fids, laundromat_work.work);
@@ -642,18 +722,28 @@ static void cfids_laundromat_worker(struct work_struct *work)
642722

643723
list_for_each_entry_safe(cfid, q, &entry, entry) {
644724
list_del(&cfid->entry);
645-
/*
646-
* Cancel and wait for the work to finish in case we are racing
647-
* with it.
648-
*/
649-
cancel_work_sync(&cfid->lease_break);
650-
/*
651-
* Drop the ref-count from above, either the lease-ref (if there
652-
* was one) or the extra one acquired.
653-
*/
654-
kref_put(&cfid->refcount, smb2_close_cached_fid);
725+
726+
spin_lock(&cfid->fid_lock);
727+
dentry = cfid->dentry;
728+
cfid->dentry = NULL;
729+
spin_unlock(&cfid->fid_lock);
730+
731+
dput(dentry);
732+
if (cfid->is_open) {
733+
spin_lock(&cifs_tcp_ses_lock);
734+
++cfid->tcon->tc_count;
735+
trace_smb3_tcon_ref(cfid->tcon->debug_id, cfid->tcon->tc_count,
736+
netfs_trace_tcon_ref_get_cached_laundromat);
737+
spin_unlock(&cifs_tcp_ses_lock);
738+
queue_work(serverclose_wq, &cfid->close_work);
739+
} else
740+
/*
741+
* Drop the ref-count from above, either the lease-ref (if there
742+
* was one) or the extra one acquired.
743+
*/
744+
kref_put(&cfid->refcount, smb2_close_cached_fid);
655745
}
656-
queue_delayed_work(cifsiod_wq, &cfids->laundromat_work,
746+
queue_delayed_work(cfid_put_wq, &cfids->laundromat_work,
657747
dir_cache_timeout * HZ);
658748
}
659749

@@ -666,9 +756,11 @@ struct cached_fids *init_cached_dirs(void)
666756
return NULL;
667757
spin_lock_init(&cfids->cfid_list_lock);
668758
INIT_LIST_HEAD(&cfids->entries);
759+
INIT_LIST_HEAD(&cfids->dying);
669760

761+
INIT_WORK(&cfids->invalidation_work, cfids_invalidation_worker);
670762
INIT_DELAYED_WORK(&cfids->laundromat_work, cfids_laundromat_worker);
671-
queue_delayed_work(cifsiod_wq, &cfids->laundromat_work,
763+
queue_delayed_work(cfid_put_wq, &cfids->laundromat_work,
672764
dir_cache_timeout * HZ);
673765

674766
return cfids;
@@ -687,13 +779,19 @@ void free_cached_dirs(struct cached_fids *cfids)
687779
return;
688780

689781
cancel_delayed_work_sync(&cfids->laundromat_work);
782+
cancel_work_sync(&cfids->invalidation_work);
690783

691784
spin_lock(&cfids->cfid_list_lock);
692785
list_for_each_entry_safe(cfid, q, &cfids->entries, entry) {
693786
cfid->on_list = false;
694787
cfid->is_open = false;
695788
list_move(&cfid->entry, &entry);
696789
}
790+
list_for_each_entry_safe(cfid, q, &cfids->dying, entry) {
791+
cfid->on_list = false;
792+
cfid->is_open = false;
793+
list_move(&cfid->entry, &entry);
794+
}
697795
spin_unlock(&cfids->cfid_list_lock);
698796

699797
list_for_each_entry_safe(cfid, q, &entry, entry) {

fs/smb/client/cached_dir.h

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,8 @@ struct cached_fid {
4444
spinlock_t fid_lock;
4545
struct cifs_tcon *tcon;
4646
struct dentry *dentry;
47-
struct work_struct lease_break;
47+
struct work_struct put_work;
48+
struct work_struct close_work;
4849
struct smb2_file_all_info file_all_info;
4950
struct cached_dirents dirents;
5051
};
@@ -53,10 +54,13 @@ struct cached_fid {
5354
struct cached_fids {
5455
/* Must be held when:
5556
* - accessing the cfids->entries list
57+
* - accessing the cfids->dying list
5658
*/
5759
spinlock_t cfid_list_lock;
5860
int num_entries;
5961
struct list_head entries;
62+
struct list_head dying;
63+
struct work_struct invalidation_work;
6064
struct delayed_work laundromat_work;
6165
};
6266

fs/smb/client/cifsfs.c

Lines changed: 11 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,7 @@ struct workqueue_struct *fileinfo_put_wq;
157157
struct workqueue_struct *cifsoplockd_wq;
158158
struct workqueue_struct *deferredclose_wq;
159159
struct workqueue_struct *serverclose_wq;
160+
struct workqueue_struct *cfid_put_wq;
160161
__u32 cifs_lock_secret;
161162

162163
/*
@@ -1920,9 +1921,16 @@ init_cifs(void)
19201921
goto out_destroy_deferredclose_wq;
19211922
}
19221923

1924+
cfid_put_wq = alloc_workqueue("cfid_put_wq",
1925+
WQ_FREEZABLE|WQ_MEM_RECLAIM, 0);
1926+
if (!cfid_put_wq) {
1927+
rc = -ENOMEM;
1928+
goto out_destroy_serverclose_wq;
1929+
}
1930+
19231931
rc = cifs_init_inodecache();
19241932
if (rc)
1925-
goto out_destroy_serverclose_wq;
1933+
goto out_destroy_cfid_put_wq;
19261934

19271935
rc = cifs_init_netfs();
19281936
if (rc)
@@ -1990,6 +1998,8 @@ init_cifs(void)
19901998
cifs_destroy_netfs();
19911999
out_destroy_inodecache:
19922000
cifs_destroy_inodecache();
2001+
out_destroy_cfid_put_wq:
2002+
destroy_workqueue(cfid_put_wq);
19932003
out_destroy_serverclose_wq:
19942004
destroy_workqueue(serverclose_wq);
19952005
out_destroy_deferredclose_wq:

fs/smb/client/cifsfs.h

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,6 @@ extern const struct export_operations cifs_export_ops;
146146
#endif /* CONFIG_CIFS_NFSD_EXPORT */
147147

148148
/* when changing internal version - update following two lines at same time */
149-
#define SMB3_PRODUCT_BUILD 51
150-
#define CIFS_VERSION "2.51"
149+
#define SMB3_PRODUCT_BUILD 52
150+
#define CIFS_VERSION "2.52"
151151
#endif /* _CIFSFS_H */

0 commit comments

Comments
 (0)