@@ -715,6 +715,16 @@ cifs_find_inode(struct inode *inode, void *opaque)
715
715
if (CIFS_I (inode )-> uniqueid != fattr -> cf_uniqueid )
716
716
return 0 ;
717
717
718
+ /*
719
+ * uh oh -- it's a directory. We can't use it since hardlinked dirs are
720
+ * verboten. Disable serverino and return it as if it were found, the
721
+ * caller can discard it, generate a uniqueid and retry the find
722
+ */
723
+ if (S_ISDIR (inode -> i_mode ) && !list_empty (& inode -> i_dentry )) {
724
+ fattr -> cf_flags |= CIFS_FATTR_INO_COLLISION ;
725
+ cifs_autodisable_serverino (CIFS_SB (inode -> i_sb ));
726
+ }
727
+
718
728
return 1 ;
719
729
}
720
730
@@ -734,15 +744,22 @@ cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
734
744
unsigned long hash ;
735
745
struct inode * inode ;
736
746
747
+ retry_iget5_locked :
737
748
cFYI (1 , ("looking for uniqueid=%llu" , fattr -> cf_uniqueid ));
738
749
739
750
/* hash down to 32-bits on 32-bit arch */
740
751
hash = cifs_uniqueid_to_ino_t (fattr -> cf_uniqueid );
741
752
742
753
inode = iget5_locked (sb , hash , cifs_find_inode , cifs_init_inode , fattr );
743
-
744
- /* we have fattrs in hand, update the inode */
745
754
if (inode ) {
755
+ /* was there a problematic inode number collision? */
756
+ if (fattr -> cf_flags & CIFS_FATTR_INO_COLLISION ) {
757
+ iput (inode );
758
+ fattr -> cf_uniqueid = iunique (sb , ROOT_I );
759
+ fattr -> cf_flags &= ~CIFS_FATTR_INO_COLLISION ;
760
+ goto retry_iget5_locked ;
761
+ }
762
+
746
763
cifs_fattr_to_inode (inode , fattr );
747
764
if (sb -> s_flags & MS_NOATIME )
748
765
inode -> i_flags |= S_NOATIME | S_NOCMTIME ;
0 commit comments