linux-packaging-mono/mcs/class/System/Mono.AppleTls/MonoCertificatePal.Mobile.cs
Xamarin Public Jenkins (auto-signing) e19d552987 Imported Upstream version 5.18.0.161
Former-commit-id: 4db48158d3a35497b8f118ab21b5f08ac3d86d98
2018-10-19 08:34:24 +00:00

189 lines
7.1 KiB
C#

//
// MonoCertificatePal.Mobile.cs
//
// Authors:
// Miguel de Icaza
// Sebastien Pouliot <sebastien@xamarin.com>
// Martin Baulig <mabaul@microsoft.com>
//
// Copyright 2010 Novell, Inc
// Copyright 2011-2014 Xamarin Inc.
// Copyright (c) 2018 Xamarin Inc. (http://www.xamarin.com)
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System;
using System.Threading;
using System.Runtime.InteropServices;
using System.Security.Cryptography.Apple;
using System.Security.Cryptography.X509Certificates;
using Microsoft.Win32.SafeHandles;
namespace Mono.AppleTls
{
static partial class MonoCertificatePal
{
static int initialized;
static CFString ImportExportPassphase;
static CFString ImportItemIdentity;
static IntPtr MatchLimitAll;
static IntPtr MatchLimitOne;
static IntPtr MatchLimit;
static IntPtr SecClassKey;
static IntPtr SecClassIdentity;
static IntPtr SecClassCertificate;
static IntPtr ReturnRef;
static IntPtr MatchSearchList;
static void Initialize ()
{
if (Interlocked.CompareExchange (ref initialized, 1, 0) != 0)
return;
var handle = CFObject.dlopen (SecurityLibrary, 0);
if (handle == IntPtr.Zero)
return;
try {
ImportExportPassphase = CFObject.GetStringConstant (handle, "kSecImportExportPassphrase");
ImportItemIdentity = CFObject.GetStringConstant (handle, "kSecImportItemIdentity");
MatchLimit = CFObject.GetIntPtr (handle, "kSecMatchLimit");
MatchLimitAll = CFObject.GetIntPtr (handle, "kSecMatchLimitAll");
MatchLimitOne = CFObject.GetIntPtr (handle, "kSecMatchLimitOne");
SecClassKey = CFObject.GetIntPtr (handle, "kSecClass");
SecClassIdentity = CFObject.GetIntPtr (handle, "kSecClassIdentity");
SecClassCertificate = CFObject.GetIntPtr (handle, "kSecClassCertificate");
ReturnRef = CFObject.GetIntPtr (handle, "kSecReturnRef");
MatchSearchList = CFObject.GetIntPtr (handle, "kSecMatchSearchList");
} finally {
CFObject.dlclose (handle);
}
}
static SafeSecIdentityHandle ImportIdentity (byte[] data, string password)
{
if (data == null)
throw new ArgumentNullException (nameof (data));
if (string.IsNullOrEmpty (password)) // SecPKCS12Import() doesn't allow empty passwords.
throw new ArgumentException (nameof (password));
Initialize ();
using (var pwstring = CFString.Create (password))
using (var optionDict = CFDictionary.FromObjectAndKey (pwstring.Handle, ImportExportPassphase.Handle)) {
var result = ImportPkcs12 (data, optionDict, out var array);
if (result != SecStatusCode.Success)
throw new InvalidOperationException (result.ToString ());
return new SafeSecIdentityHandle (array [0].GetValue (ImportItemIdentity.Handle));
}
}
[DllImport (SecurityLibrary)]
extern static SecStatusCode SecPKCS12Import (IntPtr pkcs12_data, IntPtr options, out IntPtr items);
static SecStatusCode ImportPkcs12 (byte[] buffer, CFDictionary options, out CFDictionary[] array)
{
using (CFData data = CFData.FromData (buffer)) {
return ImportPkcs12 (data, options, out array);
}
}
static SecStatusCode ImportPkcs12 (CFData data, CFDictionary options, out CFDictionary[] array)
{
if (options == null)
throw new ArgumentNullException (nameof (options));
var code = SecPKCS12Import (data.Handle, options.Handle, out var handle);
array = CFArray.ArrayFromHandle<CFDictionary> (handle, h => new CFDictionary (h, false));
if (handle != IntPtr.Zero)
CFObject.CFRelease (handle);
return code;
}
public static SafeSecIdentityHandle ImportIdentity (X509Certificate2 certificate)
{
if (certificate == null)
throw new ArgumentNullException (nameof (certificate));
if (!certificate.HasPrivateKey)
throw new InvalidOperationException ("Need X509Certificate2 with a private key.");
SafeSecIdentityHandle identity;
/*
* SecPSK12Import does not allow any empty passwords, so let's generate
* a semi-random one here.
*/
Initialize ();
var password = Guid.NewGuid ().ToString ();
var pkcs12 = certificate.Export (X509ContentType.Pfx, password);
identity = ImportIdentity (pkcs12, password);
return identity ?? new SafeSecIdentityHandle ();
}
[DllImport (SecurityLibrary)]
extern static SecStatusCode SecItemCopyMatching (/* CFDictionaryRef */ IntPtr query, /* CFTypeRef* */ out IntPtr result);
public static SafeSecIdentityHandle FindIdentity (SafeSecCertificateHandle certificate, bool throwOnError = false)
{
if (certificate == null || certificate.IsInvalid)
throw new ObjectDisposedException (nameof (certificate));
var identity = FindIdentity (cert => MonoCertificatePal.Equals (certificate, cert)) ?? new SafeSecIdentityHandle ();
if (!throwOnError || identity.IsInvalid)
return identity;
var subject = MonoCertificatePal.GetSubjectSummary (certificate);
throw new InvalidOperationException ($"Could not find SecIdentity for certificate '{subject}' in keychain.");
}
static SafeSecIdentityHandle FindIdentity (Predicate<SafeSecCertificateHandle> filter)
{
Initialize ();
/*
* Unfortunately, SecItemCopyMatching() does not allow any search
* filters when looking up an identity.
*
* The following lookup will return all identities from the keychain -
* we then need need to find the right one.
*/
using (var query = CFMutableDictionary.Create ()) {
query.SetValue (SecClassKey, SecClassIdentity);
query.SetValue (CFBoolean.True.Handle, ReturnRef);
query.SetValue (MatchLimitAll, MatchLimit);
var status = SecItemCopyMatching (query.Handle, out var ptr);
if (status != SecStatusCode.Success || ptr == IntPtr.Zero)
return null;
using (var array = new CFArray (ptr, false)) {
for (int i = 0; i < array.Count; i++) {
var item = array[i];
if (!MonoCertificatePal.IsSecIdentity (item))
throw new InvalidOperationException ();
using (var identity = new SafeSecIdentityHandle (item))
using (var certificate = MonoCertificatePal.GetCertificate (identity)) {
if (filter (certificate))
return new SafeSecIdentityHandle (item);
}
}
}
}
return null;
}
}
}