X-Git-Url: https://git.saurik.com/apple/security.git/blobdiff_plain/80e2389990082500d76eb566d4946be3e786c3ef..d8f41ccd20de16f8ebe2ccc84d47bf1cb2b26bbb:/Security/libsecurity_cdsa_utilities/lib/cssmdates.cpp?ds=inline diff --git a/Security/libsecurity_cdsa_utilities/lib/cssmdates.cpp b/Security/libsecurity_cdsa_utilities/lib/cssmdates.cpp new file mode 100644 index 00000000..e5b46b8c --- /dev/null +++ b/Security/libsecurity_cdsa_utilities/lib/cssmdates.cpp @@ -0,0 +1,195 @@ +/* + * Copyright (c) 2000-2004,2006,2011-2012,2014 Apple Inc. All Rights Reserved. + * + * @APPLE_LICENSE_HEADER_START@ + * + * 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. + * + * @APPLE_LICENSE_HEADER_END@ + */ + + +// +// Manage the Tower of Babel of CSSM dates and times +// +#include +#include +#include +#include + + +// +// A (private) PODwrapper for CFGregorianDate +// +struct Gregorian : public PodWrapper { + Gregorian() { } + + Gregorian(int y, int m, int d, int h = 0, int min = 0, double sec = 0) + { + year = y; month = m; day = d; + hour = h; minute = min; second = sec; + } + + Gregorian(CFAbsoluteTime ref) + { static_cast(*this) = CFAbsoluteTimeGetGregorianDate(ref, NULL); } + + operator CFAbsoluteTime () const + { return CFGregorianDateGetAbsoluteTime(*this, NULL); } +}; + + +// +// The CssmDate PODwrapper +// +CssmDate::CssmDate(const char *y, const char *m, const char *d) +{ + assign(years(), 4, y); + assign(months(), 2, m); + assign(days(), 2, d); +} + +CssmDate::CssmDate(int y, int m, int d) +{ + // internal format is "yyyymmdd" (no null termination) + char str[9]; + if (8 != snprintf(str, 9, "%4.4d%2.2d%2.2d", y, m, d)) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + memcpy(this, str, 8); +} + +int CssmDate::year() const +{ return atoi(string(years(), 4).c_str()); } + +int CssmDate::month() const +{ return atoi(string(months(), 2).c_str()); } + +int CssmDate::day() const +{ return atoi(string(days(), 2).c_str()); } + +// right-adjust fill +void CssmDate::assign(char *dest, int width, const char *src) +{ + // pick last width characters of src at most + size_t len = strlen(src); + if (len > width) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + memset(dest, '0', width - len); + memcpy(dest + width - len, src, len); +} + + +// +// CssmUniformDate core functions +// + + +// +// Uniform conversions with CFDateRef +// +CssmUniformDate::CssmUniformDate(CFDateRef ref) +{ + mTime = CFDateGetAbsoluteTime(ref); +} + +CssmUniformDate::operator CFDateRef() const +{ + return CFDateCreate(NULL, mTime); +} + + +// +// Uniform conversions with CssmDates +// +CssmUniformDate::CssmUniformDate(const CssmDate &date) +{ + mTime = CFGregorianDateGetAbsoluteTime(Gregorian(date.year(), date.month(), date.day()), + NULL); +} + +CssmUniformDate::operator CssmDate () const +{ + Gregorian greg(mTime); + return CssmDate(greg.year, greg.month, greg.day); +} + + +// +// Uniform conversions with CssmData (1999-06-30_15:05:39 form) +// +CssmUniformDate::CssmUniformDate(const CSSM_DATA &inData) +{ + const CssmData &data = CssmData::overlay(inData); + if (data.length() != 19) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + setFromString(reinterpret_cast(inData.Data), "%ld-%d-%d_%d:%d:%lf", 19); +} + +void CssmUniformDate::convertTo(CssmOwnedData &data) const +{ + Gregorian greg(mTime); + char str[20]; + if (19 != snprintf(str, 20, "%4.4d-%2.2d-%2.2d_%2.2d:%2.2d:%2.2d", + int(greg.year), greg.month, greg.day, greg.hour, greg.minute, int(greg.second))) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + data = CssmData(str, 19); +} + + +// +// Uniform conversions with CSSM_TIMESTRING (19990630150539 form) +// +CssmUniformDate::CssmUniformDate(const char *src) +{ + setFromString(src, "%4ld%2d%2d%2d%2d%2lf", 14); +} + +void CssmUniformDate::convertTo(char *dst, size_t length) const +{ + if (length < 14) + CssmError::throwMe(CSSMERR_CSSM_BUFFER_TOO_SMALL); + Gregorian greg(mTime); + char str[15]; + if (14 != snprintf(str, 15, "%4.4d%2.2d%2.2d%2.2d%2.2d%2.2d", + int(greg.year), greg.month, greg.day, greg.hour, greg.minute, int(greg.second))) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + memcpy(dst, str, length == 14 ? 14 : 15); // null terminate if there's room +} + + +// +// Generalized parse-from-string setup +// +void CssmUniformDate::setFromString(const char *src, const char *format, size_t fieldWidth) +{ + // use a stack buffer + char str[20]; + assert(fieldWidth < sizeof(str)); + + // make a copy with proper null terminator + memcpy(str, src, fieldWidth); + str[fieldWidth] = '\0'; + + // parse (with limited checks for bad field formats) + long year; + int month, day, hour, minute; + double second; + if (6 != sscanf(str, format, + &year, &month, &day, &hour, &minute, &second)) + CssmError::throwMe(CSSM_ERRCODE_UNKNOWN_FORMAT); + + // success + mTime = Gregorian((int)year, month, day, hour, minute, second); +}