You've already forked UnrealEngineUWP
mirror of
https://github.com/izzy2lost/UnrealEngineUWP.git
synced 2026-03-26 18:15:20 -07:00
#lockdown Nick.Penwarden #rb none ========================== MAJOR FEATURES + CHANGES ========================== Change 3175510 on 2016/10/26 by Josh.Adams - New Wolf SDK support (11). - Added new input plugin now that extra NDA is lifted Change 3176629 on 2016/10/27 by Josh.Adams Merging //UE4/Dev-Main to Dev-Platform (//UE4/Dev-Platform) Change 3177232 on 2016/10/27 by Josh.Adams - Minor comment change Change 3177348 on 2016/10/27 by Dmitry.Rekman Linux: default to GL4. Change 3177523 on 2016/10/27 by Dmitry.Rekman Linux: update libc++ to 3.9 and add AArch64. Change 3178208 on 2016/10/28 by Daniel.Lamb Enable multithreaded lightmap encoding. Change 3178273 on 2016/10/28 by Luke.Thatcher [PLATFORM] [PS4] [!] Fix crash in PS4 packaging step. - Parallel-for accessing the same log files, causing IOException. Change 3178573 on 2016/10/28 by Dmitry.Rekman Linux: fix for projects not having proper version associations (UE-5954). - Fixed by CengizT. Change3180487on 2016/10/31 by Josh.Adams Moved new file to peoper spot Change 3180508 on 2016/10/31 by Josh.Adams - Fixed crash on audio free for Wolf Change 3181821 on 2016/11/01 by Josh.Adams - Fixed ShooterGame cooking after sync from main Change 3182469 on 2016/11/01 by Josh.Adams - test/shipping build wolf fixes Change 3183078 on 2016/11/02 by Josh.Adams - Added AllDesktop back in for Windows (File | Package) Change 3183229 on 2016/11/02 by Josh.Adams - Fixed wrong path in JunkManifest.txt Change 3184245 on 2016/11/02 by Dmitry.Rekman Linux: add AArch64 (ARM64) libs. Change 3184326 on 2016/11/02 by Dmitry.Rekman Linux: add more files for AArch64. Change 3184353 on 2016/11/02 by Dmitry.Rekman Linux: Add missed AArch64 libpng. Change 3184871 on 2016/11/03 by Luke.Thatcher [PLATFORM] [PS4] [!] Fix broken DownloadImage blueprint node on PS4. - Node should return a UTexture2DDynamic, otherwise the RHI assumes the data has been pre-formatted for the GPU, and we get pitch/layout issues. #jira UE-36365 Change 3185407 on 2016/11/03 by Dmitry.Rekman Linux: fix PhysX on AArch64. (Edigrating 3184484 from Wombat to Dev-Platform). Change 3187488 on 2016/11/04 by Josh.Adams Copying //Tasks/UE4/Private-Platform-Switch to Dev-Platform-Minimal (//UE4/Dev-Platform-Minimal) Change 3187740 on 2016/11/04 by Josh.Adams - Re-copying the Switch files, now with proper case in the directory names Change 3188304 on 2016/11/07 by Dan.Mahashin - Removed deprecated functions in NVN window creation Change3188865on 2016/11/07 by Luke.Thatcher [PLATFORM] [PS4] [~] Move PS4 console input handler into engine classes from OrionGame. - Enables console input from Sony's "Console Output" tool for all games, in debug/development builds. #jira UE-37672 Change 3189517 on 2016/11/07 by Jeff.Campeau Fix incorrect local platform identification in device manager. #jira UE-38312 Change 3189897 on 2016/11/08 by Luke.Thatcher [PLATFORM] [!] Fix width/height mismatch in DownloadImage blueprint node. Change 3190042 on 2016/11/08 by Josh.Adams - Fixed default and Shooter App Ids for Switch Change 3190181 on 2016/11/08 by Joe.Barnes [UE-37275] Split reflection capture error message into two UE_LOG()s. Line length causes truncation and line wrap on some platforms. Change 3190185 on 2016/11/08 by Joe.Barnes Fix another instance of UE_LOG() where the string was being truncated on Switch platform. Change 3190272 on 2016/11/08 by Daniel.Lamb Add file hashes to depependency tracking info. Moved partial gc controlling code outside of the cooker. Store cooked file hashes in cooked asset registry. Cooked asset registry is now part of the cooker instead of chunking manifest. #test cook paragon Change 3190332 on 2016/11/08 by Omar.Rodriguez Fixing issues with iOS remote notifications * Update UPlatformGameInstance::FPlatformRegisteredForRemoteNotificationsDelegate signature so the parameter is const& which will work with BlueprintAssignable * Fix misspelling when doing respondsToSelector check * Update generated Xcode project to use the generated entitlements file * Add remote-notification as a background mode * Update the generated entitlements file contents to include APS environment for push notifications * Added bEnableRemoteNotificationsSupport ini parameter to control whether iOS push notifications code is compiled Change 3190391 on 2016/11/08 by Brent.Pease UE-31739 - Crash when Deploying to iPad Air with BC4 Texture Compression Setting (Josh's suggestion worked out of the box) Change 3190786 on 2016/11/08 by Bart.Hawthorne Fix some missing PLATFORM_WOLF changes to PLATFORM_SWITCH in ShooterGame Change 3190902 on 2016/11/08 by Alicia.Cano Allow RTTI and exceptions to be enabled for Android #jira UE-37845 #android Change 3190973 on 2016/11/08 by Chris.Babcock Add ability to set element value field with new text parameter for UPL #jira UE-37390 #PR #2869 #ue4 #upl Change 3191411 on 2016/11/09 by Josh.Stoddard Warn when user tries to use a shared pak reader on the wrong thread #jira UE-38049 Change 3191635 on 2016/11/09 by Josh.Stoddard More useful message during cook when AT9 assets fail to encode using SCE's tool #jira UE-38053 Change 3191663 on 2016/11/09 by Peter.Sauerbrei fix for ios build from PC Change 3191701 on 2016/11/09 by Brent.Pease implement iOS device logs on windows Change 3191794 on 2016/11/09 by Daniel.Lamb Fixed up compile error missing header file. #test Compile editor #jira UE-38414 Change 3191807 on 2016/11/09 by Josh.Adams - Fixed one chage that was missed in the WolfPlat->Switch rename Change 3191867 on 2016/11/09 by Josh.Adams - Enabled Switch for ShooterGame project Change 3191958 on 2016/11/09 by Jeff.Campeau Add warning for anyone still using XP Change 3192185 on 2016/11/09 by Josh.Adams - Updated to SDK 0.11.12 - Added TrackLotCheckItem API to track guidelines with limits (nothing using it yet) Change 3192241 on 2016/11/09 by Josh.Adams Merging //UE4/Dev-Main to Dev-Platform (//UE4/Dev-Platform) Change 3192324 on 2016/11/09 by Josh.Adams - Worked around an issue with RunOnTarget stripping quotes around paths with spaces #jira UE-38388 Change 3192387 on 2016/11/09 by Josh.Adams - Updating editor icon for Switch #jira UE-38295 Change 3192476 on 2016/11/09 by Dmitry.Rekman Linux: put correct OpenAL lib per architecture. (Edigrating CL 3185947 from Wombat to Dev-Platform) Change 3192527 on 2016/11/09 by Josh.Adams - Fixed a shadow variable warning #jira UE-38408 Change 3192606 on 2016/11/09 by Jeff.Campeau XP option removed #jira UEPLAT-1542 Change 3192644 on 2016/11/09 by Josh.Adams - Fixed a CIS error (not sure why I don't get it, but hey) Change 3192659 on 2016/11/09 by Josh.Adams - Fixed a crash in DeploymentServer Change 3192672 on 2016/11/09 by Jeff.Campeau Fix WinXP message spamming Change 3193252 on 2016/11/10 by Josh.Adams - Remove assertion in SwitchTextureFormat when the SDK can't be found (if you are sharing DLLs) Change 3193756 on 2016/11/10 by Dmitry.Rekman Linux: add support for touch events. (Edigrating CL 3188159 from Wombat to Dev-Platform). Change 3194297 on 2016/11/10 by Jeff.Campeau HarfBuzz implementation for Xbox One #jira UE-28590 Change 3194299 on 2016/11/10 by Jeff.Campeau Pump Xbox One messaging during slow startup tasks #jira UEPLAT-1276 Change 3194300 on 2016/11/10 by Jeff.Campeau Use response files when building for Xbox One #jira UEPLAT-1296 Change 3194313 on 2016/11/11 by Jeff.Campeau Stop uploading symbols on the first error Show a more detailed error message if symbol uploading fails Add a command line option to disable upload of symbols #1852 #jira UE-24425 Change 3194327 on 2016/11/11 by Jeff.Campeau Deduplicate Xbox One build.cs setup for several modules #jira UE-37540 Change 3194402 on 2016/11/11 by Dmitry.Rekman Linux: do not apply mouse workaround unnecessarily. - Only matters when there is more than one window. (Edigrating 3194399 from Wombat to Dev-Platform). Change 3194434 on 2016/11/11 by Dan.Mahashin - Ported fix CL 3193690: Add workaround to file I/O error about ResultInvalidCurrentMemory when reloading levels - remove uncached attribute during memory pool finalization Change 3194569 on 2016/11/11 by Daniel.Lamb Fixed issue with CreateLinker failing to return LinkerLoad but creating a UPackage which can't be cleaned up. Change 3194570 on 2016/11/11 by Daniel.Lamb Fix for DiffSerializeArchive not using the correct archive when saving packages. #test Cook paragon Change 3194571 on 2016/11/11 by Daniel.Lamb Make sure dependent packages are valid before using them. Added FastBuildCookRun bat file for paragon testing. #test Cook Paragon Change 3194575 on 2016/11/11 by Daniel.Lamb Reworked a warning for the cooker. Change 3194698 on 2016/11/11 by Daniel.Lamb Skip skin verify only runs on build machines now. Saves paragon cook time. Change 3194699 on 2016/11/11 by Daniel.Lamb Changed the wording of skip editor content setting so it's more clear. #test none Change 3194702 on 2016/11/11 by Daniel.Lamb Potential fix for default materials not being in chunk zero. #test run ps4 cooked build paragon Change3194711on 2016/11/11 by Alicia.Cano Allow RTTI and exceptions to be enabled for Android Allow RTTI to be enabled for IOS, Mac #jira UE-37845, UE-20314 #android #ios #mac Change 3194956 on 2016/11/11 by Josh.Adams - Removed the crash with unknown socket error code, left in the warning Change 3195028 on 2016/11/11 by Dmitry.Rekman Linux: repair launch on. (Edigrating 3194384 from //UE4/Private-Wombat/... to //UE4/Dev-Platform/...) Change 3195041 on 2016/11/11 by Dmitry.Rekman Linux: support selecting architectures per project. (Edigrating 3192783 from Wombat to Dev-Platform). Change 3195058 on 2016/11/11 by Dmitry.Rekman Linux: fix code to determine number of cores. - ARM topology seems not to be in line with the assumptions made by x86-centric code. (Merging 3184632 from Wombat to Dev-Platform). Change 3195082 on 2016/11/11 by Josh.Adams - Fixed name of packaged Switch builds to have the config in it (Shipping, etc) #jira UE-38394 Change 3195151 on 2016/11/11 by Bart.Hawthorne - Add game server settings to project settings to connect to the actual game server, instead of the debug login - Use the system software dialog box to show error codes for login failures Change 3195153 on 2016/11/11 by Josh.Adams - Fixed copy and paste logs errors Change 3195156 on 2016/11/11 by Josh.Adams - Fixed some default values, especially for save games (uses their default of 4MB size) - Added some LotCheck write tracking Change 3195285 on 2016/11/11 by Jeff.Campeau Fix HarfBuzz warning on Xbox One Change 3195477 on 2016/11/11 by Josh.Adams - Fixed up some IsGameOnly calls #jira UE-37575 Change 3195490 on 2016/11/11 by Dmitry.Rekman UAT: fix CIS (removed old variables). Change 3195724 on 2016/11/11 by Josh.Adams - Final fix for name of .nsp (content only projects in Shipping config, etc) #jira UE-38394 Change 3195755 on 2016/11/11 by Josh.Adams - Made translucent Switch icons Change 3195771 on 2016/11/11 by Josh.Adams - Fixed some Switch "space in path" issues #jira UE-38393 Change 3195801 on 2016/11/11 by Josh.Adams - Handle making sure the save is completed before we shutdown #jira UE-38215 Change 3196593 on 2016/11/14 by Michael.Trepka Implemented Info string in AvfMedia for display in Media Player Editor #jira UE-35386 Change 3196782 on 2016/11/14 by Josh.Adams - Added a comment for a workaround Change 3196784 on 2016/11/14 by Michael.Trepka Alembic importer for Mac #jira UE-37708 Change 3196901 on 2016/11/14 by Alicia.Cano ADB over wifi fails to deploy on Launch on. #jira UE-37957 #android Change 3197055 on 2016/11/14 by Josh.Adams - Fixed BinnedAllocator crash that happened with PoisonProxy and large allocations with large alignment Change 3197059 on 2016/11/14 by Josh.Adams - Removed some stat code with no STATS Change 3197066 on 2016/11/14 by Josh.Adams - Fixed the generic growableallocator to not free metadata before it's used for stats, and cleaned up a couple minor things Change 3197176 on 2016/11/14 by Josh.Adams - Added some helper scripts to switch in and out of debug mode on Switch Change 3197183 on 2016/11/14 by Bart.Hawthorne Error dialog fixes based on peer review feedback from JoshA Change 3197339 on 2016/11/14 by Josh.Adams Allow -htcs on the commandline now to override disabling Htcs in packaged builds Change 3197401 on 2016/11/14 by Josh.Adams - Fixed the Switch package installation script to remove the path of the package, since it causes problems with spaces, and also it makes the script less portable! #jira UE-38556 Change 3197691 on 2016/11/14 by Dmitry.Rekman Linux: save added devices. (Edigrating 3196529 from Wombat to Dev-Platform). Change 3197854 on 2016/11/15 by Dan.Mahashin - MemoryProfiler2: fixed Switch parser file path in the csproj Change 3197960 on 2016/11/15 by Dan.Mahashin - NVN RHITransitionResources() directly uses a barrier instead of relying on CopyToResolveTarget() side effect (UE-33834) Change 3198488 on 2016/11/15 by Bart.Hawthorne Submit missing NoRedist/DefaultEngine.ini file Change 3198970 on 2016/11/15 by Michael.Trepka Don't try to use installed Mono 4.6 on Mac as it's known to have issues on macOS 10.12 (for example building the editor with UBT often fails with Mono running out of file desriptors) Change 3199050 on 2016/11/15 by Daniel.Lamb Some more output to help track down iterative cooking scenarios #test Cook paragon Change 3199097 on 2016/11/15 by Josh.Adams - Fixed up Switch packaging to re-generate the meta data in case it changed since compile time (esp with content only projects - Fixed default Program Id in code - Fixed a problem with Run with a space in the path #jira UE-38608 Change 3199181 on 2016/11/15 by Dmitry.Rekman Fix CIS (compiling LinuxTargetDevice without engine). Change 3199253 on 2016/11/15 by Dmitry.Rekman Hopeful fix for a static analysis warning. Change 3199325 on 2016/11/15 by Joe.Barnes Start a new CommandBuffer immediately upon ending one. Prevents fetching when there's no CommandBuffer. Needed for Loading Screen movie playback. Change 3199814 on 2016/11/15 by Dmitry.Rekman Linux: remove forced -windowed when launching. (Merging CL 3199789 from Wombat to Dev-Platform) Change 3200580 on 2016/11/16 by Josh.Adams Updasted DeploymentServer Change 3200595 on 2016/11/16 by Joe.Barnes Removed inadvertent SleepThread() when starting movie playback. Change 3200604 on 2016/11/16 by Josh.Adams - Added NN_MIDDLEWARE macros to tag ths apps as using UE4 middleware Change 3200632 on 2016/11/16 by Brent.Pease Update PlatformShowcase with latest tests Change 3200704 on 2016/11/16 by Dmitry.Rekman Linux: fix native compilation. Change 3200711 on 2016/11/16 by Brent.Pease - Support ios audio streaming from disk - Flushed out ADPCMAudioInfo to be more flexible with buffer management in addition to support streaming from disk. This should make more code platform independent. + Other platforms should work fine but will need to be updated to use the new buffer flexability (and hence simplify their own code and buffer management) - IOS audio implementation simplified to use new ADPCMAudioInfo functionality - Fixed adpcm seamless looping NOTE: While everything works with my testing (admittedly simple tests) a little more code cleanup needs to happen... Change 3201015 on 2016/11/16 by Josh.Adams Merging //UE4/Dev-Main to Dev-Platform (//UE4/Dev-Platform) Change 3201023 on 2016/11/16 by Josh.Stoddard Fix splash screen assignment for iPad #jira UE-38623 Change 3201215 on 2016/11/16 by Brent.Pease Hopefully final fix for build breakage Change 3201259 on 2016/11/16 by Josh.Adams - Removed the clock rate settings from the Project Settings, and the cvars - it was just confusing - Further improved the metadata recreation during packaging (can get rid of the temp meta/desc files now I believe) - Reduced audio pool alignment to not waste massive memory from it Change3202332on 2016/11/17 by Daniel.Lamb Changed build scripts to support iterative cooking #test Ran new build scripts Change 3202371 on 2016/11/17 by Michael.Trepka Changed FAppleHttpResponse::GetContentLength to return expected content size instead of payload size so it's consistent with other implementations #jira UE-38392 Change 3202421 on 2016/11/17 by Michael.Trepka Decrease the number of max open files for a thread on Apple platforms from 256 to 192 to leave more file descriptors to Cocoa #jira UE-18343 Change 3202462 on 2016/11/17 by Michael.Trepka Fixed HTTP If-None-Match response code on Mac and iOS Fixed by iktomi, https://answers.unrealengine.com/questions/492514/http-if-none-match-logic-isnt-working-on-mac-due-t.html #jira UE-36317 Change 3202620 on 2016/11/17 by Daniel.Lamb Fixed issue with some objects being garbage collected which shouldn't be because the collection handler didn't get registered. Commandlets now do not always have GIsRequestingExit true. Made crash handler check for commandlets running and exit appropriately. #test Rebuild lighting QAGame Change 3202955 on 2016/11/17 by Daniel.Lamb Add support for clearing all the cached cooked platform data for a platform when requested. #test cook QA game #jira UE-38361 Change 3202983 on 2016/11/17 by Daniel.Lamb Added support to rebuild lightmaps commandlet for building lightmaps in seperate files. #test rebuild lighting Custom QAGame maps. #jira OR-31907 Change 3203128 on 2016/11/17 by Josh.Adams - Fixed split screen user selection in ShooterGame (brought over some changes from NickD for it as well) Change 3203537 on 2016/11/18 by Dmitry.Rekman Fix ProjectWorldToScreen node for letterboxed viewports. (Merging CL 3201546 from Wombat to Dev-Platform). Change 3203540 on 2016/11/18 by Dmitry.Rekman Linux: be more verbose when setting vblank sync. (Merging CL 3199633 from Private-Wombat to Dev-Platform). Change 3203599 on 2016/11/18 by Dmitry.Rekman Speedup bForceCompilationAtStartup=True when nothing changed (UE-37067). - PR #2849: Contributed by slonopotamus. Change 3203610 on 2016/11/18 by Dmitry.Rekman Add CEF support for Linux (UE-6743). Change 3203615 on 2016/11/18 by Dmitry.Rekman Linux: fix bootstrap script so it is independent of working dir (UE-37016). - PR #2842 contributed by slonopotamus Change 3203645 on 2016/11/18 by Dmitry.Rekman Linux: fix UnrealCEFSubProcess. Change 3203658 on 2016/11/18 by Dmitry.Rekman Remove hard-coded paths to mono binary (UE-35228). - Another way to implement pull request #2741. Change 3203770 on 2016/11/18 by Josh.Adams - Brought over some changes from Dev-Core to not crash in AsyncLoading with debug code Change 3204244 on 2016/11/18 by Dmitry.Rekman Unsuppress mistakenly suppressed warnings and fix one more (UE-38788). Change 3204277 on 2016/11/18 by Brent.Pease + Fix seamless looping bug found on Dan's QAGame test + Fix static analyzer warning (which was a real bug with uncompressed streaming) + Code review feedback from Aaron + Small addition from channel sync ios bug fix Change 3204576 on 2016/11/18 by Omar.Rodriguez Expose the bEnableRemoteNotificationsSupport ini setting in the iOS project settings. Change 3204629 on 2016/11/18 by Chris.Babcock Fix case of VulkanSwapChain.h #include #jira UE-38843 #ue4 #vulkan Change 3204708 on 2016/11/18 by Josh.Adams - Set SwitchMoviePlayer to include the libs from the proper directory Change 3204730 on 2016/11/18 by Josh.Adams - Changed a check to a checkf to narrow down why FMaterialUniformExpressionType::GetTypeMap().FindRef(TypeName) is returning nullptr on tvOS Change 3204865 on 2016/11/18 by Brent.Pease + Turn off ios console logs on Windows to help sort through ios packaging and launch-on issues - This is NOT a fix but it should make it easier to track down the problem with it off. Change 3204883 on 2016/11/18 by Dmitry.Rekman Linux: fix native LaunchOn (UE-38616). Change 3204914 on 2016/11/18 by Brent.Pease + Turn off the device check to prevent it from conflicting with remote packaging/launch-on Change 3204940 on 2016/11/18 by Josh.Adams Backing out changes to the profiler for Switch. Shouldn't have checked it in today during smoke Change 3204952 on 2016/11/18 by Dmitry.Rekman Linux: fix bootstrap script (UE-38851). - Caused by UE-37016. Change 3205630 on 2016/11/21 by Brent.Pease + Fix audio sound queuing bug by ensuring audio buffers are not reused by different sound source objects. + Cleaned up the locking mechanism around stopping sound sources to make its intent and function are clear + Cleaned up memory tracking and freeing. #jira ue-38846 Change 3205787 on 2016/11/21 by Josh.Adams Merging //UE4/Dev-Main to Dev-Platform (//UE4/Dev-Platform) [CL 3206922 by Josh Adams in Main branch]
1537 lines
53 KiB
C#
1537 lines
53 KiB
C#
// Software License Agreement (BSD License)
|
|
//
|
|
// Copyright (c) 2007, Peter Dennis Bartok <PeterDennisBartok@gmail.com>
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use of this software in source and binary forms, with or without modification, are
|
|
// permitted provided that the following conditions are met:
|
|
//
|
|
// * Redistributions of source code must retain the above
|
|
// copyright notice, this list of conditions and the
|
|
// following disclaimer.
|
|
//
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the
|
|
// following disclaimer in the documentation and/or other
|
|
// materials provided with the distribution.
|
|
//
|
|
// * Neither the name of Peter Dennis Bartok nor the names of its
|
|
// contributors may be used to endorse or promote products
|
|
// derived from this software without specific prior
|
|
// written permission of Yahoo! Inc.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
|
|
// WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
// PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
// ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
|
|
// TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
//
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.IO;
|
|
using System.Runtime.InteropServices;
|
|
using System.Text;
|
|
using System.Diagnostics;
|
|
using MobileDeviceInterface;
|
|
|
|
namespace Manzana
|
|
{
|
|
public class MobileDeviceInstanceManager
|
|
{
|
|
/// <summary>
|
|
/// Registered device notification callback
|
|
/// </summary>
|
|
private static DeviceNotificationCallback DeviceCallbackHandle;
|
|
|
|
/// <summary>
|
|
/// The <c>Connect</c> event is triggered when a iPhone is connected to the computer
|
|
/// </summary>
|
|
public static event ConnectEventHandler ConnectEH;
|
|
|
|
/// <summary>
|
|
/// The <c>Disconnect</c> event is triggered when the iPhone is disconnected from the computer
|
|
/// </summary>
|
|
public static event ConnectEventHandler DisconnectEH;
|
|
|
|
/// <summary>
|
|
/// List of connected devices (device ptr -> device instance)
|
|
/// </summary>
|
|
public static Dictionary<TypedPtr<AppleMobileDeviceConnection>, MobileDeviceInstance> ConnectedDevices = new Dictionary<TypedPtr<AppleMobileDeviceConnection>, MobileDeviceInstance>();
|
|
|
|
public static IEnumerable<MobileDeviceInstance> GetSnapshotInstanceList()
|
|
{
|
|
// Clone a copy to prevent problems from delayed enumeration
|
|
List<MobileDeviceInstance> Result = new List<MobileDeviceInstance>();
|
|
Result.AddRange(ConnectedDevices.Values);
|
|
|
|
return Result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns true if any devices are currently connected
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public static bool AreAnyDevicesConnected()
|
|
{
|
|
lock (ConnectedDevices)
|
|
{
|
|
return ConnectedDevices.Count > 0;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initialize the mobile device manager, which handles discovery of connected Apple mobile devices
|
|
/// </summary>
|
|
/// <param name="myConnectHandler"></param>
|
|
/// <param name="myDisconnectHandler"></param>
|
|
public static void Initialize(ConnectEventHandler myConnectHandler, ConnectEventHandler myDisconnectHandler)
|
|
{
|
|
if (MobileDevice.DeviceImpl == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
ConnectEH += myConnectHandler;
|
|
DisconnectEH += myDisconnectHandler;
|
|
|
|
DeviceCallbackHandle = new DeviceNotificationCallback(NotifyCallback);
|
|
|
|
int ret = MobileDevice.DeviceImpl.NotificationSubscribe(DeviceCallbackHandle);
|
|
if (ret != 0)
|
|
{
|
|
throw new Exception("AMDeviceNotificationSubscribe failed with error " + ret);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Raises the <see>Connect</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="ConnectEventArgs"/> that contains the event data.</param>
|
|
protected static void OnConnect(ConnectEventArgs args)
|
|
{
|
|
ConnectEventHandler handler = ConnectEH;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(null, args);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Raises the <see>Disconnect</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="ConnectEventArgs"/> that contains the event data.</param>
|
|
protected static void OnDisconnect(ConnectEventArgs args)
|
|
{
|
|
ConnectEventHandler handler = DisconnectEH;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(null, args);
|
|
}
|
|
}
|
|
|
|
private static void NotifyCallback(ref AMDeviceNotificationCallbackInfo callback)
|
|
{
|
|
if (callback.msg == NotificationMessage.Connected)
|
|
{
|
|
MobileDeviceInstance Inst;
|
|
if (ConnectedDevices.TryGetValue(callback.dev, out Inst))
|
|
{
|
|
// Already connected, not sure why we got another message...
|
|
}
|
|
else
|
|
{
|
|
Inst = new MobileDeviceInstance(callback.dev);
|
|
ConnectedDevices.Add(callback.dev, Inst);
|
|
}
|
|
|
|
if (Inst.ConnectToPhone())
|
|
{
|
|
OnConnect(new ConnectEventArgs(callback));
|
|
}
|
|
}
|
|
else if (callback.msg == NotificationMessage.Disconnected)
|
|
{
|
|
MobileDeviceInstance Inst;
|
|
if (ConnectedDevices.TryGetValue(callback.dev, out Inst))
|
|
{
|
|
Inst.connected = false;
|
|
OnDisconnect(new ConnectEventArgs(callback));
|
|
ConnectedDevices.Remove(callback.dev);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Exposes access to a mobile device running iOS
|
|
/// </summary>
|
|
public class MobileDeviceInstance
|
|
{
|
|
#region Locals
|
|
|
|
private DeviceRestoreNotificationCallback drn1;
|
|
private DeviceRestoreNotificationCallback drn2;
|
|
private DeviceRestoreNotificationCallback drn3;
|
|
private DeviceRestoreNotificationCallback drn4;
|
|
|
|
internal TypedPtr<AppleMobileDeviceConnection> iPhoneHandle;
|
|
internal TypedPtr<AFCCommConnection> AFCCommsHandle;
|
|
internal IntPtr hService;
|
|
internal IntPtr hSyslogService;
|
|
internal IntPtr hInstallService;
|
|
public bool connected;
|
|
private string current_directory;
|
|
#endregion // Locals
|
|
|
|
#region Constructors
|
|
/// <summary>
|
|
/// Initializes a new iPhone object.
|
|
/// </summary>
|
|
private void doConstruction()
|
|
{
|
|
drn1 = new DeviceRestoreNotificationCallback(DfuConnectCallback);
|
|
drn2 = new DeviceRestoreNotificationCallback(RecoveryConnectCallback);
|
|
drn3 = new DeviceRestoreNotificationCallback(DfuDisconnectCallback);
|
|
drn4 = new DeviceRestoreNotificationCallback(RecoveryDisconnectCallback);
|
|
|
|
int ret = MobileDevice.DeviceImpl.RestoreRegisterForDeviceNotifications(drn1, drn2, drn3, drn4, 0, IntPtr.Zero);
|
|
if (ret != 0)
|
|
{
|
|
throw new Exception("AMRestoreRegisterForDeviceNotifications failed with error " + ret);
|
|
}
|
|
current_directory = "/";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new iPhone object. If an iPhone is connected to the computer, a connection will automatically be opened.
|
|
/// </summary>
|
|
public MobileDeviceInstance(TypedPtr<AppleMobileDeviceConnection> Connection)
|
|
{
|
|
iPhoneHandle = Connection;
|
|
doConstruction();
|
|
}
|
|
#endregion // Constructors
|
|
|
|
#region Properties
|
|
/// <summary>
|
|
/// Gets the current activation state of the phone
|
|
/// </summary>
|
|
public string ActivationState
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "ActivationState");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns true if an iPhone is connected to the computer
|
|
/// </summary>
|
|
public bool IsConnected
|
|
{
|
|
get
|
|
{
|
|
return connected;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the Device information about the connected iPhone
|
|
/// </summary>
|
|
public TypedPtr<AppleMobileDeviceConnection> Device
|
|
{
|
|
get
|
|
{
|
|
return iPhoneHandle;
|
|
}
|
|
}
|
|
|
|
///<summary>
|
|
/// Returns the 40-character UUID of the device
|
|
///</summary>
|
|
public string DeviceId
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "UniqueDeviceID");
|
|
}
|
|
}
|
|
|
|
///<summary>
|
|
/// Returns the type of the device, should be either 'iPhone' or 'iPod'.
|
|
///</summary>
|
|
public string DeviceType
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "DeviceClass");
|
|
}
|
|
}
|
|
|
|
///<summary>
|
|
/// Returns the current OS version running on the device (2.0, 2.2, 3.0, 3.1, etc).
|
|
///</summary>
|
|
public string DeviceVersion
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "ProductVersion");
|
|
}
|
|
}
|
|
///<summary>
|
|
/// Returns the name of the device, like "Dan's iPhone"
|
|
///</summary>
|
|
public string DeviceName
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "DeviceName");
|
|
}
|
|
}
|
|
|
|
///<summary>
|
|
/// Returns the model number of the device, like "MA712"
|
|
///</summary>
|
|
public string ModelNumber
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "ModelNumber");
|
|
}
|
|
}
|
|
///<summary>
|
|
/// Returns the product type of the device, like "iPhone1,1"
|
|
///</summary>
|
|
public string ProductType
|
|
{
|
|
get
|
|
{
|
|
return MobileDevice.AMDeviceCopyValue(iPhoneHandle, "ProductType");
|
|
}
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Returns the handle to the iPhone com.apple.afc service
|
|
/// </summary>
|
|
public TypedPtr<AFCCommConnection> AFCHandle
|
|
{
|
|
get
|
|
{
|
|
return AFCCommsHandle;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets/Sets the current working directory, used by all file and directory methods
|
|
/// </summary>
|
|
public string CurrentDirectory
|
|
{
|
|
get
|
|
{
|
|
return current_directory;
|
|
}
|
|
|
|
set
|
|
{
|
|
string new_path = FullPath(current_directory, value);
|
|
if (!IsDirectory(new_path))
|
|
{
|
|
throw new Exception("Invalid directory specified");
|
|
}
|
|
current_directory = new_path;
|
|
}
|
|
}
|
|
#endregion // Properties
|
|
|
|
#region Events
|
|
|
|
/// <summary>
|
|
/// Write Me
|
|
/// </summary>
|
|
public event EventHandler DfuConnect;
|
|
|
|
/// <summary>
|
|
/// Raises the <see>DfuConnect</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="DeviceNotificationEventArgs"/> that contains the event data.</param>
|
|
protected void OnDfuConnect(DeviceNotificationEventArgs args)
|
|
{
|
|
EventHandler handler = DfuConnect;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(this, args);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Write Me
|
|
/// </summary>
|
|
public event EventHandler DfuDisconnect;
|
|
|
|
/// <summary>
|
|
/// Raises the <see>DfiDisconnect</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="DeviceNotificationEventArgs"/> that contains the event data.</param>
|
|
protected void OnDfuDisconnect(DeviceNotificationEventArgs args)
|
|
{
|
|
EventHandler handler = DfuDisconnect;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(this, args);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// The RecoveryModeEnter event is triggered when the attached iPhone enters Recovery Mode
|
|
/// </summary>
|
|
public event EventHandler RecoveryModeEnter;
|
|
|
|
/// <summary>
|
|
/// Raises the <see>RecoveryModeEnter</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="DeviceNotificationEventArgs"/> that contains the event data.</param>
|
|
protected void OnRecoveryModeEnter(DeviceNotificationEventArgs args)
|
|
{
|
|
EventHandler handler = RecoveryModeEnter;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(this, args);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// The RecoveryModeLeave event is triggered when the attached iPhone leaves Recovery Mode
|
|
/// </summary>
|
|
public event EventHandler RecoveryModeLeave;
|
|
|
|
/// <summary>
|
|
/// Raises the <see>RecoveryModeLeave</see> event.
|
|
/// </summary>
|
|
/// <param name="args">A <see cref="DeviceNotificationEventArgs"/> that contains the event data.</param>
|
|
protected void OnRecoveryModeLeave(DeviceNotificationEventArgs args)
|
|
{
|
|
EventHandler handler = RecoveryModeLeave;
|
|
|
|
if (handler != null)
|
|
{
|
|
handler(this, args);
|
|
}
|
|
}
|
|
|
|
#endregion // Events
|
|
|
|
#region Filesystem
|
|
|
|
|
|
/// <summary>
|
|
/// Sanitizes a filename for use on PC (just the filename, not a full path)
|
|
/// </summary>
|
|
static public string SanitizeFilename(string InputFilename)
|
|
{
|
|
char[] Filename = InputFilename.ToCharArray();
|
|
char[] BadChars = Path.GetInvalidFileNameChars();
|
|
for (int i = 0; i < Filename.Length; ++i)
|
|
{
|
|
foreach (char BadChar in BadChars)
|
|
{
|
|
if (Filename[i] == BadChar)
|
|
{
|
|
Filename[i] = '_';
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return new string(Filename);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sanitizes a path for use on PC (just the path, no filename)
|
|
/// </summary>
|
|
static public string SanitizePathNoFilename(string InputPath)
|
|
{
|
|
char[] DirectoryName = InputPath.ToCharArray();
|
|
char[] BadChars = Path.GetInvalidPathChars();
|
|
for (int i = 0; i < DirectoryName.Length; ++i)
|
|
{
|
|
foreach (char BadChar in BadChars)
|
|
{
|
|
if ((DirectoryName[i] == BadChar) || (DirectoryName[i] == ':'))
|
|
{
|
|
DirectoryName[i] = '_';
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return new string(DirectoryName);
|
|
}
|
|
|
|
void RecursiveBackup(string SourceFolderOnDevice, string TargetFolderOnPC)
|
|
{
|
|
string[] Directories = GetDirectories(SourceFolderOnDevice);
|
|
foreach (string Directory in Directories)
|
|
{
|
|
string NewSourceFolder = SourceFolderOnDevice + Directory + "/";
|
|
string NewTargetFolder = Path.Combine(TargetFolderOnPC, SanitizePathNoFilename(Directory));
|
|
|
|
RecursiveBackup(NewSourceFolder, NewTargetFolder);
|
|
}
|
|
|
|
string[] Filenames = GetFiles(SourceFolderOnDevice);
|
|
foreach (string Filename in Filenames)
|
|
{
|
|
string SourceFilename = SourceFolderOnDevice + Filename;
|
|
string DestFilename = Path.Combine(TargetFolderOnPC, SanitizeFilename(Filename));
|
|
WriteProgressLine("Copying '{0}' -> '{1}' ...", 0, SourceFilename, DestFilename);
|
|
CopyFileFromPhone(DestFilename, SourceFilename, 1024 * 1024);
|
|
}
|
|
}
|
|
|
|
void RecursiveCopy(string SourceFolderOnPC, string TargetFolderOnDevice)
|
|
{
|
|
string[] Directories = System.IO.Directory.GetDirectories(SourceFolderOnPC);
|
|
foreach (string Directory in Directories)
|
|
{
|
|
string NewSourceFolder = Directory;
|
|
string NewTargetFolder = TargetFolderOnDevice + Directory.Substring(Directory.LastIndexOf(Path.DirectorySeparatorChar)+1) + "/";
|
|
|
|
WriteProgressLine("Copying folder {0} to {1}", 0, NewSourceFolder, NewTargetFolder);
|
|
RecursiveCopy(NewSourceFolder, NewTargetFolder);
|
|
}
|
|
|
|
string[] Filenames = System.IO.Directory.GetFiles(SourceFolderOnPC);
|
|
foreach (string Filename in Filenames)
|
|
{
|
|
string SourceFilename = Filename;
|
|
string DestFilename = TargetFolderOnDevice + Path.GetFileName(Filename);
|
|
WriteProgressLine("Copying '{0}' -> '{1}' ...", 0, SourceFilename, DestFilename);
|
|
CopyFileToPhone(SourceFilename, DestFilename, 1024 * 1024);
|
|
}
|
|
}
|
|
|
|
public void DumpInstalledApplications()
|
|
{
|
|
Dictionary<string, object> AppBundles;
|
|
MobileDevice.DeviceImpl.LookupApplications(iPhoneHandle, IntPtr.Zero, out AppBundles);
|
|
|
|
foreach (var Bundle in AppBundles)
|
|
{
|
|
WriteProgressLine(String.Format("Application bundle {0} has the following pairs:", Bundle.Key), 0);
|
|
|
|
Dictionary<object, object> BundlePairs = (Dictionary<object, object>)Bundle.Value;
|
|
foreach (var KVP in BundlePairs)
|
|
{
|
|
WriteProgressLine(String.Format(" {0} -> {1}", KVP.Key, KVP.Value), 0);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to back up all of the files on a phone in a particular directory to the PC
|
|
/// (requires the bundle identifier to be able to mount that directory)
|
|
/// </summary>
|
|
public bool TryBackup(string BundleIdentifier, string SourceFolderOnDevice, string TargetFolderOnPC)
|
|
{
|
|
if (ConnectToBundle(BundleIdentifier))
|
|
{
|
|
WriteProgressLine("Connected to bundle '{0}'", 0, BundleIdentifier);
|
|
|
|
try
|
|
{
|
|
RecursiveBackup(SourceFolderOnDevice, TargetFolderOnPC);
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
WriteProgressLine("Failed to transfer a file, extended error is '{0}'", 100, ex.Message);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
WriteProgressLine("Error: Failed to connect to bundle '{0}'", 100, BundleIdentifier);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to copy all of the files in a particular directory on the PC to the phone directory
|
|
/// (requires the bundle identifier to be able to mount that directory)
|
|
/// </summary>
|
|
public bool TryCopy(string BundleIdentifier, string SourceFolderOnPC, string TargetFolderOnDevice)
|
|
{
|
|
if (ConnectToBundle(BundleIdentifier))
|
|
{
|
|
WriteProgressLine("Connected to bundle '{0}'", 0, BundleIdentifier);
|
|
|
|
try
|
|
{
|
|
RecursiveCopy(SourceFolderOnPC, TargetFolderOnDevice);
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
WriteProgressLine("Failed to transfer a file, extended error is '{0}'", 100, ex.Message);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
WriteProgressLine("Error: Failed to connect to bundle '{0}'", 100, BundleIdentifier);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to copy all of the files in the manifest on the PC to the documents directory
|
|
/// (requires the bundle identifier to be able to mount that directory)
|
|
/// </summary>
|
|
public bool TryCopy(string BundleIdentifier, string Manifest)
|
|
{
|
|
if (ConnectToBundle(BundleIdentifier))
|
|
{
|
|
WriteProgressLine("Connected to bundle '{0}'", 0, BundleIdentifier);
|
|
|
|
try
|
|
{
|
|
string BaseFolder = Path.GetDirectoryName(Manifest);
|
|
string Files = File.ReadAllText(Manifest);
|
|
string[] FileList = Files.Split('\n');
|
|
foreach (string Filename in FileList)
|
|
{
|
|
if (!string.IsNullOrEmpty(Filename) && !string.IsNullOrWhiteSpace(Filename))
|
|
{
|
|
string Trimmed = Filename.Trim();
|
|
string SourceFilename = BaseFolder + "\\" + Trimmed;
|
|
SourceFilename = SourceFilename.Replace('/', '\\');
|
|
string DestFilename = "/Documents/" + Trimmed.Replace("cookeddata/", "");
|
|
DestFilename = DestFilename.Replace('\\', '/');
|
|
SourceFilename = SourceFilename.Replace('\\', Path.DirectorySeparatorChar);
|
|
WriteProgressLine("Copying '{0}' -> '{1}' ...", 0, SourceFilename, DestFilename);
|
|
CopyFileToPhone(SourceFilename, DestFilename, 1024 * 1024);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
WriteProgressLine("Failed to transfer a file, extended error is '{0}'", 100, ex.Message);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
WriteProgressLine("Error: Failed to connect to bundle '{0}'", 100, BundleIdentifier);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to copy all of the files in a particular directory on the PC to the phone directory
|
|
/// (requires the bundle identifier to be able to mount that directory)
|
|
/// </summary>
|
|
public bool TryBackup(string BundleIdentifier, string[] Files)
|
|
{
|
|
if (ConnectToBundle(BundleIdentifier))
|
|
{
|
|
WriteProgressLine("Connected to bundle '{0}'", 0, BundleIdentifier);
|
|
|
|
try
|
|
{
|
|
string SafeDeviceName = MobileDeviceInstance.SanitizePathNoFilename(DeviceName);
|
|
foreach (string Filename in Files)
|
|
{
|
|
// string BaseFolder = Path.GetDirectoryName(Filename);
|
|
string Manifest = Path.GetDirectoryName(Filename) + "\\" + SafeDeviceName + "_" + Path.GetFileName(Filename);
|
|
CopyFileFromPhone(Manifest, "/Documents/" + Path.GetFileName(Filename), 1024 * 1024);
|
|
}
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
WriteProgressLine("Failed to transfer a file, extended error is '{0}'", 100, ex.Message);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
string SafeDeviceName = MobileDeviceInstance.SanitizePathNoFilename(DeviceName);
|
|
foreach (string Filename in Files)
|
|
{
|
|
// string BaseFolder = Path.GetDirectoryName(Filename);
|
|
string Manifest = Path.GetDirectoryName (Filename) + "\\" + SafeDeviceName + "_" + Path.GetFileName (Filename);
|
|
WriteProgressLine("File to be written '{0}'", 100, Manifest);
|
|
File.WriteAllText (Manifest, "");
|
|
}
|
|
WriteProgressLine("Error: Failed to connect to bundle '{0}'", 100, BundleIdentifier);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the names of files in a specified directory
|
|
/// </summary>
|
|
/// <param name="path">The directory from which to retrieve the files.</param>
|
|
/// <returns>A <c>String</c> array of file names in the specified directory. Names are relative to the provided directory</returns>
|
|
public string[] GetFiles(string path)
|
|
{
|
|
return GetFiles(path, false);
|
|
}
|
|
|
|
public string[] GetFiles(string path, bool bIncludeDirs)
|
|
{
|
|
if (!IsConnected)
|
|
{
|
|
throw new Exception("Not connected to phone");
|
|
}
|
|
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
|
|
IntPtr hAFCDir = IntPtr.Zero;
|
|
if (MobileDevice.DeviceImpl.DirectoryOpen(AFCCommsHandle, full_path, ref hAFCDir) != 0)
|
|
{
|
|
throw new Exception("Path does not exist");
|
|
}
|
|
|
|
string buffer = null;
|
|
ArrayList paths = new ArrayList();
|
|
MobileDevice.DeviceImpl.DirectoryRead(AFCCommsHandle, hAFCDir, ref buffer);
|
|
|
|
while (buffer != null)
|
|
{
|
|
if (!IsDirectory(FullPath(full_path, buffer)))
|
|
{
|
|
paths.Add(buffer);
|
|
}
|
|
else
|
|
{
|
|
if (bIncludeDirs)
|
|
{
|
|
paths.Add(buffer + "/");
|
|
}
|
|
}
|
|
MobileDevice.DeviceImpl.DirectoryRead(AFCCommsHandle, hAFCDir, ref buffer);
|
|
}
|
|
MobileDevice.DeviceImpl.DirectoryClose(AFCCommsHandle, hAFCDir);
|
|
return (string[])paths.ToArray(typeof(string));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the FileInfo dictionary
|
|
/// </summary>
|
|
/// <param name="path">The file or directory for which to retrieve information.</param>
|
|
public Dictionary<string, string> GetFileInfo(string path)
|
|
{
|
|
Dictionary<string, string> ans = new Dictionary<string, string>();
|
|
TypedPtr<AFCDictionary> Data;
|
|
|
|
int ret = MobileDevice.DeviceImpl.FileInfoOpen(AFCCommsHandle, path, out Data);
|
|
if ((ret == 0) && (Data.Handle != IntPtr.Zero))
|
|
{
|
|
IntPtr pname;
|
|
IntPtr pvalue;
|
|
|
|
while (MobileDevice.DeviceImpl.KeyValueRead(Data, out pname, out pvalue) == 0 && pname != IntPtr.Zero && pvalue != IntPtr.Zero)
|
|
{
|
|
string name = Marshal.PtrToStringAnsi(pname);
|
|
string value = Marshal.PtrToStringAnsi(pvalue);
|
|
if ((name != null) && (value != null))
|
|
{
|
|
ans.Add(name, value);
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
MobileDevice.DeviceImpl.KeyValueClose(Data);
|
|
}
|
|
|
|
return ans;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the st_ifmt of a path
|
|
/// </summary>
|
|
/// <param name="path">Path to query</param>
|
|
/// <returns>string representing value of st_ifmt</returns>
|
|
private string Get_st_ifmt(string path)
|
|
{
|
|
Dictionary<string, string> fi = GetFileInfo(path);
|
|
return fi["st_ifmt"];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the size and type of the specified file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The file or directory for which to retrieve information.</param>
|
|
/// <param name="size">Returns the size of the specified file or directory</param>
|
|
/// <param name="directory">Returns <c>true</c> if the given path describes a directory, false if it is a file.</param>
|
|
public void GetFileInfo(string path, out ulong size, out bool directory)
|
|
{
|
|
Dictionary<string, string> fi = GetFileInfo(path);
|
|
|
|
size = fi.ContainsKey("st_size") ? System.UInt64.Parse(fi["st_size"]) : 0;
|
|
|
|
bool SLink = false;
|
|
directory = false;
|
|
if (fi.ContainsKey("st_ifmt"))
|
|
{
|
|
switch (fi["st_ifmt"])
|
|
{
|
|
case "S_IFDIR": directory = true; break;
|
|
case "S_IFLNK": SLink = true; break;
|
|
}
|
|
}
|
|
|
|
if (SLink)
|
|
{
|
|
// test for symbolic directory link
|
|
IntPtr hAFCDir = IntPtr.Zero;
|
|
|
|
if (directory = (MobileDevice.DeviceImpl.DirectoryOpen(AFCCommsHandle, path, ref hAFCDir) == 0))
|
|
{
|
|
MobileDevice.DeviceImpl.DirectoryClose(AFCCommsHandle, hAFCDir);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the size of the specified file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The file or directory for which to obtain the size.</param>
|
|
/// <returns></returns>
|
|
public ulong FileSize(string path)
|
|
{
|
|
bool is_dir;
|
|
ulong size;
|
|
|
|
GetFileInfo(path, out size, out is_dir);
|
|
return size;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates the directory specified in path
|
|
/// </summary>
|
|
/// <param name="path">The directory path to create</param>
|
|
/// <returns>true if directory was created</returns>
|
|
public bool CreateDirectory(string path)
|
|
{
|
|
return !(MobileDevice.DeviceImpl.DirectoryCreate(AFCCommsHandle, FullPath(CurrentDirectory, path)) != 0);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the names of subdirectories in a specified directory.
|
|
/// </summary>
|
|
/// <param name="path">The path for which an array of subdirectory names is returned.</param>
|
|
/// <returns>An array of type <c>String</c> containing the names of subdirectories in <c>path</c>.</returns>
|
|
public string[] GetDirectories(string path)
|
|
{
|
|
if (!IsConnected)
|
|
{
|
|
Reconnect();
|
|
if (!IsConnected)
|
|
{
|
|
throw new Exception("Not connected to phone");
|
|
}
|
|
}
|
|
|
|
IntPtr hAFCDir = IntPtr.Zero;
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
//full_path = "/private"; // bug test
|
|
|
|
int res = MobileDevice.DeviceImpl.DirectoryOpen(AFCCommsHandle, full_path, ref hAFCDir);
|
|
if (res != 0)
|
|
{
|
|
throw new Exception("Path does not exist: " + res.ToString());
|
|
}
|
|
|
|
string buffer = null;
|
|
ArrayList paths = new ArrayList();
|
|
MobileDevice.DeviceImpl.DirectoryRead(AFCCommsHandle, hAFCDir, ref buffer);
|
|
|
|
while (buffer != null)
|
|
{
|
|
if ((buffer != ".") && (buffer != "..") && IsDirectory(FullPath(full_path, buffer)))
|
|
{
|
|
paths.Add(buffer);
|
|
}
|
|
MobileDevice.DeviceImpl.DirectoryRead(AFCCommsHandle, hAFCDir, ref buffer);
|
|
}
|
|
MobileDevice.DeviceImpl.DirectoryClose(AFCCommsHandle, hAFCDir);
|
|
return (string[])paths.ToArray(typeof(string));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Moves a file or a directory and its contents to a new location or renames a file or directory if the old and new parent path matches.
|
|
/// </summary>
|
|
/// <param name="sourceName">The path of the file or directory to move or rename.</param>
|
|
/// <param name="destName">The path to the new location for <c>sourceName</c>.</param>
|
|
/// <remarks>Files cannot be moved across filesystem boundaries.</remarks>
|
|
public bool Rename(string sourceName, string destName)
|
|
{
|
|
return MobileDevice.DeviceImpl.RenamePath(AFCCommsHandle,
|
|
FullPath(CurrentDirectory, sourceName),
|
|
FullPath(CurrentDirectory, destName)) == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the root information for the specified path.
|
|
/// </summary>
|
|
/// <param name="path">The path of a file or directory.</param>
|
|
/// <returns>A string containing the root information for the specified path. </returns>
|
|
public string GetDirectoryRoot(string path)
|
|
{
|
|
return "/";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determines whether the given path refers to an existing file or directory on the phone.
|
|
/// </summary>
|
|
/// <param name="path">The path to test.</param>
|
|
/// <returns><c>true</c> if path refers to an existing file or directory, otherwise <c>false</c>.</returns>
|
|
public bool Exists(string path)
|
|
{
|
|
TypedPtr<AFCDictionary> data = IntPtr.Zero;
|
|
|
|
int ret = MobileDevice.DeviceImpl.FileInfoOpen(AFCCommsHandle, path, out data);
|
|
if (ret == 0)
|
|
{
|
|
MobileDevice.DeviceImpl.KeyValueClose(data);
|
|
}
|
|
|
|
return ret == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determines whether the given path refers to an existing directory on the phone.
|
|
/// </summary>
|
|
/// <param name="path">The path to test.</param>
|
|
/// <returns><c>true</c> if path refers to an existing directory or is a symbolic link to one, otherwise <c>false</c>.</returns>
|
|
public bool IsDirectory(string path)
|
|
{
|
|
bool is_dir;
|
|
ulong size;
|
|
|
|
GetFileInfo(path, out size, out is_dir);
|
|
return is_dir;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Test if path represents a regular file
|
|
/// </summary>
|
|
/// <param name="path">path to query</param>
|
|
/// <returns>true if path refers to a regular file, false if path is a link or directory</returns>
|
|
public bool IsFile(string path)
|
|
{
|
|
return Get_st_ifmt(path) == "S_IFREG";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Test if path represents a link
|
|
/// </summary>
|
|
/// <param name="path">path to test</param>
|
|
/// <returns>true if path is a symbolic link</returns>
|
|
public bool IsLink(string path)
|
|
{
|
|
return Get_st_ifmt(path) == "S_IFLNK";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes an empty directory from a specified path.
|
|
/// </summary>
|
|
/// <param name="path">The name of the empty directory to remove. This directory must be writable and empty.</param>
|
|
public void DeleteDirectory(string path)
|
|
{
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
if (IsDirectory(full_path))
|
|
{
|
|
MobileDevice.DeviceImpl.RemovePath(AFCCommsHandle, full_path);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes the specified directory and, if indicated, any subdirectories in the directory.
|
|
/// </summary>
|
|
/// <param name="path">The name of the directory to remove.</param>
|
|
/// <param name="recursive"><c>true</c> to remove directories, subdirectories, and files in path; otherwise, <c>false</c>. </param>
|
|
public void DeleteDirectory(string path, bool recursive)
|
|
{
|
|
if (!recursive)
|
|
{
|
|
DeleteDirectory(path);
|
|
return;
|
|
}
|
|
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
if (IsDirectory(full_path))
|
|
{
|
|
InternalDeleteDirectory(path);
|
|
}
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes the specified file.
|
|
/// </summary>
|
|
/// <param name="path">The name of the file to remove.</param>
|
|
public void DeleteFile(string path)
|
|
{
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
if (Exists(full_path))
|
|
{
|
|
MobileDevice.DeviceImpl.RemovePath(AFCCommsHandle, full_path);
|
|
}
|
|
}
|
|
#endregion // Filesystem
|
|
|
|
#region Public Methods
|
|
|
|
/// <summary>
|
|
/// Close the AFC connection
|
|
/// </summary>
|
|
public void Disconnect()
|
|
{
|
|
if (AFCCommsHandle.Handle != IntPtr.Zero)
|
|
{
|
|
MobileDevice.DeviceImpl.ConnectionClose(AFCCommsHandle);
|
|
MobileDevice.DeviceImpl.StopSession(iPhoneHandle);
|
|
MobileDevice.DeviceImpl.Disconnect(iPhoneHandle);
|
|
}
|
|
|
|
AFCCommsHandle = IntPtr.Zero;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Close and Reopen AFC Connection
|
|
/// </summary>
|
|
/// <returns>status from reopen</returns>
|
|
public void Reconnect()
|
|
{
|
|
Disconnect();
|
|
ConnectToPhone();
|
|
}
|
|
|
|
#endregion // public Methods
|
|
|
|
void CopyFileToPhone(string PathOnPC, string PathOnPhone)
|
|
{
|
|
CopyFileToPhone(PathOnPC, PathOnPhone, 1024 * 1024);
|
|
}
|
|
|
|
void CopyFileToPhone(string PathOnPC, string PathOnPhone, int ChunkSize)
|
|
{
|
|
DateTime StartTime = DateTime.Now;
|
|
|
|
byte[] buffer = new byte[ChunkSize];
|
|
|
|
// verify we are still connected
|
|
if (!IsConnected)
|
|
{
|
|
Reconnect();
|
|
}
|
|
|
|
// Make sure the directory exists on the phone
|
|
string DirectoryOnPhone = PathOnPhone.Remove(PathOnPhone.LastIndexOf('/'));
|
|
if (!IsDirectory(DirectoryOnPhone))
|
|
{
|
|
Console.WriteLine("Directory (" + DirectoryOnPhone + ") doesn't exist!");
|
|
if (!CreateDirectory(DirectoryOnPhone))
|
|
{
|
|
Console.WriteLine("CreateDirectory (" + DirectoryOnPhone + ") failed");
|
|
// throw new IOException("CreateDirectory (" + DirectoryOnPhone + ") failed");
|
|
}
|
|
}
|
|
|
|
|
|
FileStream SourceFile = File.OpenRead(PathOnPC);
|
|
iPhoneFile DestinationFile = iPhoneFile.OpenWrite(this, PathOnPhone);
|
|
long ProgressInterval = Math.Max(buffer.Length, (SourceFile.Length / TransferProgressDivisor));
|
|
|
|
long NextProgressPrintout = ProgressInterval;
|
|
long TotalBytesRead = 0;
|
|
|
|
|
|
int BytesRead = SourceFile.Read(buffer, 0, buffer.Length);
|
|
while (BytesRead > 0)
|
|
{
|
|
if (TotalBytesRead >= NextProgressPrintout)
|
|
{
|
|
NextProgressPrintout += ProgressInterval;
|
|
|
|
if (OnGenericProgress != null)
|
|
{
|
|
int PercentDone = (int)((100 * TotalBytesRead) / SourceFile.Length);
|
|
string Msg = "Transferred " + (SourceFile.Position / 1024) + " KB of " + (SourceFile.Length / 1024) + " KB";
|
|
OnGenericProgress(Msg, PercentDone);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine(" ... Transferred " + (SourceFile.Position / 1024) + " KB of " + (SourceFile.Length / 1024) + " KB");
|
|
}
|
|
}
|
|
|
|
DestinationFile.Write(buffer, 0, BytesRead);
|
|
BytesRead = SourceFile.Read(buffer, 0, buffer.Length);
|
|
|
|
TotalBytesRead += BytesRead;
|
|
}
|
|
|
|
// DestinationFile.Flush();
|
|
DestinationFile.Close();
|
|
SourceFile.Close();
|
|
|
|
TimeSpan CopyLength = DateTime.Now - StartTime;
|
|
Console.WriteLine(" ... Finished copying to '{1}' in {0:0.00} s", CopyLength.TotalSeconds, PathOnPhone);
|
|
}
|
|
|
|
// Default level is 6, and 0,3,7 are regularly used. Bump the logging level up to 7 to get verbose logs.
|
|
void SetLoggingLevel(int Threshold)
|
|
{
|
|
Int32 LoggingThreshold = Math.Min(Math.Max(0, Threshold), 7);
|
|
MobileDevice.CoreImpl.CFPreferencesSetAppValue(
|
|
(IntPtr)MobileDevice.CFStringMakeConstantString("LogLevel"),
|
|
(IntPtr)MobileDevice.CFNumberCreate(LoggingThreshold),
|
|
(IntPtr)MobileDevice.CFStringMakeConstantString("com.apple.MobileDevice"));
|
|
}
|
|
|
|
void WriteProgressLine(string Fmt, int ProgressCount, params object[] Args)
|
|
{
|
|
string Line = String.Format(Fmt, Args);
|
|
|
|
if (OnGenericProgress != null)
|
|
{
|
|
OnGenericProgress(Line, ProgressCount);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine(Line);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies a file from the phone to the PC
|
|
/// </summary>
|
|
void CopyFileFromPhone(string PathOnPC, string PathOnPhone, int ChunkSize)
|
|
{
|
|
DateTime StartTime = DateTime.Now;
|
|
|
|
byte[] buffer = new byte[ChunkSize];
|
|
|
|
// Make sure the directory exists on the PC
|
|
string DirectoryOnPC = Path.GetDirectoryName(PathOnPC);
|
|
Directory.CreateDirectory(DirectoryOnPC);
|
|
|
|
iPhoneFile SourceFile = iPhoneFile.OpenRead(this, PathOnPhone);
|
|
FileStream DestinationFile = File.OpenWrite(PathOnPC);
|
|
|
|
long TotalBytesRead = 0;
|
|
|
|
int BytesRead = SourceFile.Read(buffer, 0, buffer.Length);
|
|
while (BytesRead > 0)
|
|
{
|
|
DestinationFile.Write(buffer, 0, BytesRead);
|
|
BytesRead = SourceFile.Read(buffer, 0, buffer.Length);
|
|
|
|
TotalBytesRead += BytesRead;
|
|
}
|
|
|
|
DestinationFile.Flush();
|
|
DestinationFile.Close();
|
|
SourceFile.Close();
|
|
|
|
TimeSpan CopyLength = DateTime.Now - StartTime;
|
|
WriteProgressLine(" ... Finished copying to '{1}' in {0:0.00} s", 100, CopyLength.TotalSeconds, PathOnPC);
|
|
}
|
|
|
|
|
|
string MakeUnixPath(string PlatformPath)
|
|
{
|
|
string Result = Path.GetFullPath(PlatformPath);
|
|
|
|
// Convert C:\ to /C\, the \ will get converted to / in the next step
|
|
if (Path.IsPathRooted(Result))
|
|
{
|
|
string Root = Path.GetPathRoot(Result);
|
|
|
|
Result = '/' + Root.Replace(":", "") + Result.Substring(Root.Length);
|
|
}
|
|
|
|
// Convert slash directions
|
|
if (Path.DirectorySeparatorChar != '/')
|
|
{
|
|
Result = Result.Replace(Path.DirectorySeparatorChar, '/');
|
|
}
|
|
|
|
return Result;
|
|
}
|
|
|
|
void ReconnectWithInstallProxy()
|
|
{
|
|
Reconnect();
|
|
if (MobileDevice.DeviceImpl.StartService(iPhoneHandle, MobileDevice.CFStringMakeConstantString("com.apple.mobile.installation_proxy"), ref hInstallService, IntPtr.Zero) != 0)
|
|
{
|
|
Console.WriteLine("Unable to start installation_proxy service!");
|
|
}
|
|
}
|
|
|
|
public delegate void ProgressCallback(string Message, int PercentDone);
|
|
|
|
public ProgressCallback OnGenericProgress = null;
|
|
public int TransferProgressDivisor = 25;
|
|
|
|
/// <summary>
|
|
/// Generic progress callback implementation (used for install/uninstall/etc...)
|
|
/// </summary>
|
|
void HandleProgressCallback(string OuterFunction, TypedPtr<CFDictionary> SourceDict)
|
|
{
|
|
Dictionary<string, object> Dict = MobileDevice.ConvertCFDictionaryToDictionaryString(SourceDict);
|
|
|
|
// Expecting:
|
|
// string,string -> "Status",PhaseOfInstaller
|
|
// string,number -> "PercentComplete",%Done
|
|
try
|
|
{
|
|
string Phase = Dict["Status"] as string;
|
|
int PercentDone = (int)((Double)(Dict["PercentComplete"]));
|
|
|
|
if (OnGenericProgress != null)
|
|
{
|
|
string Msg = String.Format("{0} is {1}% complete at phase '{2}'", OuterFunction, PercentDone, Phase);
|
|
OnGenericProgress(Msg, PercentDone);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine(" ... {0} {1}% complete (phase '{2}')", OuterFunction, PercentDone, Phase);
|
|
}
|
|
}
|
|
catch (System.Exception)
|
|
{
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Progress callback for upgrading or installing
|
|
/// </summary>
|
|
void InstallProgressCallback(IntPtr UntypedSourceDict, IntPtr UserData)
|
|
{
|
|
HandleProgressCallback("Install", UntypedSourceDict);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Progress callback for uninstalling
|
|
/// </summary>
|
|
void UninstallProgressCallback(IntPtr UntypedSourceDict, IntPtr UserData)
|
|
{
|
|
HandleProgressCallback("Uninstall", UntypedSourceDict);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Try to install an IPA on the mobile device (it must have already been copied to the PublicStaging directory, with the same filename as the path passed in)
|
|
/// </summary>
|
|
public bool TryInstall(string IPASourcePathOnPC)
|
|
{
|
|
DateTime StartTime = DateTime.Now;
|
|
|
|
ReconnectWithInstallProxy();
|
|
|
|
IntPtr LiveConnection = IntPtr.Zero;
|
|
IntPtr ClientOptions = IntPtr.Zero;
|
|
|
|
TypedPtr<CFURL> UrlPath = MobileDevice.CreateFileUrlHelper(IPASourcePathOnPC, false);
|
|
// string UrlPathAsString = MobileDevice.GetStringForUrl(UrlPath);
|
|
|
|
int Result = MobileDevice.DeviceImpl.SecureInstallApplication(LiveConnection, iPhoneHandle, UrlPath, ClientOptions, InstallProgressCallback, IntPtr.Zero);
|
|
if (Result == 0)
|
|
{
|
|
Console.WriteLine("Install of \"{0}\" completed successfully in {2:0.00} seconds", Path.GetFileName(IPASourcePathOnPC), Result, (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("Install of \"{0}\" failed with error code {1} in {2:0.00} seconds", Path.GetFileName(IPASourcePathOnPC), MobileDevice.GetErrorString(Result), (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
|
|
return Result == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Try to update an IPA on the mobile device (it must have already been copied to the PublicStaging directory, with the same filename as the path passed in)
|
|
/// </summary>
|
|
public bool TryUpgrade(string IPASourcePathOnPC)
|
|
{
|
|
DateTime StartTime = DateTime.Now;
|
|
|
|
ReconnectWithInstallProxy();
|
|
|
|
IntPtr LiveConnection = IntPtr.Zero;
|
|
IntPtr ClientOptions = IntPtr.Zero;
|
|
|
|
TypedPtr<CFURL> UrlPath = MobileDevice.CreateFileUrlHelper(IPASourcePathOnPC, false);
|
|
// string UrlPathAsString = MobileDevice.GetStringForUrl(UrlPath);
|
|
|
|
int Result = MobileDevice.DeviceImpl.SecureUpgradeApplication(LiveConnection, iPhoneHandle, UrlPath, ClientOptions, InstallProgressCallback, IntPtr.Zero);
|
|
|
|
if (Result == 0)
|
|
{
|
|
Console.WriteLine("Install \\ Update of \"{0}\" finished in {2:0.00} seconds", Path.GetFileName(IPASourcePathOnPC), Result, (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("Install \\ Update of \"{0}\" failed with {1} in {2:0.00} seconds", Path.GetFileName(IPASourcePathOnPC), MobileDevice.GetErrorString(Result), (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
|
|
return Result == 0;
|
|
}
|
|
|
|
public bool TryUninstall(string ApplicationIdentifier)
|
|
{
|
|
DateTime StartTime = DateTime.Now;
|
|
|
|
ReconnectWithInstallProxy();
|
|
|
|
TypedPtr<CFString> CF_ApplicationIdentifier = MobileDevice.CFStringMakeConstantString(ApplicationIdentifier);
|
|
|
|
IntPtr CF_ClientOptions = IntPtr.Zero;
|
|
// IntPtr ExtraKey = IntPtr.Zero;
|
|
// IntPtr ExtraValue = IntPtr.Zero;
|
|
IntPtr ConnectionHandle = IntPtr.Zero;
|
|
|
|
int Result = MobileDevice.DeviceImpl.SecureUninstallApplication(ConnectionHandle, iPhoneHandle, CF_ApplicationIdentifier, CF_ClientOptions, UninstallProgressCallback, IntPtr.Zero);
|
|
|
|
if (Result == 0)
|
|
{
|
|
Console.WriteLine("Uninstall of \"{0}\" completed successfully in {2:0.00} seconds", ApplicationIdentifier, Result, (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("Uninstall of \"{0}\" failed with {1} in {2:0.00} seconds", ApplicationIdentifier, MobileDevice.GetErrorString(Result), (DateTime.Now - StartTime).TotalSeconds);
|
|
}
|
|
|
|
return Result == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies a file to the PublicStaging directory on the mobile device, preserving the filename
|
|
/// </summary>
|
|
/// <param name="SourceFile"></param>
|
|
public void CopyFileToPublicStaging(string SourceFile)
|
|
{
|
|
string IpaFilename = Path.GetFileName(SourceFile);
|
|
CopyFileToPhone(SourceFile, "PublicStaging/" + IpaFilename);
|
|
//Dictionary<string, string> fi = GetFileInfo("/PublicStaging/" + IpaFilename);
|
|
}
|
|
|
|
public bool StartSyslogService()
|
|
{
|
|
if (MobileDevice.DeviceImpl.Connect(iPhoneHandle) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Failed to Connect");
|
|
return false;
|
|
}
|
|
|
|
if (MobileDevice.DeviceImpl.StartSession(iPhoneHandle) == 1)
|
|
{
|
|
Console.WriteLine("Connect: Couldn't start session");
|
|
return false;
|
|
}
|
|
|
|
if (MobileDevice.DeviceImpl.SecureStartService(iPhoneHandle, MobileDevice.CFStringMakeConstantString("com.apple.syslog_relay"), 0, ref hSyslogService) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Couldn't Start syslog relay service");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public string GetSyslogData()
|
|
{
|
|
return MobileDevice.DeviceImpl.ServiceConnectionReceive(hSyslogService);
|
|
}
|
|
|
|
public void StopSyslogService()
|
|
{
|
|
MobileDevice.DeviceImpl.StopSession(iPhoneHandle);
|
|
MobileDevice.DeviceImpl.Disconnect(iPhoneHandle);
|
|
}
|
|
|
|
#region Private Methods
|
|
public bool ConnectToPhone()
|
|
{
|
|
SetLoggingLevel(7);
|
|
|
|
// Make sure we can connect to the phone and that it has been paired with this machine previously
|
|
if (MobileDevice.DeviceImpl.Connect(iPhoneHandle) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Failed to Connect");
|
|
return false;
|
|
}
|
|
|
|
if (MobileDevice.DeviceImpl.IsPaired(iPhoneHandle) != 1)
|
|
{
|
|
Console.WriteLine("Connect: Device is not paired");
|
|
return false;
|
|
}
|
|
|
|
if (MobileDevice.DeviceImpl.ValidatePairing(iPhoneHandle) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Pairing not valid");
|
|
return false;
|
|
}
|
|
|
|
// Start a session
|
|
if (MobileDevice.DeviceImpl.StartSession(iPhoneHandle) == 1)
|
|
{
|
|
Console.WriteLine("Connect: Couldn't start session");
|
|
return false;
|
|
}
|
|
|
|
if (MobileDevice.DeviceImpl.StartService(iPhoneHandle, MobileDevice.CFStringMakeConstantString("com.apple.afc"), ref hService, IntPtr.Zero) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Couldn't Start AFC service");
|
|
return false;
|
|
}
|
|
|
|
|
|
// Open a file sharing connection
|
|
if (MobileDevice.DeviceImpl.ConnectionOpen(hService, 0, out AFCCommsHandle) != 0)
|
|
{
|
|
Console.WriteLine("Connect: Couldn't Open File Sharing Connection");
|
|
return false;
|
|
}
|
|
|
|
connected = true;
|
|
return true;
|
|
}
|
|
|
|
public bool ConnectToBundle(string BundleName)
|
|
{
|
|
Reconnect();
|
|
|
|
TypedPtr<CFString> CFBundleName = MobileDevice.CFStringMakeConstantString(BundleName);
|
|
|
|
// Open the bundle
|
|
int Result = MobileDevice.DeviceImpl.StartHouseArrestService(iPhoneHandle, CFBundleName, IntPtr.Zero, ref hService, 0);
|
|
if (Result != 0)
|
|
{
|
|
Console.WriteLine("Failed to connect to bundle '{0}' with {1}", BundleName, MobileDevice.GetErrorString(Result));
|
|
return false;
|
|
}
|
|
|
|
// Open a file sharing connection
|
|
if (MobileDevice.DeviceImpl.ConnectionOpen(hService, 0, out AFCCommsHandle) != 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private void DfuConnectCallback(ref AMRecoveryDevice callback)
|
|
{
|
|
OnDfuConnect(new DeviceNotificationEventArgs(callback));
|
|
}
|
|
|
|
private void DfuDisconnectCallback(ref AMRecoveryDevice callback)
|
|
{
|
|
OnDfuDisconnect(new DeviceNotificationEventArgs(callback));
|
|
}
|
|
|
|
private void RecoveryConnectCallback(ref AMRecoveryDevice callback)
|
|
{
|
|
OnRecoveryModeEnter(new DeviceNotificationEventArgs(callback));
|
|
}
|
|
|
|
private void RecoveryDisconnectCallback(ref AMRecoveryDevice callback)
|
|
{
|
|
OnRecoveryModeLeave(new DeviceNotificationEventArgs(callback));
|
|
}
|
|
|
|
private void InternalDeleteDirectory(string path)
|
|
{
|
|
string full_path = FullPath(CurrentDirectory, path);
|
|
string[] contents = GetFiles(path);
|
|
for (int i = 0; i < contents.Length; i++)
|
|
{
|
|
DeleteFile(full_path + "/" + contents[i]);
|
|
}
|
|
|
|
contents = GetDirectories(path);
|
|
for (int i = 0; i < contents.Length; i++)
|
|
{
|
|
InternalDeleteDirectory(full_path + "/" + contents[i]);
|
|
}
|
|
|
|
DeleteDirectory(path);
|
|
}
|
|
|
|
static char[] path_separators = { '/' };
|
|
internal string FullPath(string path1, string path2)
|
|
{
|
|
|
|
if ((path1 == null) || (path1 == String.Empty))
|
|
{
|
|
path1 = "/";
|
|
}
|
|
|
|
if ((path2 == null) || (path2 == String.Empty))
|
|
{
|
|
path2 = "/";
|
|
}
|
|
|
|
string[] path_parts;
|
|
if (path2[0] == '/')
|
|
{
|
|
path_parts = path2.Split(path_separators);
|
|
}
|
|
else if (path1[0] == '/')
|
|
{
|
|
path_parts = (path1 + "/" + path2).Split(path_separators);
|
|
}
|
|
else
|
|
{
|
|
path_parts = ("/" + path1 + "/" + path2).Split(path_separators);
|
|
}
|
|
|
|
string[] result_parts = new string[path_parts.Length];
|
|
int target_index = 0;
|
|
|
|
for (int i = 0; i < path_parts.Length; i++)
|
|
{
|
|
if (path_parts[i] == "..")
|
|
{
|
|
if (target_index > 0)
|
|
{
|
|
target_index--;
|
|
}
|
|
}
|
|
else if ((path_parts[i] == ".") || (path_parts[i] == ""))
|
|
{
|
|
// Do nothing
|
|
}
|
|
else
|
|
{
|
|
result_parts[target_index++] = path_parts[i];
|
|
}
|
|
}
|
|
|
|
return "/" + String.Join("/", result_parts, 0, target_index);
|
|
}
|
|
#endregion // Private Methods
|
|
}
|
|
}
|