The branch, v4-14-test has been updated
       via  66285f55862 VERSION: Bump version up to Samba 4.14.13...
       via  038282f5236 Merge tag 'samba-4.14.12' into v4-14-test
       via  4c6b1950c47 VERSION: Disable GIT_SNAPSHOT for the 4.14.12 release.
       via  a500b68e3e6 WHATSNEW: Add release notes for Samba 4.14.12.
       via  092e9baee3b CVE-2021-44142: libadouble: harden parsing code
       via  3dab3dd8d57 CVE-2021-44142: libadouble: add basic cmocka tests
       via  5da11db4cd9 CVE-2021-44142: libadouble: harden ad_unpack_xattrs()
       via  4672487f3db CVE-2021-44142: smbd: add Netatalk xattr used by 
vfs_fruit to the list of private Samba xattrs
       via  8b7d5751eee CVE-2021-44142: libadouble: add defines for icon lengths
       via  8d0114ea973 CVE-2022-0336: s4/dsdb/samldb: Don't return early when 
an SPN is re-added to an object
       via  c4d576baaf0 CVE-2022-0336: pytest: Add a test for an SPN conflict 
with a re-added SPN
      from  96f88613a5e blackbox.ndrdump: fix 
test_ndrdump_fuzzed_NULL_struct_ntlmssp_CHALLENGE_MESSAGE test

https://git.samba.org/?p=samba.git;a=shortlog;h=v4-14-test


- Log -----------------------------------------------------------------
commit 66285f55862c7c57ab1809c973589063ac7b1c84
Author: Jule Anger <jan...@samba.org>
Date:   Mon Jan 31 14:40:13 2022 +0100

    VERSION: Bump version up to Samba 4.14.13...
    
    and re-enable GIT_SNAPSHOT.
    
    Signed-off-by: Jule Anger <jan...@samba.org>
    Signed-off-by: Stefan Metzmacher <me...@samba.org>

commit 038282f523616427923eb4ba7df800bda57fab98
Merge: 96f88613a5e 4c6b1950c47
Author: Jule Anger <jan...@samba.org>
Date:   Mon Jan 31 14:38:48 2022 +0100

    Merge tag 'samba-4.14.12' into v4-14-test
    
    samba: tag release samba-4.14.12

-----------------------------------------------------------------------

Summary of changes:
 VERSION                                 |   2 +-
 WHATSNEW.txt                            |  50 +++-
 python/samba/tests/ldap_spn.py          |   7 +
 selftest/tests.py                       |   2 +
 source3/lib/adouble.c                   | 136 +++++++++--
 source3/lib/adouble.h                   |   2 +
 source3/lib/test_adouble.c              | 389 ++++++++++++++++++++++++++++++++
 source3/smbd/trans2.c                   |   2 +
 source3/wscript_build                   |   5 +
 source4/dsdb/samdb/ldb_modules/samldb.c |   3 +-
 10 files changed, 576 insertions(+), 22 deletions(-)
 create mode 100644 source3/lib/test_adouble.c


Changeset truncated at 500 lines:

diff --git a/VERSION b/VERSION
index 4b80d608672..21204af08a4 100644
--- a/VERSION
+++ b/VERSION
@@ -25,7 +25,7 @@
 ########################################################
 SAMBA_VERSION_MAJOR=4
 SAMBA_VERSION_MINOR=14
-SAMBA_VERSION_RELEASE=12
+SAMBA_VERSION_RELEASE=13
 
 ########################################################
 # If a official release has a serious bug              #
diff --git a/WHATSNEW.txt b/WHATSNEW.txt
index ea20a3ea952..2f8e8c31500 100644
--- a/WHATSNEW.txt
+++ b/WHATSNEW.txt
@@ -1,3 +1,50 @@
+                   ===============================
+                   Release Notes for Samba 4.14.12
+                          January 31, 2022
+                   ===============================
+
+
+This is a security release in order to address the following defects:
+
+o CVE-2021-44142: Out-of-Bound Read/Write on Samba vfs_fruit module.
+                  https://www.samba.org/samba/security/CVE-2021-44142.html
+
+o CVE-2022-0336:  Re-adding an SPN skips subsequent SPN conflict checks.
+                  https://www.samba.org/samba/security/CVE-2022-0336.html
+
+
+Changes since 4.14.11
+---------------------
+
+o  Ralph Boehme <s...@samba.org>
+   * BUG 14914: CVE-2021-44142.
+
+o  Joseph Sutton <josephsut...@catalyst.net.nz>
+   * BUG 14950: CVE-2022-0336.
+
+
+#######################################
+Reporting bugs & Development Discussion
+#######################################
+
+Please discuss this release on the samba-technical mailing list or by
+joining the #samba-technical IRC channel on irc.freenode.net.
+
+If you do report problems then please try to send high quality
+feedback. If you don't provide vital information to help us track down
+the problem then you will probably be ignored.  All bug reports should
+be filed under the Samba 4.1 and newer product in the project's Bugzilla
+database (https://bugzilla.samba.org/).
+
+
+======================================================================
+== Our Code, Our Bugs, Our Responsibility.
+== The Samba Team
+======================================================================
+
+
+Release notes for older releases follow:
+----------------------------------------
                    ===============================
                    Release Notes for Samba 4.14.11
                           December 15, 2021
@@ -83,8 +130,7 @@ database (https://bugzilla.samba.org/).
 ======================================================================
 
 
-Release notes for older releases follow:
-----------------------------------------
+----------------------------------------------------------------------
                    ===============================
                    Release Notes for Samba 4.14.10
                            November 9, 2021
diff --git a/python/samba/tests/ldap_spn.py b/python/samba/tests/ldap_spn.py
index 8a398ffaa49..6ebdf8f9a32 100644
--- a/python/samba/tests/ldap_spn.py
+++ b/python/samba/tests/ldap_spn.py
@@ -268,6 +268,8 @@ class LdapSpnTestBase(TestCase):
             for k in ('dNSHostName', 'servicePrincipalName'):
                 if isinstance(m.get(k), str):
                     m[k] = m[k].format(dnsname=f"x.{REALM}")
+                elif isinstance(m.get(k), list):
+                    m[k] = [x.format(dnsname=f"x.{REALM}") for x in m[k]]
 
             msg = ldb.Message.from_dict(samdb, m, op)
 
@@ -727,6 +729,11 @@ class LdapSpnSambaOnlyTest(LdapSpnTestBase):
          ('user:C', 'host/{dnsname}', '*', ok),
          ('user:D', 'www/{dnsname}', 'D', denied),
         ),
+        ("add a conflict, along with a re-added SPN",
+         ('A', 'cifs/{dnsname}', '*', ok),
+         ('B', 'cifs/heeble.example.net', 'B', ok),
+         ('B', ['cifs/heeble.example.net', 'host/{dnsname}'], 'B', constraint),
+        ),
 
         ("changing dNSHostName after host",
          ('A', {'dNSHostName': '{dnsname}'}, '*', ok),
diff --git a/selftest/tests.py b/selftest/tests.py
index af1e4606185..f4a1056f1dc 100644
--- a/selftest/tests.py
+++ b/selftest/tests.py
@@ -427,3 +427,5 @@ if with_elasticsearch_backend:
                   [os.path.join(bindir(), 
"default/source3/test_mdsparser_es")] + [configuration])
 plantestsuite("samba.unittests.credentials", "none",
               [os.path.join(bindir(), "default/auth/credentials/test_creds")])
+plantestsuite("samba.unittests.adouble", "none",
+              [os.path.join(bindir(), "test_adouble")])
diff --git a/source3/lib/adouble.c b/source3/lib/adouble.c
index 0ab9019cfb5..48cc0007c23 100644
--- a/source3/lib/adouble.c
+++ b/source3/lib/adouble.c
@@ -269,6 +269,95 @@ size_t ad_setentryoff(struct adouble *ad, int eid, size_t 
off)
        return ad->ad_eid[eid].ade_off = off;
 }
 
+/*
+ * All entries besides FinderInfo and resource fork must fit into the
+ * buffer. FinderInfo is special as it may be larger then the default 32 bytes
+ * if it contains marshalled xattrs, which we will fixup that in
+ * ad_convert(). The first 32 bytes however must also be part of the buffer.
+ *
+ * The resource fork is never accessed directly by the ad_data buf.
+ */
+static bool ad_entry_check_size(uint32_t eid,
+                               size_t bufsize,
+                               uint32_t off,
+                               uint32_t got_len)
+{
+       struct {
+               off_t expected_len;
+               bool fixed_size;
+               bool minimum_size;
+       } ad_checks[] = {
+               [ADEID_DFORK] = {-1, false, false}, /* not applicable */
+               [ADEID_RFORK] = {-1, false, false}, /* no limit */
+               [ADEID_NAME] = {ADEDLEN_NAME, false, false},
+               [ADEID_COMMENT] = {ADEDLEN_COMMENT, false, false},
+               [ADEID_ICONBW] = {ADEDLEN_ICONBW, true, false},
+               [ADEID_ICONCOL] = {ADEDLEN_ICONCOL, false, false},
+               [ADEID_FILEI] = {ADEDLEN_FILEI, true, false},
+               [ADEID_FILEDATESI] = {ADEDLEN_FILEDATESI, true, false},
+               [ADEID_FINDERI] = {ADEDLEN_FINDERI, false, true},
+               [ADEID_MACFILEI] = {ADEDLEN_MACFILEI, true, false},
+               [ADEID_PRODOSFILEI] = {ADEDLEN_PRODOSFILEI, true, false},
+               [ADEID_MSDOSFILEI] = {ADEDLEN_MSDOSFILEI, true, false},
+               [ADEID_SHORTNAME] = {ADEDLEN_SHORTNAME, false, false},
+               [ADEID_AFPFILEI] = {ADEDLEN_AFPFILEI, true, false},
+               [ADEID_DID] = {ADEDLEN_DID, true, false},
+               [ADEID_PRIVDEV] = {ADEDLEN_PRIVDEV, true, false},
+               [ADEID_PRIVINO] = {ADEDLEN_PRIVINO, true, false},
+               [ADEID_PRIVSYN] = {ADEDLEN_PRIVSYN, true, false},
+               [ADEID_PRIVID] = {ADEDLEN_PRIVID, true, false},
+       };
+
+       if (eid >= ADEID_MAX) {
+               return false;
+       }
+       if (got_len == 0) {
+               /* Entry present, but empty, allow */
+               return true;
+       }
+       if (ad_checks[eid].expected_len == 0) {
+               /*
+                * Shouldn't happen: implicitly initialized to zero because
+                * explicit initializer missing.
+                */
+               return false;
+       }
+       if (ad_checks[eid].expected_len == -1) {
+               /* Unused or no limit */
+               return true;
+       }
+       if (ad_checks[eid].fixed_size) {
+               if (ad_checks[eid].expected_len != got_len) {
+                       /* Wrong size fo fixed size entry. */
+                       return false;
+               }
+       } else {
+               if (ad_checks[eid].minimum_size) {
+                       if (got_len < ad_checks[eid].expected_len) {
+                               /*
+                                * Too small for variable sized entry with
+                                * minimum size.
+                                */
+                               return false;
+                       }
+               } else {
+                       if (got_len > ad_checks[eid].expected_len) {
+                               /* Too big for variable sized entry. */
+                               return false;
+                       }
+               }
+       }
+       if (off + got_len < off) {
+               /* wrap around */
+               return false;
+       }
+       if (off + got_len > bufsize) {
+               /* overflow */
+               return false;
+       }
+       return true;
+}
+
 /**
  * Return a pointer to an AppleDouble entry
  *
@@ -276,8 +365,15 @@ size_t ad_setentryoff(struct adouble *ad, int eid, size_t 
off)
  **/
 char *ad_get_entry(const struct adouble *ad, int eid)
 {
+       size_t bufsize = talloc_get_size(ad->ad_data);
        off_t off = ad_getentryoff(ad, eid);
        size_t len = ad_getentrylen(ad, eid);
+       bool valid;
+
+       valid = ad_entry_check_size(eid, bufsize, off, len);
+       if (!valid) {
+               return NULL;
+       }
 
        if (off == 0 || len == 0) {
                return NULL;
@@ -707,14 +803,27 @@ static bool ad_pack(struct vfs_handle_struct *handle,
 static bool ad_unpack_xattrs(struct adouble *ad)
 {
        struct ad_xattr_header *h = &ad->adx_header;
+       size_t bufsize = talloc_get_size(ad->ad_data);
        const char *p = ad->ad_data;
        uint32_t hoff;
        uint32_t i;
 
+       if (ad->ad_type != ADOUBLE_RSRC) {
+               return false;
+       }
+
        if (ad_getentrylen(ad, ADEID_FINDERI) <= ADEDLEN_FINDERI) {
                return true;
        }
 
+       /*
+        * Ensure the buffer ad->ad_data was allocated by ad_alloc() for an
+        * ADOUBLE_RSRC type (._ AppleDouble file on-disk).
+        */
+       if (bufsize != AD_XATTR_MAX_HDR_SIZE) {
+               return false;
+       }
+
        /* 2 bytes padding */
        hoff = ad_getentryoff(ad, ADEID_FINDERI) + ADEDLEN_FINDERI + 2;
 
@@ -901,20 +1010,11 @@ static bool ad_unpack(struct adouble *ad, const size_t 
nentries,
                        return false;
                }
 
-               /*
-                * All entries besides FinderInfo and resource fork
-                * must fit into the buffer. FinderInfo is special as
-                * it may be larger then the default 32 bytes (if it
-                * contains marshalled xattrs), but we will fixup that
-                * in ad_convert(). And the resource fork is never
-                * accessed directly by the ad_data buf (also see
-                * comment above) anyway.
-                */
-               if ((eid != ADEID_RFORK) &&
-                   (eid != ADEID_FINDERI) &&
-                   ((off + len) > bufsize)) {
-                       DEBUG(1, ("bogus eid %d: off: %" PRIu32 ", len: %" 
PRIu32 "\n",
-                                 eid, off, len));
+               ok = ad_entry_check_size(eid, bufsize, off, len);
+               if (!ok) {
+                       DBG_ERR("bogus eid [%"PRIu32"] bufsize [%zu] "
+                               "off [%"PRIu32"] len [%"PRIu32"]\n",
+                               eid, bufsize, off, len);
                        return false;
                }
 
@@ -964,9 +1064,11 @@ static bool ad_unpack(struct adouble *ad, const size_t 
nentries,
                ad->ad_eid[eid].ade_len = len;
        }
 
-       ok = ad_unpack_xattrs(ad);
-       if (!ok) {
-               return false;
+       if (ad->ad_type == ADOUBLE_RSRC) {
+               ok = ad_unpack_xattrs(ad);
+               if (!ok) {
+                       return false;
+               }
        }
 
        return true;
diff --git a/source3/lib/adouble.h b/source3/lib/adouble.h
index 90a825c502e..e3b9263a1f9 100644
--- a/source3/lib/adouble.h
+++ b/source3/lib/adouble.h
@@ -101,6 +101,8 @@ typedef enum {ADOUBLE_META, ADOUBLE_RSRC} adouble_type_t;
 #define ADEDLEN_MACFILEI        4
 #define ADEDLEN_PRODOSFILEI     8
 #define ADEDLEN_MSDOSFILEI      2
+#define ADEDLEN_ICONBW          128
+#define ADEDLEN_ICONCOL         1024
 #define ADEDLEN_DID             4
 #define ADEDLEN_PRIVDEV         8
 #define ADEDLEN_PRIVINO         8
diff --git a/source3/lib/test_adouble.c b/source3/lib/test_adouble.c
new file mode 100644
index 00000000000..615c22469c9
--- /dev/null
+++ b/source3/lib/test_adouble.c
@@ -0,0 +1,389 @@
+/*
+ * Unix SMB/CIFS implementation.
+ *
+ * Copyright (C) 2021 Ralph Boehme <s...@samba.org>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "adouble.c"
+#include <cmocka.h>
+
+static int setup_talloc_context(void **state)
+{
+       TALLOC_CTX *frame = talloc_stackframe();
+
+       *state = frame;
+       return 0;
+}
+
+static int teardown_talloc_context(void **state)
+{
+       TALLOC_CTX *frame = *state;
+
+       TALLOC_FREE(frame);
+       return 0;
+}
+
+/*
+ * Basic and sane buffer.
+ */
+static uint8_t ad_basic[] = {
+       0x00, 0x05, 0x16, 0x07, /* Magic */
+       0x00, 0x02, 0x00, 0x00, /* Version */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x02,             /* Count */
+       /* adentry 1: FinderInfo */
+       0x00, 0x00, 0x00, 0x09, /* eid: FinderInfo */
+       0x00, 0x00, 0x00, 0x32, /* offset */
+       0x00, 0x00, 0x00, 0x20, /* length */
+       /* adentry 2: Resourcefork */
+       0x00, 0x00, 0x00, 0x02, /* eid: Resourcefork */
+       0x00, 0x00, 0x00, 0x52, /* offset */
+       0xff, 0xff, 0xff, 0x00, /* length */
+       /* FinderInfo data: 32 bytes */
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+};
+
+/*
+ * An empty FinderInfo entry.
+ */
+static uint8_t ad_finderinfo1[] = {
+       0x00, 0x05, 0x16, 0x07, /* Magic */
+       0x00, 0x02, 0x00, 0x00, /* Version */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x02,             /* Count */
+       /* adentry 1: FinderInfo */
+       0x00, 0x00, 0x00, 0x09, /* eid: FinderInfo */
+       0x00, 0x00, 0x00, 0x52, /* off: points at end of buffer */
+       0x00, 0x00, 0x00, 0x00, /* len: 0, so off+len don't exceed bufferlen */
+       /* adentry 2: Resourcefork */
+       0x00, 0x00, 0x00, 0x02, /* eid: Resourcefork */
+       0x00, 0x00, 0x00, 0x52, /* offset */
+       0xff, 0xff, 0xff, 0x00, /* length */
+       /* FinderInfo data: 32 bytes */
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+};
+
+/*
+ * A dangerous FinderInfo with correct length exceeding buffer by one byte.
+ */
+static uint8_t ad_finderinfo2[] = {
+       0x00, 0x05, 0x16, 0x07, /* Magic */
+       0x00, 0x02, 0x00, 0x00, /* Version */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x02,             /* Count */
+       /* adentry 1: FinderInfo */
+       0x00, 0x00, 0x00, 0x09, /* eid: FinderInfo */
+       0x00, 0x00, 0x00, 0x33, /* off: points at beginng of data + 1 */
+       0x00, 0x00, 0x00, 0x20, /* len: 32, so off+len exceeds bufferlen by 1 */
+       /* adentry 2: Resourcefork */
+       0x00, 0x00, 0x00, 0x02, /* eid: Resourcefork */
+       0x00, 0x00, 0x00, 0x52, /* offset */
+       0xff, 0xff, 0xff, 0x00, /* length */
+       /* FinderInfo data: 32 bytes */
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+};
+
+static uint8_t ad_finderinfo3[] = {
+       0x00, 0x05, 0x16, 0x07, /* Magic */
+       0x00, 0x02, 0x00, 0x00, /* Version */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x02,             /* Count */
+       /* adentry 1: FinderInfo */
+       0x00, 0x00, 0x00, 0x09, /* eid: FinderInfo */
+       0x00, 0x00, 0x00, 0x33, /* off: points at beginng of data + 1 */
+       0x00, 0x00, 0x00, 0x1f, /* len: 31, so off+len don't exceed buf */
+       /* adentry 2: Resourcefork */
+       0x00, 0x00, 0x00, 0x02, /* eid: Resourcefork */
+       0x00, 0x00, 0x00, 0x52, /* offset */
+       0xff, 0xff, 0xff, 0x00, /* length */
+       /* FinderInfo data: 32 bytes */
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+};
+
+/*
+ * A dangerous name entry.
+ */
+static uint8_t ad_name[] = {
+       0x00, 0x05, 0x16, 0x07, /* Magic */
+       0x00, 0x02, 0x00, 0x00, /* Version */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x00, 0x00, 0x00, /* Filler */
+       0x00, 0x02,             /* Count */
+       /* adentry 1: FinderInfo */
+       0x00, 0x00, 0x00, 0x09, /* eid: FinderInfo */
+       0x00, 0x00, 0x00, 0x32, /* offset */
+       0x00, 0x00, 0x00, 0x20, /* length */
+       /* adentry 2: Name */
+       0x00, 0x00, 0x00, 0x03, /* eid: Name */
+       0x00, 0x00, 0x00, 0x52, /* off: points at end of buffer */
+       0x00, 0x00, 0x00, 0x01, /* len: 1, so off+len exceeds bufferlen */
+       /* FinderInfo data: 32 bytes */
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,
+       0x00, 0x00, 0x00, 0x00,


-- 
Samba Shared Repository

Reply via email to