--- /dev/null
+APPLE PUBLIC SOURCE LICENSE
+Version 2.0 - August 6, 2003
+
+Please read this License carefully before downloading this software. By
+downloading or using this software, you are agreeing to be bound by the terms
+of this License. If you do not or cannot agree to the terms of this License,
+please do not download or use the software.
+
+Apple Note: In January 2007, Apple changed its corporate name from "Apple
+Computer, Inc." to "Apple Inc." This change has been reflected below and
+copyright years updated, but no other changes have been made to the APSL 2.0.
+
+1. General; Definitions. This License applies to any program or other
+work which Apple Inc. ("Apple") makes publicly available and which contains a
+notice placed by Apple identifying such program or work as "Original Code" and
+stating that it is subject to the terms of this Apple Public Source License
+version 2.0 ("License"). As used in this License:
+
+1.1 "Applicable Patent Rights" mean: (a) in the case where Apple is the
+grantor of rights, (i) claims of patents that are now or hereafter acquired,
+owned by or assigned to Apple and (ii) that cover subject matter contained in
+the Original Code, but only to the extent necessary to use, reproduce and/or
+distribute the Original Code without infringement; and (b) in the case where
+You are the grantor of rights, (i) claims of patents that are now or hereafter
+acquired, owned by or assigned to You and (ii) that cover subject matter in
+Your Modifications, taken alone or in combination with Original Code.
+
+1.2 "Contributor" means any person or entity that creates or contributes to
+the creation of Modifications.
+
+1.3 "Covered Code" means the Original Code, Modifications, the combination
+of Original Code and any Modifications, and/or any respective portions thereof.
+
+1.4 "Externally Deploy" means: (a) to sublicense, distribute or otherwise
+make Covered Code available, directly or indirectly, to anyone other than You;
+and/or (b) to use Covered Code, alone or as part of a Larger Work, in any way
+to provide a service, including but not limited to delivery of content, through
+electronic communication with a client other than You.
+
+1.5 "Larger Work" means a work which combines Covered Code or portions
+thereof with code not governed by the terms of this License.
+
+1.6 "Modifications" mean any addition to, deletion from, and/or change to,
+the substance and/or structure of the Original Code, any previous
+Modifications, the combination of Original Code and any previous Modifications,
+and/or any respective portions thereof. When code is released as a series of
+files, a Modification is: (a) any addition to or deletion from the contents of
+a file containing Covered Code; and/or (b) any new file or other representation
+of computer program statements that contains any part of Covered Code.
+
+1.7 "Original Code" means (a) the Source Code of a program or other work as
+originally made available by Apple under this License, including the Source
+Code of any updates or upgrades to such programs or works made available by
+Apple under this License, and that has been expressly identified by Apple as
+such in the header file(s) of such work; and (b) the object code compiled from
+such Source Code and originally made available by Apple under this License
+
+1.8 "Source Code" means the human readable form of a program or other work
+that is suitable for making modifications to it, including all modules it
+contains, plus any associated interface definition files, scripts used to
+control compilation and installation of an executable (object code).
+
+1.9 "You" or "Your" means an individual or a legal entity exercising rights
+under this License. For legal entities, "You" or "Your" includes any entity
+which controls, is controlled by, or is under common control with, You, where
+"control" means (a) the power, direct or indirect, to cause the direction or
+management of such entity, whether by contract or otherwise, or (b) ownership
+of fifty percent (50%) or more of the outstanding shares or beneficial
+ownership of such entity.
+
+2. Permitted Uses; Conditions & Restrictions. Subject to the terms and
+conditions of this License, Apple hereby grants You, effective on the date You
+accept this License and download the Original Code, a world-wide, royalty-free,
+non-exclusive license, to the extent of Apple's Applicable Patent Rights and
+copyrights covering the Original Code, to do the following:
+
+2.1 Unmodified Code. You may use, reproduce, display, perform, internally
+distribute within Your organization, and Externally Deploy verbatim, unmodified
+copies of the Original Code, for commercial or non-commercial purposes,
+provided that in each instance:
+
+(a) You must retain and reproduce in all copies of Original Code the
+copyright and other proprietary notices and disclaimers of Apple as they appear
+in the Original Code, and keep intact all notices in the Original Code that
+refer to this License; and
+
+(b) You must include a copy of this License with every copy of Source Code
+of Covered Code and documentation You distribute or Externally Deploy, and You
+may not offer or impose any terms on such Source Code that alter or restrict
+this License or the recipients' rights hereunder, except as permitted under
+Section 6.
+
+2.2 Modified Code. You may modify Covered Code and use, reproduce,
+display, perform, internally distribute within Your organization, and
+Externally Deploy Your Modifications and Covered Code, for commercial or
+non-commercial purposes, provided that in each instance You also meet all of
+these conditions:
+
+(a) You must satisfy all the conditions of Section 2.1 with respect to the
+Source Code of the Covered Code;
+
+(b) You must duplicate, to the extent it does not already exist, the notice
+in Exhibit A in each file of the Source Code of all Your Modifications, and
+cause the modified files to carry prominent notices stating that You changed
+the files and the date of any change; and
+
+(c) If You Externally Deploy Your Modifications, You must make Source Code
+of all Your Externally Deployed Modifications either available to those to whom
+You have Externally Deployed Your Modifications, or publicly available. Source
+Code of Your Externally Deployed Modifications must be released under the terms
+set forth in this License, including the license grants set forth in Section 3
+below, for as long as you Externally Deploy the Covered Code or twelve (12)
+months from the date of initial External Deployment, whichever is longer. You
+should preferably distribute the Source Code of Your Externally Deployed
+Modifications electronically (e.g. download from a web site).
+
+2.3 Distribution of Executable Versions. In addition, if You Externally
+Deploy Covered Code (Original Code and/or Modifications) in object code,
+executable form only, You must include a prominent notice, in the code itself
+as well as in related documentation, stating that Source Code of the Covered
+Code is available under the terms of this License with information on how and
+where to obtain such Source Code.
+
+2.4 Third Party Rights. You expressly acknowledge and agree that although
+Apple and each Contributor grants the licenses to their respective portions of
+the Covered Code set forth herein, no assurances are provided by Apple or any
+Contributor that the Covered Code does not infringe the patent or other
+intellectual property rights of any other entity. Apple and each Contributor
+disclaim any liability to You for claims brought by any other entity based on
+infringement of intellectual property rights or otherwise. As a condition to
+exercising the rights and licenses granted hereunder, You hereby assume sole
+responsibility to secure any other intellectual property rights needed, if any.
+For example, if a third party patent license is required to allow You to
+distribute the Covered Code, it is Your responsibility to acquire that license
+before distributing the Covered Code.
+
+3. Your Grants. In consideration of, and as a condition to, the licenses
+granted to You under this License, You hereby grant to any person or entity
+receiving or distributing Covered Code under this License a non-exclusive,
+royalty-free, perpetual, irrevocable license, under Your Applicable Patent
+Rights and other intellectual property rights (other than patent) owned or
+controlled by You, to use, reproduce, display, perform, modify, sublicense,
+distribute and Externally Deploy Your Modifications of the same scope and
+extent as Apple's licenses under Sections 2.1 and 2.2 above.
+
+4. Larger Works. You may create a Larger Work by combining Covered Code
+with other code not governed by the terms of this License and distribute the
+Larger Work as a single product. In each such instance, You must make sure the
+requirements of this License are fulfilled for the Covered Code or any portion
+thereof.
+
+5. Limitations on Patent License. Except as expressly stated in Section
+2, no other patent rights, express or implied, are granted by Apple herein.
+Modifications and/or Larger Works may require additional patent licenses from
+Apple which Apple may grant in its sole discretion.
+
+6. Additional Terms. You may choose to offer, and to charge a fee for,
+warranty, support, indemnity or liability obligations and/or other rights
+consistent with the scope of the license granted herein ("Additional Terms") to
+one or more recipients of Covered Code. However, You may do so only on Your own
+behalf and as Your sole responsibility, and not on behalf of Apple or any
+Contributor. You must obtain the recipient's agreement that any such Additional
+Terms are offered by You alone, and You hereby agree to indemnify, defend and
+hold Apple and every Contributor harmless for any liability incurred by or
+claims asserted against Apple or such Contributor by reason of any such
+Additional Terms.
+
+7. Versions of the License. Apple may publish revised and/or new versions
+of this License from time to time. Each version will be given a distinguishing
+version number. Once Original Code has been published under a particular
+version of this License, You may continue to use it under the terms of that
+version. You may also choose to use such Original Code under the terms of any
+subsequent version of this License published by Apple. No one other than Apple
+has the right to modify the terms applicable to Covered Code created under this
+License.
+
+8. NO WARRANTY OR SUPPORT. The Covered Code may contain in whole or in
+part pre-release, untested, or not fully tested works. The Covered Code may
+contain errors that could cause failures or loss of data, and may be incomplete
+or contain inaccuracies. You expressly acknowledge and agree that use of the
+Covered Code, or any portion thereof, is at Your sole and entire risk. THE
+COVERED CODE IS PROVIDED "AS IS" AND WITHOUT WARRANTY, UPGRADES OR SUPPORT OF
+ANY KIND AND APPLE AND APPLE'S LICENSOR(S) (COLLECTIVELY REFERRED TO AS "APPLE"
+FOR THE PURPOSES OF SECTIONS 8 AND 9) AND ALL CONTRIBUTORS EXPRESSLY DISCLAIM
+ALL WARRANTIES AND/OR CONDITIONS, EXPRESS OR IMPLIED, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES AND/OR CONDITIONS OF MERCHANTABILITY, OF
+SATISFACTORY QUALITY, OF FITNESS FOR A PARTICULAR PURPOSE, OF ACCURACY, OF
+QUIET ENJOYMENT, AND NONINFRINGEMENT OF THIRD PARTY RIGHTS. APPLE AND EACH
+CONTRIBUTOR DOES NOT WARRANT AGAINST INTERFERENCE WITH YOUR ENJOYMENT OF THE
+COVERED CODE, THAT THE FUNCTIONS CONTAINED IN THE COVERED CODE WILL MEET YOUR
+REQUIREMENTS, THAT THE OPERATION OF THE COVERED CODE WILL BE UNINTERRUPTED OR
+ERROR-FREE, OR THAT DEFECTS IN THE COVERED CODE WILL BE CORRECTED. NO ORAL OR
+WRITTEN INFORMATION OR ADVICE GIVEN BY APPLE, AN APPLE AUTHORIZED
+REPRESENTATIVE OR ANY CONTRIBUTOR SHALL CREATE A WARRANTY. You acknowledge
+that the Covered Code is not intended for use in the operation of nuclear
+facilities, aircraft navigation, communication systems, or air traffic control
+machines in which case the failure of the Covered Code could lead to death,
+personal injury, or severe physical or environmental damage.
+
+9. LIMITATION OF LIABILITY. TO THE EXTENT NOT PROHIBITED BY LAW, IN NO
+EVENT SHALL APPLE OR ANY CONTRIBUTOR BE LIABLE FOR ANY INCIDENTAL, SPECIAL,
+INDIRECT OR CONSEQUENTIAL DAMAGES ARISING OUT OF OR RELATING TO THIS LICENSE OR
+YOUR USE OR INABILITY TO USE THE COVERED CODE, OR ANY PORTION THEREOF, WHETHER
+UNDER A THEORY OF CONTRACT, WARRANTY, TORT (INCLUDING NEGLIGENCE), PRODUCTS
+LIABILITY OR OTHERWISE, EVEN IF APPLE OR SUCH CONTRIBUTOR HAS BEEN ADVISED OF
+THE POSSIBILITY OF SUCH DAMAGES AND NOTWITHSTANDING THE FAILURE OF ESSENTIAL
+PURPOSE OF ANY REMEDY. SOME JURISDICTIONS DO NOT ALLOW THE LIMITATION OF
+LIABILITY OF INCIDENTAL OR CONSEQUENTIAL DAMAGES, SO THIS LIMITATION MAY NOT
+APPLY TO YOU. In no event shall Apple's total liability to You for all damages
+(other than as may be required by applicable law) under this License exceed the
+amount of fifty dollars ($50.00).
+
+10. Trademarks. This License does not grant any rights to use the
+trademarks or trade names "Apple", "Mac", "Mac OS", "QuickTime", "QuickTime
+Streaming Server" or any other trademarks, service marks, logos or trade names
+belonging to Apple (collectively "Apple Marks") or to any trademark, service
+mark, logo or trade name belonging to any Contributor. You agree not to use
+any Apple Marks in or as part of the name of products derived from the Original
+Code or to endorse or promote products derived from the Original Code other
+than as expressly permitted by and in strict compliance at all times with
+Apple's third party trademark usage guidelines which are posted at
+http://www.apple.com/legal/guidelinesfor3rdparties.html.
+
+11. Ownership. Subject to the licenses granted under this License, each
+Contributor retains all rights, title and interest in and to any Modifications
+made by such Contributor. Apple retains all rights, title and interest in and
+to the Original Code and any Modifications made by or on behalf of Apple
+("Apple Modifications"), and such Apple Modifications will not be automatically
+subject to this License. Apple may, at its sole discretion, choose to license
+such Apple Modifications under this License, or on different terms from those
+contained in this License or may choose not to license them at all.
+
+12. Termination.
+
+12.1 Termination. This License and the rights granted hereunder will
+terminate:
+
+(a) automatically without notice from Apple if You fail to comply with any
+term(s) of this License and fail to cure such breach within 30 days of becoming
+aware of such breach;
+(b) immediately in the event of the circumstances described in Section
+13.5(b); or
+(c) automatically without notice from Apple if You, at any time during the
+term of this License, commence an action for patent infringement against Apple;
+provided that Apple did not first commence an action for patent infringement
+against You in that instance.
+
+12.2 Effect of Termination. Upon termination, You agree to immediately stop
+any further use, reproduction, modification, sublicensing and distribution of
+the Covered Code. All sublicenses to the Covered Code which have been properly
+granted prior to termination shall survive any termination of this License.
+Provisions which, by their nature, should remain in effect beyond the
+termination of this License shall survive, including but not limited to
+Sections 3, 5, 8, 9, 10, 11, 12.2 and 13. No party will be liable to any other
+for compensation, indemnity or damages of any sort solely as a result of
+terminating this License in accordance with its terms, and termination of this
+License will be without prejudice to any other right or remedy of any party.
+
+13. Miscellaneous.
+
+13.1 Government End Users. The Covered Code is a "commercial item" as
+defined in FAR 2.101. Government software and technical data rights in the
+Covered Code include only those rights customarily provided to the public as
+defined in this License. This customary commercial license in technical data
+and software is provided in accordance with FAR 12.211 (Technical Data) and
+12.212 (Computer Software) and, for Department of Defense purchases, DFAR
+252.227-7015 (Technical Data -- Commercial Items) and 227.7202-3 (Rights in
+Commercial Computer Software or Computer Software Documentation). Accordingly,
+all U.S. Government End Users acquire Covered Code with only those rights set
+forth herein.
+
+13.2 Relationship of Parties. This License will not be construed as
+creating an agency, partnership, joint venture or any other form of legal
+association between or among You, Apple or any Contributor, and You will not
+represent to the contrary, whether expressly, by implication, appearance or
+otherwise.
+
+13.3 Independent Development. Nothing in this License will impair Apple's
+right to acquire, license, develop, have others develop for it, market and/or
+distribute technology or products that perform the same or similar functions
+as, or otherwise compete with, Modifications, Larger Works, technology or
+products that You may develop, produce, market or distribute.
+
+13.4 Waiver; Construction. Failure by Apple or any Contributor to enforce
+any provision of this License will not be deemed a waiver of future enforcement
+of that or any other provision. Any law or regulation which provides that the
+language of a contract shall be construed against the drafter will not apply to
+this License.
+
+13.5 Severability. (a) If for any reason a court of competent jurisdiction
+finds any provision of this License, or portion thereof, to be unenforceable,
+that provision of the License will be enforced to the maximum extent
+permissible so as to effect the economic benefits and intent of the parties,
+and the remainder of this License will continue in full force and effect. (b)
+Notwithstanding the foregoing, if applicable law prohibits or restricts You
+from fully and/or specifically complying with Sections 2 and/or 3 or prevents
+the enforceability of either of those Sections, this License will immediately
+terminate and You must immediately discontinue any use of the Covered Code and
+destroy all copies of it that are in your possession or control.
+
+13.6 Dispute Resolution. Any litigation or other dispute resolution between
+You and Apple relating to this License shall take place in the Northern
+District of California, and You and Apple hereby consent to the personal
+jurisdiction of, and venue in, the state and federal courts within that
+District with respect to this License. The application of the United Nations
+Convention on Contracts for the International Sale of Goods is expressly
+excluded.
+
+13.7 Entire Agreement; Governing Law. This License constitutes the entire
+agreement between the parties with respect to the subject matter hereof. This
+License shall be governed by the laws of the United States and the State of
+California, except that body of California law concerning conflicts of law.
+
+Where You are located in the province of Quebec, Canada, the following clause
+applies: The parties hereby confirm that they have requested that this License
+and all related documents be drafted in English. Les parties ont exigé que le
+présent contrat et tous les documents connexes soient rédigés en anglais.
+
+EXHIBIT A.
+
+"Portions Copyright (c) 1999-2007 Apple Inc. All Rights Reserved.
+
+This file contains Original Code and/or Modifications of Original Code as
+defined in and that are subject to the Apple Public Source License Version 2.0
+(the 'License'). You may not use this file except in compliance with the
+License. Please obtain a copy of the License at
+http://www.opensource.apple.com/apsl/ and read it before using this file.
+
+The Original Code and all software distributed under the License are
+distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS
+OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, INCLUDING WITHOUT
+LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
+PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. Please see the License for the
+specific language governing rights and limitations under the License."
+
/*
- * Copyright (c) 2004 Apple Computer, Inc. All rights reserved.
+ * Copyright (c) 2004-2010 Apple, Inc. All rights reserved.
*
* @APPLE_LICENSE_HEADER_START@
*
#include <sys/stat.h>
#include <sys/time.h>
#include <sys/xattr.h>
+#include <sys/attr.h>
#include <sys/syscall.h>
#include <sys/param.h>
#include <sys/mount.h>
#include <fts.h>
#include <libgen.h>
+#ifdef VOL_CAP_FMT_DECMPFS_COMPRESSION
+# include <Kernel/sys/decmpfs.h>
+#endif
+
#include <TargetConditionals.h>
#if !TARGET_OS_EMBEDDED
#include <quarantine.h>
static void * qtn_file_alloc(void) { return NULL; }
static int qtn_file_init_with_fd(void *x, int y) { return -1; }
static void qtn_file_free(void *x) { return; }
-static int qtn_file_apply_to_fd(void *x, int y) { return -1; }
+static int qtn_file_apply_to_fd(void *x, int y) { return 0; }
static char *qtn_error(int x) { return NULL; }
static int qtn_file_to_data(void *x, char *y, size_t z) { return -1; }
static void *qtn_file_clone(void *x) { return NULL; }
#include "copyfile.h"
enum cfInternalFlags {
- cfDelayAce = 1,
+ cfDelayAce = 1 << 0,
+ cfMakeFileInvisible = 1 << 1,
+ cfSawDecmpEA = 1 << 2,
};
/*
filesec_t permissive_fsec;
off_t totalCopied;
int err;
+ char *xattr_name;
};
struct acl_entry {
return ((ps1->ap_perms == ps2->ap_perms) ? 1 : 0);
}
+
+static int
+doesdecmpfs(int fd) {
+#ifdef DECMPFS_XATTR_NAME
+ int rv;
+ struct attrlist attrs;
+ char volroot[MAXPATHLEN + 1];
+ struct statfs sfs;
+ struct {
+ uint32_t length;
+ vol_capabilities_attr_t volAttrs;
+ } volattrs;
+
+ (void)fstatfs(fd, &sfs);
+ strlcpy(volroot, sfs.f_mntonname, sizeof(volroot));
+
+ memset(&attrs, 0, sizeof(attrs));
+ attrs.bitmapcount = ATTR_BIT_MAP_COUNT;
+ attrs.volattr = ATTR_VOL_CAPABILITIES;
+
+ rv = getattrlist(volroot, &attrs, &volattrs, sizeof(volattrs), 0);
+
+ if (rv != -1 &&
+ (volattrs.volAttrs.capabilities[VOL_CAPABILITIES_FORMAT] & VOL_CAP_FMT_DECMPFS_COMPRESSION) &&
+ (volattrs.volAttrs.valid[VOL_CAPABILITIES_FORMAT] & VOL_CAP_FMT_DECMPFS_COMPRESSION)) {
+ return 1;
+ }
+#endif
+ return 0;
+}
+
+
+static void
+sort_xattrname_list(void *start, size_t length)
+{
+ char **ptrs = NULL;
+ int nel;
+ char *tmp;
+ int indx = 0;
+
+ /* If it's not a proper C string at the end, don't do anything */
+ if (((char*)start)[length] != 0)
+ return;
+ /*
+ * In order to sort the list of names, we need to
+ * make a list of pointers to strings. To do that,
+ * we need to run through the buffer, and find the
+ * beginnings of strings.
+ */
+ nel = 10; // Most files don't have many EAs
+ ptrs = (char**)calloc(nel, sizeof(char*));
+
+ if (ptrs == NULL)
+ goto done;
+
+#ifdef DEBUG
+{
+ char *curPtr = start;
+ while (curPtr < (char*)start + length) {
+ printf("%s\n", curPtr);
+ curPtr += strlen(curPtr) + 1;
+ }
+}
+#endif
+
+ tmp = ptrs[indx++] = (char*)start;
+
+ while (tmp = memchr(tmp, 0, ((char*)start + length) - tmp)) {
+ if (indx == nel) {
+ nel += 10;
+ ptrs = realloc(ptrs, sizeof(char**) * nel);
+ if (ptrs == NULL)
+ goto done;
+ }
+ ptrs[indx++] = ++tmp;
+ }
+#ifdef DEBUG
+ printf("Unsorted:\n");
+ for (nel = 0; nel < indx-1; nel++) {
+ printf("\tEA %d = `%s'\n", nel, ptrs[nel]);
+ }
+#endif
+ qsort_b(ptrs, indx-1, sizeof(char*), ^(const void *left, const void *right) {
+ int rv;
+ char *lstr = *(char**)left, *rstr = *(char**)right;
+ rv = strcmp(lstr, rstr);
+ return rv;
+ });
+#ifdef DEBUG
+ printf("Sorted:\n");
+ for (nel = 0; nel < indx-1; nel++) {
+ printf("\tEA %d = `%s'\n", nel, ptrs[nel]);
+ }
+#endif
+ /*
+ * Now that it's sorted, we need to make a copy, so we can
+ * move the strings around into the new order. Then we
+ * copy that on top of the old buffer, and we're done.
+ */
+ char *copy = malloc(length);
+ if (copy) {
+ int i;
+ char *curPtr = copy;
+
+ for (i = 0; i < indx-1; i++) {
+ size_t len = strlen(ptrs[i]);
+ memcpy(curPtr, ptrs[i], len+1);
+ curPtr += len+1;
+ }
+ memcpy(start, copy, length);
+ free(copy);
+ }
+
+done:
+ if (ptrs)
+ free(ptrs);
+ return;
+}
+
/*
* Internally, the process is broken into a series of
* private functions.
*/
if (acl_create_entry_np(acl, &entry, ACL_FIRST_ENTRY) == -1)
goto error_exit;
- if (acl_get_permset(entry, &permset) == -1)
+ if (acl_get_permset(entry, &permset) == -1) {
+ copyfile_warn("acl_get_permset");
goto error_exit;
- if (acl_clear_perms(permset) == -1)
+ }
+ if (acl_clear_perms(permset) == -1) {
+ copyfile_warn("acl_clear_permset");
goto error_exit;
- if (acl_add_perm(permset, ACL_WRITE_DATA) == -1)
+ }
+ if (acl_add_perm(permset, ACL_WRITE_DATA) == -1) {
+ copyfile_warn("add ACL_WRITE_DATA");
goto error_exit;
- if (acl_add_perm(permset, ACL_WRITE_ATTRIBUTES) == -1)
+ }
+ if (acl_add_perm(permset, ACL_WRITE_ATTRIBUTES) == -1) {
+ copyfile_warn("add ACL_WRITE_ATTRIBUTES");
goto error_exit;
- if (acl_add_perm(permset, ACL_WRITE_EXTATTRIBUTES) == -1)
+ }
+ if (acl_add_perm(permset, ACL_WRITE_EXTATTRIBUTES) == -1) {
+ copyfile_warn("add ACL_WRITE_EXTATTRIBUTES");
goto error_exit;
- if (acl_add_perm(permset, ACL_APPEND_DATA) == -1)
+ }
+ if (acl_add_perm(permset, ACL_APPEND_DATA) == -1) {
+ copyfile_warn("add ACL_APPEND_DATA");
goto error_exit;
- if (acl_add_perm(permset, ACL_WRITE_SECURITY) == -1)
+ }
+ if (acl_add_perm(permset, ACL_WRITE_SECURITY) == -1) {
+ copyfile_warn("add ACL_WRITE_SECURITY");
goto error_exit;
- if (acl_set_tag_type(entry, ACL_EXTENDED_ALLOW) == -1)
+ }
+ if (acl_set_tag_type(entry, ACL_EXTENDED_ALLOW) == -1) {
+ copyfile_warn("set ACL_EXTENDED_ALLOW");
goto error_exit;
+ }
- if(acl_set_permset(entry, permset) == -1)
+ if(acl_set_permset(entry, permset) == -1) {
+ copyfile_warn("acl_set_permset");
goto error_exit;
- if(acl_set_qualifier(entry, qual) == -1)
+ }
+ if(acl_set_qualifier(entry, qual) == -1) {
+ copyfile_warn("acl_set_qualifier");
goto error_exit;
+ }
return 0;
error_exit:
acl_t tacl;
acl_entry_t tentry;
acl_tag_t tag;
- guid_t *qual;
+ guid_t *qual = NULL;
uuid_t myuuid;
// Who am I, and who is the ACE for?
done:
+ if (qual)
+ acl_free(qual);
+
if (tacl)
acl_free(tacl);
retval = -1;
goto done;
}
- if (sbuf.st_mode & S_IFDIR) {
+ if ((sbuf.st_mode & S_IFMT) == S_IFDIR) {
srcisdir = 1;
}
goto stopit;
}
}
- rv = copyfile(ftsent->fts_path, dstfile, tstate, flags);
+ int tmp_flags = (cmd == COPYFILE_RECURSE_DIR) ? (flags & ~COPYFILE_STAT) : flags;
+ rv = copyfile(ftsent->fts_path, dstfile, tstate, tmp_flags);
if (rv < 0) {
if (status) {
rv = (*status)(cmd, COPYFILE_ERR, tstate, ftsent->fts_path, dstfile, s->ctx);
}
}
} else if (cmd == COPYFILE_RECURSE_DIR_CLEANUP) {
- int tfd;
-
if (status) {
rv = (*status)(cmd, COPYFILE_START, tstate, ftsent->fts_path, dstfile, s->ctx);
if (rv == COPYFILE_QUIT) {
goto skipit;
}
}
- tfd = open(dstfile, O_RDONLY);
- if (tfd != -1) {
- struct stat sb;
- if (s->flags & COPYFILE_STAT) {
- (s->flags & COPYFILE_NOFOLLOW_SRC ? lstat : stat)(ftsent->fts_path, &sb);
- } else {
- (s->flags & COPYFILE_NOFOLLOW_DST ? lstat : stat)(dstfile, &sb);
- }
- remove_uberace(tfd, &sb);
- close(tfd);
- if (status) {
- rv = (*status)(COPYFILE_RECURSE_DIR_CLEANUP, COPYFILE_FINISH, tstate, ftsent->fts_path, dstfile, s->ctx);
- if (rv == COPYFILE_QUIT) {
- rv = -1; errno = 0;
- goto stopit;
- }
- }
- } else {
+ rv = copyfile(ftsent->fts_path, dstfile, tstate, (flags & COPYFILE_NOFOLLOW) | COPYFILE_STAT);
+ if (rv < 0) {
if (status) {
rv = (*status)(COPYFILE_RECURSE_DIR_CLEANUP, COPYFILE_ERR, tstate, ftsent->fts_path, dstfile, s->ctx);
if (rv == COPYFILE_QUIT) {
retval = -1;
goto stopit;
}
+ } else {
+ if (status) {
+ rv = (*status)(COPYFILE_RECURSE_DIR_CLEANUP, COPYFILE_FINISH, tstate, ftsent->fts_path, dstfile, s->ctx);
+ if (rv == COPYFILE_QUIT) {
+ retval = -1; errno = 0;
+ goto stopit;
+ }
+ }
}
+
rv = 0;
}
skipit:
goto error_exit;
if ((s->flags & COPYFILE_NOFOLLOW_DST) && lstat(s->dst, &dst_sb) == 0 &&
- (dst_sb.st_mode & S_IFLNK)) {
+ ((dst_sb.st_mode & S_IFMT) == S_IFLNK)) {
if (s->permissive_fsec)
free(s->permissive_fsec);
s->permissive_fsec = NULL;
} else if ((ret = copyfile_open(s)) < 0)
goto error_exit;
+ (void)fcntl(s->src_fd, F_NOCACHE, 1);
+ (void)fcntl(s->dst_fd, F_NOCACHE, 1);
+
ret = copyfile_internal(s, flags);
if (ret == -1)
goto error_exit;
reset_security(s);
+ if (s->src && (flags & COPYFILE_MOVE))
+ (void)remove(s->src);
+
exit:
if (state == NULL) {
int t = errno;
* to apply it to dst_fd. We don't care if
* it fails, not yet anyway.
*/
- if (s->qinfo)
- (void)qtn_file_apply_to_fd(s->qinfo, s->dst_fd);
+ if (s->qinfo) {
+ int qr = qtn_file_apply_to_fd(s->qinfo, s->dst_fd);
+ if (qr != 0) {
+ if (s->statuscb) {
+ int rv;
+
+ s->xattr_name = (char*)XATTR_QUARANTINE_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ s->err = errno = (qr < 0 ? ENOTSUP : qr);
+ ret = -1;
+ goto exit;
+ }
+ } else {
+ s->err = errno = (qr < 0 ? ENOTSUP : qr);
+ ret = -1;
+ goto exit;
+ }
+ }
+ }
/*
* COPYFILE_XATTR tells us to copy the extended attributes;
copyfile_warn("error closing files");
return -1;
}
+ if (s->xattr_name)
+ free(s->xattr_name);
if (s->dst)
free(s->dst);
if (s->src)
bp = calloc(1, sz);
if (bp == NULL) {
- copyfile_warn("cannot allocate %d bytes", sz);
+ copyfile_warn("cannot allocate %zd bytes", sz);
return -1;
}
if (readlink(s->src, bp, sz-1) == -1) {
}
} else if (isdir) {
mode_t mode;
- mode = s->sb.st_mode & ~S_IFMT;
+ mode = (s->sb.st_mode & ~S_IFMT) | S_IRWXU;
if (mkdir(s->dst, mode) == -1) {
if (errno != EEXIST || (s->flags & COPYFILE_EXCL)) {
if ((s->sb.st_mode & S_IFMT) != S_IFREG)
return 0;
+#ifdef VOL_CAP_FMT_DECMPFS_COMPRESSION
+ if (s->internal_flags & cfSawDecmpEA) {
+ if (s->sb.st_flags & UF_COMPRESSED) {
+ if ((s->flags & COPYFILE_STAT) == 0) {
+ if (fchflags(s->dst_fd, UF_COMPRESSED) == 0) {
+ goto exit;
+ }
+ }
+ }
+ }
+#endif
+
if (fstatfs(s->src_fd, &sfs) == -1) {
iBlocksize = s->sb.st_blksize;
} else {
if (status) {
int rv = (*status)(COPYFILE_COPY_DATA, COPYFILE_PROGRESS, s, s->src, s->dst, s->ctx);
if (rv == COPYFILE_QUIT) {
- ret = -1; s->err = ECANCELED;
+ ret = -1; s->err = errno = ECANCELED;
goto exit;
}
}
goto exit;
}
- if (ftruncate(s->dst_fd, s->sb.st_size) < 0)
+ if (ftruncate(s->dst_fd, s->totalCopied) < 0)
{
ret = -1;
goto exit;
static int copyfile_security(copyfile_state_t s)
{
int copied = 0;
- int has_uberace = 0;
- acl_flagset_t flags;
struct stat sb;
- acl_entry_t entry_src = NULL, entry_dst = NULL;
- acl_t acl_src = NULL, acl_dst = NULL;
+ acl_t acl_src = NULL, acl_tmp = NULL, acl_dst = NULL;
int ret = 0;
filesec_t tmp_fsec = NULL;
filesec_t fsec_dst = filesec_init();
else
goto error_exit;
}
- else
- {
- acl_t tmp = acl_init(4);
- acl_entry_t ace = NULL;
- int count = 0;
-
- if (tmp == NULL)
- goto error_exit;
-
-
- for (; acl_get_entry(acl_dst,
- ace == NULL ? ACL_FIRST_ENTRY : ACL_NEXT_ENTRY,
- &ace) == 0;)
- {
- if (count++ == 0 && is_uberace(ace)) {
- if ((ret = acl_create_entry(&tmp, &entry_dst)) == -1)
- break;
- if ((ret = acl_copy_entry(entry_dst, ace)) == -1)
- break;
- has_uberace = 1;
- continue;
- }
- acl_get_flagset_np(ace, &flags);
- if (acl_get_flag_np(flags, ACL_ENTRY_INHERITED))
- {
- if ((ret = acl_create_entry(&tmp, &entry_dst)) == -1)
- break;
- if ((ret = acl_copy_entry(entry_dst, ace)) == -1)
- break;
- }
- }
- acl_free(acl_dst);
- acl_dst = tmp;
-
- if (ret == -1)
- goto error_exit;
- }
if (acl_src == NULL && acl_dst == NULL)
goto no_acl;
+ acl_tmp = acl_init(4);
+ if (acl_tmp == NULL)
+ goto error_exit;
+
if (acl_src) {
- if (acl_dst == NULL)
- acl_dst = acl_init(4);
- for (copied = 0;acl_get_entry(acl_src,
- entry_src == NULL ? ACL_FIRST_ENTRY : ACL_NEXT_ENTRY,
- &entry_src) == 0;)
+ acl_entry_t ace = NULL;
+ acl_entry_t tmp = NULL;
+ for (copied = 0;
+ acl_get_entry(acl_src,
+ ace == NULL ? ACL_FIRST_ENTRY : ACL_NEXT_ENTRY,
+ &ace) == 0;)
{
- acl_get_flagset_np(entry_src, &flags);
+ acl_flagset_t flags = { 0 };
+ acl_get_flagset_np(ace, &flags);
if (!acl_get_flag_np(flags, ACL_ENTRY_INHERITED))
{
- if ((ret = acl_create_entry(&acl_dst, &entry_dst)) == -1)
+ if ((ret = acl_create_entry(&acl_tmp, &tmp)) == -1)
goto error_exit;
- if ((ret = acl_copy_entry(entry_dst, entry_src)) == -1)
+ if ((ret = acl_copy_entry(tmp, ace)) == -1)
goto error_exit;
copyfile_debug(2, "copied acl entry from %s to %s",
s->src ? s->src : "(null src)",
- s->dst ? s->dst : "(null dst)");
+ s->dst ? s->dst : "(null tmp)");
copied++;
}
}
}
- if (!has_uberace && (s->internal_flags & cfDelayAce)) {
- if (add_uberace(&acl_dst))
- goto error_exit;
+ if (acl_dst) {
+ acl_entry_t ace = NULL;
+ acl_entry_t tmp = NULL;
+ acl_flagset_t flags = { 0 };
+ for (copied = 0;acl_get_entry(acl_dst,
+ ace == NULL ? ACL_FIRST_ENTRY : ACL_NEXT_ENTRY,
+ &ace) == 0;)
+ {
+ acl_get_flagset_np(ace, &flags);
+ if (acl_get_flag_np(flags, ACL_ENTRY_INHERITED))
+ {
+ if ((ret = acl_create_entry(&acl_tmp, &tmp)) == -1)
+ goto error_exit;
+
+ if ((ret = acl_copy_entry(tmp, ace)) == -1)
+ goto error_exit;
+
+ copyfile_debug(2, "copied acl entry from %s to %s",
+ s->src ? s->src : "(null dst)",
+ s->dst ? s->dst : "(null tmp)");
+ copied++;
+ }
+ }
}
- if (!filesec_set_property(s->fsec, FILESEC_ACL, &acl_dst))
+ if (!filesec_set_property(s->fsec, FILESEC_ACL, &acl_tmp))
{
copyfile_debug(3, "altered acl");
}
* discretely. We don't care if the fchown fails, but we do care
* if the mode or ACL can't be set. For historical reasons, we
* simply log those failures, however.
+ *
+ * Big warning here: we may NOT have COPYFILE_STAT set, since
+ * we fell-through from COPYFILE_ACL. So check for the fchmod.
*/
#define NS(x) ((x) ? (x) : "(null string)")
- if (fchmod(s->dst_fd, s->sb.st_mode) == -1) {
+ if ((s->flags & COPYFILE_STAT) &&
+ fchmod(s->dst_fd, s->sb.st_mode) == -1) {
copyfile_warn("could not change mode of destination file %s to match source file %s", NS(s->dst), NS(s->src));
}
(void)fchown(s->dst_fd, s->sb.st_uid, s->sb.st_gid);
filesec_free(fsec_dst);
if (acl_src) acl_free(acl_src);
if (acl_dst) acl_free(acl_dst);
+ if (acl_tmp) acl_free(acl_tmp);
return ret;
static int copyfile_stat(copyfile_state_t s)
{
struct timeval tval[2];
+ unsigned int added_flags = 0;
+
/*
- * NFS doesn't support chflags; ignore errors unless there's reason
- * to believe we're losing bits. (Note, this still won't be right
- * if the server supports flags and we were trying to *remove* flags
- * on a file that we copied, i.e., that we didn't create.)
+ * NFS doesn't support chflags; ignore errors as a result, since
+ * we don't return failure for this.
*/
- if (fchflags(s->dst_fd, (u_int)s->sb.st_flags))
- if (errno != EOPNOTSUPP || s->sb.st_flags != 0)
- copyfile_warn("%s: set flags (was: 0%07o)", s->dst ? s->dst : "(null dst)", s->sb.st_flags);
+ if (s->internal_flags & cfMakeFileInvisible)
+ added_flags |= UF_HIDDEN;
+
+ (void)fchflags(s->dst_fd, (u_int)s->sb.st_flags | added_flags);
/* If this fails, we don't care */
(void)fchown(s->dst_fd, s->sb.st_uid, s->sb.st_gid);
tval[0].tv_sec = s->sb.st_atime;
tval[1].tv_sec = s->sb.st_mtime;
tval[0].tv_usec = tval[1].tv_usec = 0;
- if (futimes(s->dst_fd, tval))
- copyfile_warn("%s: set times", s->dst ? s->dst : "(null dst)");
+ (void)futimes(s->dst_fd, tval);
+
return 0;
}
ssize_t asize;
ssize_t nsize;
int ret = 0;
+ int look_for_decmpea = 0;
/* delete EAs on destination */
if ((nsize = flistxattr(s->dst_fd, 0, 0, 0)) > 0)
return -1;
}
+#ifdef DECMPFS_XATTR_NAME
+ if ((s->flags & COPYFILE_DATA) &&
+ (s->sb.st_flags & UF_COMPRESSED) &&
+ doesdecmpfs(s->src_fd) &&
+ doesdecmpfs(s->dst_fd)) {
+ look_for_decmpea = XATTR_SHOWCOMPRESSION;
+ }
+#endif
+
/* get name list of EAs on source */
- if ((nsize = flistxattr(s->src_fd, 0, 0, 0)) < 0)
+ if ((nsize = flistxattr(s->src_fd, 0, 0, look_for_decmpea)) < 0)
{
if (errno == ENOTSUP || errno == EPERM)
return 0;
if ((namebuf = (char *) malloc(nsize)) == NULL)
return -1;
else
- nsize = flistxattr(s->src_fd, namebuf, nsize, 0);
+ nsize = flistxattr(s->src_fd, namebuf, nsize, look_for_decmpea);
if (nsize <= 0) {
free(namebuf);
if (strncmp(name, XATTR_QUARANTINE_NAME, end - name) == 0)
continue;
- if ((xa_size = fgetxattr(s->src_fd, name, 0, 0, 0, 0)) < 0)
+ if ((xa_size = fgetxattr(s->src_fd, name, 0, 0, 0, look_for_decmpea)) < 0)
{
- ret = -1;
continue;
}
}
}
- if ((asize = fgetxattr(s->src_fd, name, xa_dataptr, xa_size, 0, 0)) < 0)
+ if ((asize = fgetxattr(s->src_fd, name, xa_dataptr, xa_size, 0, look_for_decmpea)) < 0)
{
- ret = -1;
continue;
}
if (xa_size != asize)
xa_size = asize;
- if (fsetxattr(s->dst_fd, name, xa_dataptr, xa_size, 0, 0) < 0)
+#ifdef DECMPFS_XATTR_NAME
+ if (strncmp(name, DECMPFS_XATTR_NAME, end-name) == 0)
{
- ret = -1;
- continue;
+ decmpfs_disk_header *hdr = xa_dataptr;
+
+ /*
+ * If the EA has the decmpfs name, but is too
+ * small, or doesn't have the right magic number,
+ * or isn't the right type, we'll just skip it.
+ * This means it won't end up in the destination
+ * file, and data copy will happen normally.
+ */
+ if ((size_t)xa_size < sizeof(decmpfs_disk_header)) {
+ continue;
+ }
+ if (OSSwapLittleToHostInt32(hdr->compression_magic) != DECMPFS_MAGIC) {
+ continue;
+ }
+ if (OSSwapLittleToHostInt32(hdr->compression_type) != 3 &&
+ OSSwapLittleToHostInt32(hdr->compression_type) != 4) {
+ continue;
+ }
+ s->internal_flags |= cfSawDecmpEA;
+ }
+#endif
+
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ s->xattr_name = strdup(name);
+
+ if (s->statuscb) {
+ int rv;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT) {
+ s->err = ECANCELED;
+ goto out;
+ } else if (rv == COPYFILE_SKIP) {
+ continue;
+ }
+ }
+ if (fsetxattr(s->dst_fd, name, xa_dataptr, xa_size, 0, look_for_decmpea) < 0)
+ {
+ if (s->statuscb)
+ {
+ int rv;
+ if (s->xattr_name == NULL)
+ s->xattr_name = strdup(name);
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT)
+ {
+ s->err = ECANCELED;
+ ret = -1;
+ goto out;
+ }
+ }
+ else
+ {
+ ret = -1;
+ copyfile_warn("could not set attributes %s on destination file descriptor: %s", name, strerror(errno));
+ continue;
+ }
+ }
+ if (s->statuscb) {
+ int rv;
+ if (s->xattr_name == NULL)
+ s->xattr_name = strdup(name);
+
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT) {
+ s->err = ECANCELED;
+ goto out;
+ }
}
}
+out:
if (namebuf)
free(namebuf);
free((void *) xa_dataptr);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
return ret;
}
case COPYFILE_STATE_COPIED:
*(off_t*)ret = s->totalCopied;
break;
+#endif
+#ifdef COPYFILE_STATE_XATTRNAME
+ case COPYFILE_STATE_XATTRNAME:
+ *(char**)ret = s->xattr_name;
+ break;
#endif
default:
errno = EINVAL;
#define offsetof(type, member) ((size_t)(&((type *)0)->member))
-#define XATTR_MAXATTRLEN (4*1024)
+#define XATTR_MAXATTRLEN (32*1024)
/*
buffer = calloc(1, hdrsize);
if (buffer == NULL) {
- copyfile_debug(1, "copyfile_unpack: calloc(1, %u) returned NULL", hdrsize);
+ copyfile_debug(1, "copyfile_unpack: calloc(1, %zu) returned NULL", hdrsize);
error = -1;
goto exit;
} else
if (bytes < 0)
{
- copyfile_debug(1, "pread returned: %d", bytes);
+ copyfile_debug(1, "pread returned: %zd", bytes);
error = -1;
goto exit;
}
* Remove any extended attributes on the target.
*/
- if (COPYFILE_XATTR & s->flags)
+ if ((bytes = flistxattr(s->dst_fd, 0, 0, 0)) > 0)
{
- if ((bytes = flistxattr(s->dst_fd, 0, 0, 0)) > 0)
- {
- char *namebuf, *name;
+ char *namebuf, *name;
- if ((namebuf = (char*) malloc(bytes)) == NULL)
- {
- s->err = ENOMEM;
- goto exit;
- }
- bytes = flistxattr(s->dst_fd, namebuf, bytes, 0);
-
- if (bytes > 0)
- for (name = namebuf; name < namebuf + bytes; name += strlen(name) + 1)
- (void)fremovexattr(s->dst_fd, name, 0);
-
- free(namebuf);
- }
- else if (bytes < 0)
+ if ((namebuf = (char*) malloc(bytes)) == NULL)
{
- if (errno != ENOTSUP && errno != EPERM)
+ s->err = ENOMEM;
goto exit;
}
+ bytes = flistxattr(s->dst_fd, namebuf, bytes, 0);
+
+ if (bytes > 0)
+ for (name = namebuf; name < namebuf + bytes; name += strlen(name) + 1)
+ (void)fremovexattr(s->dst_fd, name, 0);
+
+ free(namebuf);
+ }
+ else if (bytes < 0)
+ {
+ if (errno != ENOTSUP && errno != EPERM)
+ goto exit;
}
/*
int i;
if ((size_t)hdrsize < sizeof(attr_header_t)) {
- copyfile_warn("bad attribute header: %u < %u", hdrsize, sizeof(attr_header_t));
+ copyfile_warn("bad attribute header: %zu < %zu", hdrsize, sizeof(attr_header_t));
error = -1;
goto exit;
}
dataptr = (char *)attrhdr + entry->offset;
if (dataptr > endptr || dataptr < buffer) {
- copyfile_debug(1, "Entry %d overflows: offset = %u", entry->offset);
+ copyfile_debug(1, "Entry %d overflows: offset = %u", i, entry->offset);
error = -1;
s->err = EINVAL; /* Invalid buffer */
goto exit;
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn("Incomplete or corrupt attribute entry");
copyfile_debug(1, "Entry %d length overflows: offset = %u, length = %u",
- entry->offset, entry->length);
+ i, entry->offset, entry->length);
error = -1;
s->err = EINVAL; /* Invalid buffer */
goto exit;
{
int x;
x = qtn_file_apply_to_fd(tqinfo, s->dst_fd);
- if (x != 0)
+ if (x != 0) {
copyfile_warn("qtn_file_apply_to_fd failed: %s", qtn_error(x));
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = (char*)XATTR_QUARANTINE_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = s->err = x < 0 ? ENOTSUP : errno;
+ goto exit;
+ }
+ } else {
+ error = s->err = x < 0 ? ENOTSUP : errno;
+ goto exit;
+ }
+ }
}
if (tqinfo && !s->qinfo)
{
}
}
/* Look for ACL data */
- else if (COPYFILE_ACL & s->flags && strcmp((char*)entry->name, XATTR_SECURITY_NAME) == 0)
+ else if (strcmp((char*)entry->name, XATTR_SECURITY_NAME) == 0)
{
acl_t acl;
struct stat sb;
int retry = 1;
char *tcp = dataptr;
+ if (entry->length == 0) {
+ /* Not sure how we got here, but we had one case
+ * where it was 0. In a normal EA, we can have a 0-byte
+ * payload. That means nothing in this case, so we'll
+ * simply skip the EA.
+ */
+ error = 0;
+ goto acl_done;
+ }
/*
* acl_from_text() requires a NUL-terminated string. The ACL EA,
* however, may not be NUL-terminated. So in that case, we need to
acl_free(acl);
filesec_free(fsec_tmp);
+acl_done:
if (error == -1)
goto exit;
}
}
/* And, finally, everything else */
- else if (COPYFILE_XATTR & s->flags) {
- if (fsetxattr(s->dst_fd, (char *)entry->name, dataptr, entry->length, 0, 0) == -1) {
+ else
+ {
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = strdup((char*)entry->name);
+ s->totalCopied = 0;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT) {
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ }
+ if (fsetxattr(s->dst_fd, (char *)entry->name, dataptr, entry->length, 0, 0) == -1) {
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn("error %d setting attribute %s", error, entry->name);
- error = -1;
- goto exit;
+ if (s->statuscb) {
+ int rv;
+
+ s->xattr_name = strdup((char*)entry->name);
+ rv = (s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ goto exit;
+ }
+ } else {
+ error = -1;
+ goto exit;
+ }
+ } else if (s->statuscb) {
+ int rv;
+ s->xattr_name = strdup((char*)entry->name);
+ s->totalCopied = entry->length;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ goto exit;
+ }
}
}
entry = ATTR_NEXT(entry);
if (bcmp((u_int8_t*)buffer + adhdr->entries[0].offset, emptyfinfo, sizeof(emptyfinfo)) != 0)
{
+ uint16_t *fFlags;
+ uint8_t *newFinfo;
+ enum { kFinderInvisibleMask = 1 << 14 };
+
+ newFinfo = (u_int8_t*)buffer + adhdr->entries[0].offset;
+ fFlags = (uint16_t*)&newFinfo[8];
copyfile_debug(3, " extracting \"%s\" (32 bytes)", XATTR_FINDERINFO_NAME);
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = (char*)XATTR_FINDERINFO_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ goto exit;
+ } else if (rv == COPYFILE_SKIP) {
+ goto skip_fi;
+ }
+ }
error = fsetxattr(s->dst_fd, XATTR_FINDERINFO_NAME, (u_int8_t*)buffer + adhdr->entries[0].offset, sizeof(emptyfinfo), 0, 0);
- if (error)
+ if (error) {
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = XATTR_FINDERINFO_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ goto exit;
+ }
+ }
goto exit;
+ } else if (s->statuscb) {
+ int rv;
+ s->xattr_name = XATTR_FINDERINFO_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ goto exit;
+ }
+ }
+ if (SWAP16(*fFlags) & kFinderInvisibleMask)
+ s->internal_flags |= cfMakeFileInvisible;
}
+skip_fi:
/*
* Extract the Resource Fork.
rsrcforkdata = malloc(length);
if (rsrcforkdata == NULL) {
- copyfile_debug(1, "could not allocate %u bytes for rsrcforkdata",
+ copyfile_debug(1, "could not allocate %zu bytes for rsrcforkdata",
length);
error = -1;
goto bad;
error = -1;
goto bad;
}
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = XATTR_RESOURCEFORK_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ if (rsrcforkdata)
+ free(rsrcforkdata);
+ goto exit;
+ } else if (rv == COPYFILE_SKIP) {
+ goto bad;
+ }
+ }
error = fsetxattr(s->dst_fd, XATTR_RESOURCEFORK_NAME, rsrcforkdata, bytes, 0, 0);
if (error)
{
error = errno = 0;
goto bad;
}
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = XATTR_RESOURCEFORK_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_CONTINUE) {
+ error = errno = 0;
+ goto bad;
+ }
+ }
copyfile_debug(1, "error %d setting resource fork attribute", error);
error = -1;
goto bad;
+ } else if (s->statuscb) {
+ int rv;
+ s->xattr_name = XATTR_RESOURCEFORK_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ if (rsrcforkdata)
+ free(rsrcforkdata);
+ goto exit;
+ }
}
copyfile_debug(3, "extracting \"%s\" (%d bytes)",
XATTR_RESOURCEFORK_NAME, (int)length);
char *databuf = NULL;
int ret = 0;
+/*
+ * XXX
+ * do COPYFILE_COPY_XATTR here; no need to
+ * the work if we want to skip.
+ */
+
+ if (s->statuscb)
+ {
+ int rv;
+
+ s->xattr_name = (char*)XATTR_RESOURCEFORK_NAME;
+
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_SKIP) {
+ ret = 0;
+ goto done;
+ }
+ if (rv == COPYFILE_QUIT) {
+ ret = -1;
+ s->err = ECANCELED;
+ goto done;
+ }
+ }
/* Get the resource fork size */
if ((datasize = fgetxattr(s->src_fd, XATTR_RESOURCEFORK_NAME, NULL, 0, 0, 0)) < 0)
{
goto done;
}
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = (char*)XATTR_RESOURCEFORK_NAME;
+
+ s->totalCopied = 0;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_PROGRESS, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ s->err = ECANCELED;
+ ret = -1;
+ goto done;
+ }
+ }
if ((databuf = malloc(datasize)) == NULL)
{
copyfile_warn("malloc");
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn("couldn't write resource fork");
}
- copyfile_debug(3, "copied %d bytes of \"%s\" data @ offset 0x%08x",
+ if (s->statuscb)
+ {
+ int rv;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT) {
+ ret = -1;
+ goto done;
+ }
+ }
+ copyfile_debug(3, "copied %zd bytes of \"%s\" data @ offset 0x%08x",
datasize, XATTR_RESOURCEFORK_NAME, filehdr->appledouble.entries[1].offset);
filehdr->appledouble.entries[1].length = (u_int32_t)datasize;
done:
+ if (ret == -1 && s->statuscb)
+ {
+ int rv;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_CONTINUE)
+ ret = 0;
+ }
+ if (s->xattr_name) {
+ s->xattr_name = NULL;
+ }
if (databuf)
free(databuf);
+/*
+ * XXX
+ * Do status callback here
+ * If ret == -1, then error callback
+ */
return ret;
}
* Fill in the initial Attribute Header.
*/
filehdr->magic = ATTR_HDR_MAGIC;
- filehdr->debug_tag = (u_int32_t)s->sb.st_ino;
+ filehdr->debug_tag = 0;
filehdr->data_start = (u_int32_t)sizeof(attr_header_t);
/*
{
offset = strlen(XATTR_SECURITY_NAME) + 1;
strcpy(attrnamebuf, XATTR_SECURITY_NAME);
+ endnamebuf = attrnamebuf + offset;
}
if (temp_acl)
acl_free(temp_acl);
listsize = left;
}
- listsize += offset;
- endnamebuf = attrnamebuf + listsize;
+ endnamebuf = attrnamebuf + offset + (listsize > 0 ? listsize : 0);
if (endnamebuf > (attrnamebuf + ATTR_MAX_HDR_SIZE)) {
error = -1;
goto exit;
}
+ if (listsize > 0)
+ sort_xattrname_list(attrnamebuf, endnamebuf - attrnamebuf);
+
for (nameptr = attrnamebuf; nameptr < endnamebuf; nameptr += namelen)
{
namelen = strlen(nameptr) + 1;
if (namelen > XATTR_MAXNAMELEN + 1) {
namelen = XATTR_MAXNAMELEN + 1;
}
+ if (s->statuscb) {
+ int rv;
+ char eaname[namelen];
+ bcopy(nameptr, eaname, namelen);
+ eaname[namelen] = 0; // Just to be sure!
+ s->xattr_name = eaname;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ s->err = ECANCELED;
+ goto exit;
+ } else if (rv == COPYFILE_SKIP) {
+ size_t amt = endnamebuf - (nameptr + namelen);
+ memmove(nameptr, nameptr + namelen, amt);
+ endnamebuf -= namelen;
+ /* Set namelen to 0 so continue doesn't miss names */
+ namelen = 0;
+ continue;
+ }
+ }
entry->namelen = namelen;
entry->flags = 0;
if (nameptr + namelen > endnamebuf) {
error = -1;
goto exit;
}
+
bcopy(nameptr, &entry->name[0], namelen);
copyfile_debug(2, "copied name [%s]", entry->name);
*/
entry = (attr_entry_t *)((char *)filehdr + sizeof(attr_header_t));
- for (nameptr = attrnamebuf; nameptr < attrnamebuf + listsize; nameptr += namelen + 1)
+ for (nameptr = attrnamebuf; nameptr < endnamebuf; nameptr += namelen + 1)
{
namelen = strlen(nameptr);
/* Check for Finder Info. */
else if (strcmp(nameptr, XATTR_FINDERINFO_NAME) == 0)
{
+ if (s->statuscb)
+ {
+ int rv;
+ s->xattr_name = (char*)XATTR_FINDERINFO_NAME;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_START, s, s->src, s->dst, s->ctx);
+ if (rv == COPYFILE_QUIT)
+ {
+ s->xattr_name = NULL;
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ else if (rv == COPYFILE_SKIP)
+ {
+ s->xattr_name = NULL;
+ continue;
+ }
+ s->totalCopied = 0;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_PROGRESS, s, s->src, s->dst, s->ctx);
+ s->xattr_name = NULL;
+ if (rv == COPYFILE_QUIT)
+ {
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ }
datasize = fgetxattr(s->src_fd, nameptr, (u_int8_t*)filehdr + filehdr->appledouble.entries[0].offset, 32, 0, 0);
if (datasize < 0)
{
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = strdup(nameptr);
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ goto exit;
+ }
+ }
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn("skipping attr \"%s\" due to error %d", nameptr, errno);
} else if (datasize != 32)
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn(" copied 32 bytes of \"%s\" data @ offset 0x%08x",
XATTR_FINDERINFO_NAME, filehdr->appledouble.entries[0].offset);
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = strdup(nameptr);
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT) {
+ error = -1;
+ goto exit;
+ }
+ }
}
continue; /* finder info doesn't have an attribute entry */
}
} else
{
/* Just a normal attribute. */
+ if (s->statuscb)
+ {
+ int rv;
+ s->xattr_name = strdup(nameptr);
+ s->totalCopied = 0;
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_PROGRESS, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ /*
+ * Due to the nature of the packed file, we can't skip at this point.
+ */
+ if (rv == COPYFILE_QUIT)
+ {
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ }
datasize = fgetxattr(s->src_fd, nameptr, NULL, 0, 0, 0);
if (datasize == 0)
goto next;
{
if (COPYFILE_VERBOSE & s->flags)
copyfile_warn("skipping attr \"%s\" due to error %d", nameptr, errno);
+ if (s->statuscb)
+ {
+ int rv;
+ s->xattr_name = strdup(nameptr);
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_ERR, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT)
+ {
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ }
goto next;
}
if (datasize > XATTR_MAXATTRLEN)
continue;
}
datasize = fgetxattr(s->src_fd, nameptr, databuf, datasize, 0, 0);
+ if (s->statuscb) {
+ int rv;
+ s->xattr_name = strdup(nameptr);
+ rv = (*s->statuscb)(COPYFILE_COPY_XATTR, COPYFILE_FINISH, s, s->src, s->dst, s->ctx);
+ if (s->xattr_name) {
+ free(s->xattr_name);
+ s->xattr_name = NULL;
+ }
+ if (rv == COPYFILE_QUIT) {
+ s->err = ECANCELED;
+ error = -1;
+ goto exit;
+ }
+ }
}
entry->length = (u_int32_t)datasize;