You've already forked UnrealEngineUWP
mirror of
https://github.com/izzy2lost/UnrealEngineUWP.git
synced 2026-03-26 18:15:20 -07:00
========================== MAJOR FEATURES + CHANGES ========================== Change 2992821 on 2016/05/27 by Max.Chen Subway Sequencer: Add "Assets" and "Character" to the list of additional directories to cook. #jira UE-31279 #lockdown Cristina.Riveron Change 2992761 on 2016/05/27 by Max.Chen Add assets from "Directories to Always Cook". #jira UE-31279 #lockdown Cristina.Riveron Change 2992371 on 2016/05/26 by Dmitry.Rekman Fix GUBP Tools node (UE-31378). #jira UE-31378 #lockdown Josh.Adams Change 2992279 on 2016/05/26 by Dmitry.Rekman One more fix for UAT compilation failure (UE-31312). - Make EnvVarsToXML target framework v4.5. #lockdown Josh.Adams #jira UE-31312 Change 2992060 on 2016/05/26 by Josh.Adams - Reset PVRTC compression quality to default, so cooks don't take forever for IOS. We shipped with PVRTC Quality 4 for the App Store version. This is set in the Cooker Settings in the Project Settings window. #lockdown cristina.riveron #jira UE-31373 Change 2992009 on 2016/05/26 by Dmitry.Rekman Fix packaging on Linux (UE-31312). - System.Xml was spelled as System.XML. #jira UE-31312 #lockdown Josh.Adams Change 2991784 on 2016/05/26 by Martin.Wilson Fix for RecalcRequiredBones crashing when there is no lod data #jira UE-30028 #lockdown cristina.riveron Change 2991744 on 2016/05/26 by Dmitry.Rekman Fix Linux code project generation (UE-31322). - Also fixes UE-31318 (not reopening when creating BP project). - Apparently, we cannot reset all signals to default, this makes posix_spawn() fail after fork (child exits with 127). - Added logging of child's return code. #lockdown Josh.Adams #jira UE-31322 #jira UE-31318 Change 2991448 on 2016/05/26 by Nick.Darnell Disabling the logging in the git module that was added from the previous commit. #jira UE-30781 #lockdown cristina.riveron Change 2991352 on 2016/05/26 by Max.Chen Subway Sequencer: Add "Sequencer" to the list of additional directories to cook. #jira UE-31279 #lockdown Cristina.Riveron Change 2991121 on 2016/05/26 by Ben.Marsh Fix ShooterGame warnings on XboxOne. #lockdown cristina.riveron Change 2991097 on 2016/05/26 by Nick.Darnell PR #2386: Git Plugin: fix initialization of a new repository broken by new "migrate" support 4.12 (Contributed by SRombauts) #jira UE-30781 #lockdown cristina.riveron Change 2991095 on 2016/05/26 by Dmitry.Rekman Fix packaging on Linux (UE-31312). - Excludes UAT modules unsupported on the platform (e.g. TVOS). #jira UE-31312 #lockdown Josh.Adams Change 2990806 on 2016/05/25 by Michael.Gay Last minute adjustments to SubwaySequencer shots. Fixed Fade track on master and moved Event tracks to shots. #jira UE-30804 #lockdown Cristina.Riveron Change 2990739 on 2016/05/25 by Dan.Oconnor Fix for transaction buffer failing to restore preview widget trees, these are regenerated post undo/redo and should not be tagged as transactional #jira UE-31155 #lockdown cristina.riveron Change 2990657 on 2016/05/25 by Dmitry.Rekman Fix crash in mono when invoked by the engine (UE-31312). - Reset signal mask on spawning a subprocess. We mask out all signals except explicitly handled, which does not play well with mono. - See also https://answers.unrealengine.com/questions/420161/mono-process-crash.html #jira UE-31312 #lockdown Josh.Adams Change 2990564 on 2016/05/25 by Marc.Audy Undo 4.12 change to DetachFromParent when AttachTo is called with a null parent. #jira UE-00000 #lockdown Cristina.Riveron Change 2990429 on 2016/05/25 by Max.Chen Movie Capture: Fix initialization order warning. Follow up to CL #2990314 #jira UE-31285 #lockdown Nick.Penwarden Change 2990338 on 2016/05/25 by Zabir.Hoque TEMP Fix: On server enqued render thread work is dropped. So on server release Reflection capture resouce immediately instead of trying to defer enque. #jira UE-28838 #lockdown cristina.riveron Change 2990314 on 2016/05/25 by Max.Chen Movie Capture: Flush the viewport when grabbing frames. This fixes more frame accuracy issues. #jira UE-31285 #lockdown Nick.Penwarden Change 2990249 on 2016/05/25 by Max.Chen Sequencer: Fix tick prerequisites getting removed on stop and not re-set on play. This fixes frame accuracies when rendering in a separate process. #jira UE-31285 #lockdown Nick.Penwarden Change 2990243 on 2016/05/25 by Lukasz.Furman Fixed behavior tree observers not being applied correctly #jira UE-31307 #lockdown Cristina.Riveron Change 2990206 on 2016/05/25 by Daniel.Lamb Make sure min number of threads in the large thread pool is at least 2. #jira UE-31253 #lockdown Cristina.Riveron Change 2990182 on 2016/05/25 by Max.Chen Sequencer: Fix null ptr crash on trying to record from current player. This is a regression from the off by one frame fixes. #jira UE-31304 #lockdown Nick.Penwarden Change 2990124 on 2016/05/25 by Chris.Bunner Avoid creating additional inline code fragment casting matching uniform types. #lockdown cristina.riveron #jira UE-29089 Change 2989978 on 2016/05/25 by Uriel.Doyon Merged fix for issue with resolution scale in PostProcessVisualizeComplexity #jira UE-29473 #lockdown cristina.riveron Change 2989970 on 2016/05/25 by Taizyd.Korambayil #lockdown cristina.riveron #jira UE-31293 Added TestMaps Folder and moved all Non-Relevant Maps into it. Change 2989911 on 2016/05/25 by Chris.Babcock Remove warning about Android debugging since CodeWorks for Android Nsight supports VS2015 #jira UE-31292 #ue4 #android #lockdown cristina.riveron Change 2989898 on 2016/05/25 by Robert.Manuszewski Splitting inline shader registration from serialization. Serialization can happen on the async loading thread but registration should only happen on the game thread. Removed a lot of critical section locks. Reimplementing CL #2952596 #jira UE-29245 #lockdown Nick.Penwarden Change 2989849 on 2016/05/25 by Max.Preussner Sequencer: Fixed Crash when playing UMG sequence with audio tracks (UE-31289) #jira UE-31289 #lockdown nick.penwarden Change 2989793 on 2016/05/25 by Max.Chen Sequencer: Change automated capture so it captures in response to a sequence update to fix off by one frames. #jira UE-30755 #lockdown Nick.Penwarden Change 2989792 on 2016/05/25 by Max.Chen Sequencer: Put back setting MaxFPS when forcing fixed frame interval playback to fix motion blur in editor. #jira UE-30755 #lockdown Nick.Penwarden Change 2989774 on 2016/05/25 by Mike.Beach Mirroring CL 2946932 Guarding against invalid EdGraphPins (ones that have been moved to the transient package) when constructing the widget - prevents a crash that we've been unable to repro or determine the cause of (turns it instead into an ensure, so we can collect more contextual information on the issue). #lockdown cristina.riveron #jira UE-26998 Change 2989765 on 2016/05/25 by Olaf.Piesche Moivng CL 2967970 from Dev-Rendering - fix for #jira UE-27297 #lockdown nick.penwarden Change 2989481 on 2016/05/25 by Marc.Audy Properly route AttachToComponent to SetupAttachment if called from the constructor #jira UE-31055 #lockdown Cristina.Riveron Change 2989369 on 2016/05/25 by Robert.Manuszewski Don't create asset import data for archetype TileMap. Propagate component flags to TileMap if the component is an archetype. #jira UE-31033 #lockdown Nick.Penwarden Change 2988975 on 2016/05/24 by Max.Preussner Sequencer: Fixed Cinematic Camera look at tool crashes on auto save (UE-31195) #jira UE-31195 #lockdown nick.penwarden Change 2988834 on 2016/05/24 by Max.Chen Movie Capture: Crash fix - Protect against null encoding filter. #jira UE-31233 #lockdown Nick.Penwarden Change 2988764 on 2016/05/24 by Peter.Sauerbrei fix for exception when deploying to tvOS from PC #jira UE-30318 #lockdown cristina.riveron Change 2988540 on 2016/05/24 by Jeff.Campeau Disable incompatible OpenVR for Windows XP builds. Gut SteamVR and SteamVRController for Windows XP builds (rely on OpenVR). #lockdown Nick.Penwarden #jira UE-30823 Change 2988491 on 2016/05/24 by Zak.Middleton #ue4 - (4.12) Remove version check from serialization logic that fixes up stale transient properties. They would still loaded for archetypes and we always want to prevent that in the future. #lockdown cristina.riveron #jira UE-30625 Change 2988427 on 2016/05/24 by Aaron.McLeran #jira UE-31028 Stop Quietest Concurrency does not remove the quietest sound Fix is to not re-add the sound once its stopped due to max concurrency. #tests ran the QA test map that demonstrated the problem #lockdown cristina.riveron Change 2988391 on 2016/05/24 by Taizyd.Korambayil #lockdown cristina.riveron #jira UE-30301 Rebuilt Ligthing for all Content Example Maps Change 2988315 on 2016/05/24 by Allan.Bentham Re-enabled FLUTBlenderPS on vulkan devices. (it's required for protostar) #jira UE-31079 Change 2988227 on 2016/05/24 by Frank.Fella Sequencer - Add support for forcing editor and runtime evaluation to happen on exact fixed frame intervals. Updated the subway sequencer sample to work with these changes. Change missed in first checkin. #Jira UE-30755 Change 2988200 on 2016/05/24 by Robert.Manuszewski Assert if MaxObjectsInEditor or MaxObjectsInGame are too big and collide with EInternalObjectFlags #jira UE-31218 Change 2988181 on 2016/05/24 by Peter.Sauerbrei revert out the last fix and add more logging as I can't reproduce this bug #jira UE-30813 Change 2988140 on 2016/05/24 by Frank.Fella Sequencer - Add support for forcing editor and runtime evaluation to happen on exact fixed frame intervals. Updated the subway sequencer sample to work with these changes. #Jira UE-30755 Change 2988081 on 2016/05/24 by Jamie.Dale Better fix for UE-29651 that will also work with packages saved from a build without an engine version There was no version bump for the change to FFormatArgumentData, but VER_UE4_K2NODE_VAR_REFERENCEGUIDS was added at almost the same time so testing that should handle the vast majority of packages that we have internally, and will handle all external packages. #jira UE-29651 Change 2987964 on 2016/05/24 by Lee.Clark Fix empty ENV path when compiling PS4 targets. #jira UE-31210 Change 2987721 on 2016/05/23 by Dan.Oconnor Reworking node validation change done in 2910382 so that nodes that are going to spawn other nodes in the expansion step are still validated. #jira UE-31099 Change 2987696 on 2016/05/23 by Chris.Babcock Update AndroidWorks 1R1 to CodeWorks for Android 1R4 #jira UEPLAT-1312 #ue4 #android Change 2987624 on 2016/05/23 by Jeff.Campeau Fix a define protection for WinXP stack walking support. #jira UE-30823 Change 2987607 on 2016/05/23 by Jeff.Campeau Windows Stack Walk fixed to work with Windows XP. Use the ASCII calls where needed. Symbol server is unsupported and is disabled when building for Windows XP. #jira UE-30823 Change 2987593 on 2016/05/23 by Zak.Middleton #ue4 - (4.12) Reject old serialized values of UMovementComponent::UpdatedComponent and UpdatedPrimitive that were saved before those were marked transient. Mark UPawnMovementComponent::PawnOwner and UCharacterMovementComponent::CharacterOwner as transient, and similarly reject old saved values. #jira UE-30625 Change 2987548 on 2016/05/23 by Lukasz.Furman Moved newly added gameplay debugger's code out of perception component #jira UE-31090 Change 2987510 on 2016/05/23 by Lukasz.Furman Restored perception category in old gameplay debugger tool #jira UE-31090 Change 2987278 on 2016/05/23 by Ben.Marsh Rocket: Add Mac GenerateProjectFiles.sh script into installed engine distro. #jira UE-31109 Change 2987156 on 2016/05/23 by Chris.Babcock Added GoogleVR to InstalledEngineFilters.ini #jira UE-31186 #ue4 #android Change 2987129 on 2016/05/23 by Mieszko.Zielinski Fixed FNavigationFilterArea not zeroing its properties in default constuctor #UE4 #jira UE-31185 Change 2987100 on 2016/05/23 by Peter.Sauerbrei fix for crash in DeploymentServer when attempting to copy a file with a space in the path or name #jira UE-30813 Change 2987064 on 2016/05/23 by Dmitry.Rekman PR #2164: [Linux] Fix clang '&&' within '||' error (Contributed by slonopotamus) #jira UE-28537 Change 2987002 on 2016/05/23 by Aaron.McLeran #jira UE-31036 Sound volume does not change when moving past the Non Focus Azimuth range if set to greater than 90 degrees Fix was to remove the clamp on the dot-product #tests ran test map with focus factors greater than 90 degrees Change 2986880 on 2016/05/23 by Mark.Satterthwaite Fix UE-31124 due to bad array iteration logic - amazing that this hadn't been seen earlier. #jira UE-31124 Change 2986873 on 2016/05/23 by Lina.Halper #fix issue with morphtarget importings for LODs - this was caused by option not being set correctly #jira: UE-30955 #code review: Alexis.Matte Change 2986804 on 2016/05/23 by Taizyd.Korambayil #jira UE-31132 Added Missing Function to Blueprint. Change 2986801 on 2016/05/23 by Jamie.Dale SSearchBox will now only delay text changes while it has focus A text changed event when it doesn't have focus is usually triggered by code (rather than the user typing), so we need to process it immediately to avoid other operational ordering issues. #jira UE-31101 Change 2986793 on 2016/05/23 by Martin.Wilson Fix for morph curves not getting applied to meshes in cooked builds (smart names were not being corrected). (brought from dev-rendering 2983747) #Jira UE-31166 Change 2986772 on 2016/05/23 by Benn.Gallagher Fixed montage single node instances with negative rate scales only repeating the final section when looping #jira UE-31164 Change 2986766 on 2016/05/23 by Martin.Wilson Fix for preview not updating when tranform curve flags are changed. #Jira UE-31119 Change 2986569 on 2016/05/23 by Robert.Manuszewski Making hang detection disabled bu default and an opt-in for games. #jira UE-31151 Change 2986564 on 2016/05/23 by Martin.Wilson Fix for being able to set montages on an anim track segment. #jira UE-31039 Change 2986205 on 2016/05/21 by Zabir.Hoque Add new instrumentation to bucketize why we are seeing device lost so often. #jira UE-20434 Change2986071on 2016/05/20 by Dan.Oconnor Fix for TRASHCLASS sneaking into property list when recompiling a blueprint that has a dependency that is dirty and requires bytecode recompilation of its dependencies. Make sure that the dirty blueprint itself is part of the bytecode recompilation process and make sure that blueprints compiled in this way are compiled after their parent classes #jira UE-30411 Change 2986068 on 2016/05/20 by Dan.Oconnor Fix for blueprint change/compile delegates leaking #jira UE-31118 Change 2986044 on 2016/05/20 by Zabir.Hoque Make OpenGL VB allocation support alignment (16 by default). Future work should expose this up through the RHI layers. #CodeReview: Olaf.Piesche, Simon.Tovey #jira UE-29231 Change 2985934 on 2016/05/20 by Mark.Satterthwaite Further changes to ensure that UE-30710 really is fixed while also not live-leaking memory in MetalRHI. #jira UE-30710 Change 2985852 on 2016/05/20 by Max.Chen Subway Sequencer: Remove level sequence editor from plugin list since it's on by default. #jira UE-31106 Change 2985821 on 2016/05/20 by Phillip.Kavan [UE-22874] Fix UObject duplication to preserve default subobjects created by the native class ctor when the root object is duplicated. change summary: - added FObjectDuplicationHelperMethods::GatherDefaultSubobjectsForDuplication() - modified StaticDuplicateObjectEx() to map default subobjects created in the duplicated root object's ctor before entering the serialization pass. this preserves those instances instead of causing StaticConstructObject to destroy/recreate them during serialization as part of the UObject reference duplication logic. #jira UE-22874 Change 2985750 on 2016/05/20 by Michael.Gay Default Game map set to SubwaySequencer_P #jira UE-31108 Change 2985660 on 2016/05/20 by Michael.Gay Removing unused track animation #jira UE-30804 Change 2985349 on 2016/05/20 by Dan.Oconnor Fix for crash that occurs when repeatedly pasting and undoing an object with subobjects. We were not clearing the internal flags when recycling an object #jira UE-30954 Change 2985346 on 2016/05/20 by Leslie.Nivison Updating 4.12 credit #jira UEPROD-820 Change 2985297 on 2016/05/20 by Jamie.Dale Fixed VS version detection It was checking the file version (which is 12), rather than the VS version (which is 12 for 2013, and 14 for 2015). #jira UE-30977 Change 2985233 on 2016/05/20 by Gareth.Martin Fixed crash when building lighting when using "Use Landscape Lightmap" on landscape grass #jira UE-30975 Change 2985184 on 2016/05/20 by Chris.Babcock Move audio warning to show proper error result code #jira UE-31085 #ue4 #android Change 2985183 on 2016/05/20 by Chad.Taylor GoogleVR disabled by default #jira UE-30921 Change 2985145 on 2016/05/20 by Jack.Porter Fix for precision issue causing blocky landscape LOD on iPad Pro and several other iOS devices #jira UE-24792 Change 2985124 on 2016/05/20 by Alex.Delesky #jira UE-29794 If the editor cannot find the SSL DLLs when enabling the Perforce source control plugin, it will now display a warning in the Source Control log instead of crashing. Change 2985066 on 2016/05/20 by Lee.Clark Fix r.SelectiveBasePassOutputs so that it defaults to off #jira UE-30133 Change 2985063 on 2016/05/20 by Allan.Bentham Fix for modulated shadow precision issues on low end android hardware. #jira UE-29083 Change 2985061 on 2016/05/20 by Max.Chen Viewport: Fix crash when the viewport widget is null. #jira UE-31050 Change 2985059 on 2016/05/20 by Rolando.Caloca UE4.12 - Workaround for crash trying to track down other crash #jira UE-30875 Change 2984876 on 2016/05/20 by Richard.TalbotWatkin Made SceneOutliner visibility code safer, to avoid a potential crash. #jira UE-30831 - [CrashReport] UE4Editor_SceneOutliner!SceneOutliner::FGetVisibilityVisitor::RecurseChildren() [sceneoutlinergutter.cpp:24] Change 2984873 on 2016/05/20 by Richard.TalbotWatkin Clipped selection box bounds in Matinee viewport to prevent crash when reading outside of the viewport area. #jira UE-30968 - Ctrl+Alt selection drag inside to outside of Matinee window will crash the editor Change 2984844 on 2016/05/20 by Matthew.Griffin Fixing compile error in mono games Change 2984825 on 2016/05/20 by Robert.Manuszewski When the application crashes becaused the GPU driver was disabled, make sure the CrashReporterClient window gets the updated screen metrics after the driver is restored. #jira UE-30556 Change 2984693 on 2016/05/20 by Phillip.Kavan [UE-30495] Fix BP editor crash on component rename following undo of component add action. change summary: - modified USimpleConstructionScript::CreateNode() to create the initial component template object in the transient package, so that subsequent undo actions restore to that state rather than to a valid BPGC-owned state. - modified StaticConstructObject_Internal() to restore the inclusion of RF_ArchetypeObject-flagged objects in the logic that sets new objects to 'PendingKill' state before recording them into the transaction buffer. this ensures that they can be GC'd when construction is undone in the editor. Tested against sample/repro steps in UE-21240 to ensure that it no longer crashes even with the original change from CL# 2832225 reverted (that fix has since been superceded). #jira UE-30495 Change 2984684 on 2016/05/20 by Phillip.Kavan [UE-30852] Fix BPGC custom property list delta generation & post-construct initialization/serialization to properly handle array values that differ from default in length but not inner element values. change summary: - modified UBlueprintGeneratedClass::BuildCustomPropertyListForPostConstruction()/BuildCustomArrayPropertyListForPostConstruction() to return a boolean value indicating whether or not a delta value was detected. - modified UBlueprintGeneratedClass::BuildCustomArrayPropertyListForPostConstruction() and FBlueprintEditorUtils::BuildComponentInstancingData() to ensure that array properties are emitted to delta property lists if the size differs from default, even if none of the elements actually differ from the default value - removed the ensure() for the array property case in FObjectInitializer::InitPropertiesFromCustomList(), as it is now a valid case to encounter an array property delta value without any actual delta element value overrides following it in the custom property stream - restored the bCanUsePostConstructLink optimization for non-native class types in FObjectInitializer::InitProperties() - modified UArrayProperty::SerializeItem() for the ArUseCustomPropertyList case to not empty the array when a resize is needed on load (read) - this fixes an edge case in the cooked BP component data stream when array size differed from default but only one or more of the inner values actually differed, in which case all the array slots were being reset (constructed/zeroed) but only the overridden value was being serialized (loaded) from the template data stream #jira UE-30852 Change 2984651 on 2016/05/19 by Zabir.Hoque Forcing GoogleVR plugin to disabled by default since its causing even non HDM machines to render split foveated viewports. #CodeReview: Chad.Taylor, Nick.Whiting #jira UE-30921 Change 2984636 on 2016/05/19 by Zabir.Hoque Explicitly store the cubemap resolution in encoded reflection data. #CodeReview Daniel.Wright, Marcus.Wassmer #jira UE-30341 Change 2984454 on 2016/05/19 by Rolando.Caloca UE4.12 - Fix for vulkan failing to load shader Integration mirroring changelist 2984432 #jira UE-28140 Change 2984452 on 2016/05/19 by Marcus.Wassmer #jira UE-31054 Remove autocompletion for ToggleRHIThread and ShowMaterialDrawEvents as they no longer do anything Change 2984415 on 2016/05/19 by Dan.Oconnor Fix for crash when we fail to spawn the preview actor because the desired class is deprecated #jira UE-31027 Change 2984376 on 2016/05/19 by Dan.Oconnor Fix for regression in GetClassDefaults - we were not handling the 'None' case #jira UE-31034 Change 2984316 on 2016/05/19 by Aaron.McLeran #jira UE-31049 Updating the Oculus Audio SDK to vs 1.02 #tests Ran updated SDK in several test maps, confirmed HRTF spatialization is working. Change 2984315 on 2016/05/19 by Lina.Halper Fix issue with importing morphtarget LOD when it's missing between #jira: UE-30949 Change 2984237 on 2016/05/19 by Dan.Oconnor Fix for ensure/possible stale memory access in UpdateOverlaps #jira UE-30919 Change 2984170 on 2016/05/19 by Max.Chen Movie Capture: Another pass at texture streaming fix for movie capture. #jira UE-30986 Change 2984134 on 2016/05/19 by Chad.Taylor Mac compiler warning fix #jira UE-30921 Change2983903on 2016/05/19 by Taizyd.Korambayil #jira UE-30562 Replaced cube With BSP for Floor Change 2983840 on 2016/05/19 by Taizyd.Korambayil #jira UE-30979 Fixed Typo in one of the Stands Change 2983662 on 2016/05/19 by Ben.Marsh GitHub: Add an exception to allow GoogleVR files to be mirrored to GitHub Change 2983653 on 2016/05/19 by Chris.Bunner Modifed previous change to fixup incorrect ensures. #jira UE-30877 Change 2983599 on 2016/05/19 by Chris.Bunner Added ensure and null ptr check to canvas flush. #jira UE-30877 Change2983596on 2016/05/19 by Chad.Taylor FluffyBunny #jira UE-30921 Change 2983534 on 2016/05/19 by Brian.Karis 4.12 fix per pixel translucency #jira UE-30902 Change 2983530 on 2016/05/19 by Chris.Babcock Broadcast EMediaEvent::MediaOpened when media opened successfully #jira UE-31006 #ue4 #android Change 2983427 on 2016/05/19 by Richard.TalbotWatkin Conflated "Import" and "Import Scene" in the File menu; the new action is called "Import Into Level". Limited the allowed file types to .t3d and .fbx. #jira UE-30891 - CRASH: Editor crashes when Importing Actors via File > Import Change 2983386 on 2016/05/19 by Michael.Gay minor last tweaks #jira UE-30804 Change 2983280 on 2016/05/19 by Gil.Gribb UE4 - Fixed crash in FHierarchicalStaticMeshSceneProxy related to reflection captures and foliage. #jira UE-30837 Change 2983079 on 2016/05/18 by Max.Chen Movie Capture: Fix so that texture streaming option for movie capture is set when capturing in editor. #jira UE-30986 Change 2983078 on 2016/05/18 by Dmitriy.Dyomin Added more logging to track UE-30878 #jira UE-30878 Change 2983067 on 2016/05/18 by Dmitriy.Dyomin Fixed: Mobile HDR Path doesn't work on GearVR #jira UE-11846 Change 2983049 on 2016/05/18 by Max.Chen Movie Capture: Fix crash on movie rendering when in HDR mode. #jira UE-30978 Change 2982825 on 2016/05/18 by Mark.Satterthwaite Correctly wait for the dispatch semaphore when clearing the Metal resource free lists. #jira UE-30710 Change 2982697 on 2016/05/18 by Marc.Audy Fix Orion DataProvider use of AddReferencedObjects in light of CL#2982607#jira UE-00000 Change2982546on 2016/05/18 by Taizyd.Korambayil #jira UE-30862 resaved A bunc hof assets to Fix to attempt to fix Build Warnings Change 2982533 on 2016/05/18 by Daniel.Lamb When you package if you haven't saved the changes will not be reflected in the game. #jira UE-30904 Change 2982415 on 2016/05/18 by Marc.Audy Bring forgotten 4.11 CL# 2928377 to 4.12 Ensure that the compiler will throw an error when passing a non-UObject* TArray to AddReferencedObjects #jira UE-28933 Change 2982358 on 2016/05/18 by Taizyd.Korambayil #jira UE-30546 Updated TP_VehicleAdvPawn Chase Camera Location Change 2982280 on 2016/05/18 by Martin.Mittring UE-26409 Crash when Light Propagation Volume Plugin is disabled on a Project #jira:UE-26409 Change 2982229 on 2016/05/18 by Max.Chen Sequencer: Add tick prerequisites so that the level sequence actor ticks before all of the actors that it controls. This fixes some inconsistencies in the movie rendered frames not matching what's in editor. #jira UE-30755 Change 2982080 on 2016/05/18 by Max.Chen Sequence Recorder: Fix crash when component class to record is null. #jira UE-30944 Change 2982041 on 2016/05/18 by Marcus.Wassmer Protect against crashes reading from a null texture. #jira UE-30834 Change 2981915 on 2016/05/18 by Allan.Bentham Do not mosaic encode for modulate blend operations. Fixes dark 'halos' around mod shadows. #jira UE-29083 Change 2981911 on 2016/05/18 by michael.gay Set framing in sequencer, set start to 200 #jira UE-30633 Change 2981904 on 2016/05/18 by Chase.McAllister #jira UE-30943 Removing unused asset to fix DDC compiling bug Change 2981894 on 2016/05/18 by Michael.Gay removed old cameras, changed start frame to remove black at head of sequence #jira UE-30633 Change 2981827 on 2016/05/18 by Gareth.Martin Fixed crash when entering landscape mode while a landscape is selected while simulating - Landscape infos no longer get created for PIE/Simulate landscapes (they were empty anyway) #jira UE-30917 Change 2981725 on 2016/05/18 by Keith.Judge Xbox One - Fix issues with DFAO/DF Shadowing. Problems were in RHIUpdateTexture3D(). Needed to ensure temp texture had the correct bind flags, etc, and also use the graphics context rather than the DMA context to do the copying, as for some reason the DMA engine corrupts some pixels of the distance field atlas texture. #jira UE-27591 Change 2981466 on 2016/05/17 by Max.Chen Merge from Chris Bunner from Dev-SequencerGDC - Frame state fixes when Sequencer is paused; No velocity in AA, Clamp motion blur scale, Clamp to scatter blur method. #jira UE-30576 Change 2981403 on 2016/05/17 by Dan.Oconnor Fix for overzealous filtering of classes with Within markup #jira UE-29878 Change 2981342 on 2016/05/17 by Dan.Oconnor Removing overzealous check. In Dev-BP this has already been downgraded to an ensure, but no reason to ensure now that we understand why it happens. #jira UE-30792 Change 2981318 on 2016/05/17 by Max.Preussner Sequencer: Fixed crash when scrubbing attached audio tracks; reduced nesting (UE-30923) #jira: UE-30923 Change 2981221 on 2016/05/17 by Dan.Oconnor Preventing spawning components with 'Within' markup specified, it is unsupported by the SCSEditor and Core UObject logic at this time. Likely logic is CoreUObject needs to avoid type checking for RF_ArchetypeObject instances and the SCSEditor needs to be more consistent about using that flag on its template objects #jira UE-29878 Change 2981169 on 2016/05/17 by Marc.Audy Gracefully handle invalid GameSingleton class name in ini file Remove unused DefaultPreviewPawnClass and ClassName from Engine #jira UE-30829 Change 2981104 on 2016/05/17 by Mieszko.Zielinski Made AISenses not send information to listeners that are not registered for given sense #UE4 #jira UE-29939 Change 2981086 on 2016/05/17 by Taizyd.Korambayil #jira UE-30568 Added a check to make sure index being accessed was valid (BP_DemoRoom) Change 2980755 on 2016/05/17 by Taizyd.Korambayil #jira UE-30706 Set material to use Translucent Blend Change 2980753 on 2016/05/17 by Jon.Nabozny Initialize FBox used to store result for CalculateQuatACF96Bounds (bump from //UE4/Dev-Framework). #JIRA UE-30846 Change 2980682 on 2016/05/17 by Taizyd.Korambayil #jira UE-30570, UE-30575 Corrected Some Spellings Change 2980559 on 2016/05/17 by Mieszko.Zielinski Changed UNavigationSystem.AgentToNavDataMap to store weak object pointers rather than raw painters #UE4 This should make it immune to navigation data beging destroyed and not removed from AgentToNavDataMap. #jira UE-30836 Change 2980504 on 2016/05/17 by Daniel.Wright Integrate - Movable skylight now matches stationary for subsurface shading models * Two sided was broken in 4.11, Subsurface had never been handled #jira UE-30855 Change 2980467 on 2016/05/17 by Jamie.Dale Added some checks to avoid temporary worlds being added as favorites #jira UE-30613 Change 2980379 on 2016/05/17 by Jurre.deBaare Fix for static mesh merging, little too eager with changes. #jira UE-30808 Change 2980373 on 2016/05/17 by Gareth.Martin Fixed shader compile errors when applying a speedtree material to a landscape spline #jira UE-25820 Change 2980318 on 2016/05/17 by Gareth.Martin Fixed crash when calling EditorApplySpline with a null spline component Also stopped it doing anything in PIE (it's for blutilities, not runtime) #jira UE-30830 Change 2980300 on 2016/05/17 by Marc.Audy Treat Unreachable components the same as BeginDestroyed for endplay/cleanup purposes #jira UE-30839 Change 2980298 on 2016/05/17 by Gareth.Martin Fixed crash when loading landscape projects that used tessellation #jira UE-30742 Change 2980296 on 2016/05/17 by Martin.Wilson Fix crash accessing sync names from a child anim bp #jira UE-30811 Change 2980289 on 2016/05/17 by Jurre.deBaare Fix for regression with merge actor tab #jira UE-30809 Change 2980272 on 2016/05/17 by Ori.Cohen Make sure that root components do not get attached to non root components in the same actor. Fixes crash in scene outliner and other weird issues. #JIRA UE-30876 Change 2980206 on 2016/05/17 by Keith.Judge Xbox One - Bit the bullet and rewrote the occlusion query buffer handling so that we're not reliant on a finite ring buffer. Instead, each query has a small buffer of its own. removing the dependency of ordering when reading back the results. This should save memory on smaller maps too! #jira UE-30581 #jira UEPLAT-623 Change 2980094 on 2016/05/17 by Matthew.Griffin Added OSVR dlls to InstalledEngineFilters.ini so that they are included in Launcher build even though the plugin is disabled by default #jira UE-30611 Change 2979935 on 2016/05/17 by Aaron.Herzog #jira UE-30619 updating owen sk mesh with proper morph Change 2979816 on 2016/05/16 by Chad.Taylor Fix to address a crash related to multiple player VR Preview #jira UE-20109 Change 2979744 on 2016/05/16 by Mike.Beach Disabling Blueprint spawning, InitProperties() optimization until we can figure out why it is not filling out array properties properly. #jira UE-30745 Change 2979743 on 2016/05/16 by Mike.Beach Mirroring CL 2977497 Clearing property nodes and cached read-addresses when changing the details view object (so any queued actions will not operate on invalid properties). #jira UE-26392 Change 2979544 on 2016/05/16 by Daniel.Wright Fixed crash with RTDF shadows when r.DistanceFieldAO was disabled #jira UE-26319 Change 2979477 on 2016/05/16 by michael.gay Remove errant Play Rate track. #jira UE-30633 Change 2979464 on 2016/05/16 by Mark.Satterthwaite Duplicate CL #2945444: Cache the Metal fallback depth-stencil surface for the canvas tile rendering so that we only ever keep one spare depth-stencil surface around. This costs us a little more permanent memory but reduces churn. #jira UE-30849 Change 2979441 on 2016/05/16 by Rolando.Caloca UE4.12 - vk - Fix quitting taking a long time #jira UE-28239 Change 2979315 on 2016/05/16 by Michael.Trepka Rollback //UE4/Release-4.12/Engine/Source/Programs/UnrealBuildTool/System/XcodeProject.cs to revision 1 #jira UE-28016 Change2979304on 2016/05/16 by Jamie.Dale Backing out some changes from CL# 2976673 These caused an issue with Slate hit-testing. The more correct fix here is to make the Slate Windows OS layer treat window positions as relative to the top-left of the window client area, rather than relative to the top-left of the window itself (which includes the OS border). This now matches what other platforms do. To this end, FWindowsWindow::Initialize, FWindowsWindow::MoveWindowTo, and FWindowsWindow::ReshapeWindow all now consider the given window position to be relative to the window client area, and will consistently adjust it to relative to the window before moving/creating the OS window. This only impacts windows with OS borders (aka, non-fullscreen and non-Slate drawn windows). #jira UE-30276 #jira UE-30677 #jira UE-30771 Change 2979077 on 2016/05/16 by Maciej.Mroz #jira UE-28536 Attached Project Crashes on Attempting to Play in Standalone merged from 2979069 Change 2979052 on 2016/05/16 by Chase.McAllister #jira UE-30789 Resaving Maps to fix project warning Change 2978984 on 2016/05/16 by Chase.McAllister #jira UE-30789 Resaving start video assests that contained empty engine version Change 2978806 on 2016/05/16 by Mieszko.Zielinski Fixed EQS tests' scoring equation value getting reset on load #UE4 #jira UE-30470 Change 2978670 on 2016/05/16 by Max.Preussner Media: Workaround for changing Media asset path can cause crash (UE-22691) #jira: UE-22691 Change 2978638 on 2016/05/16 by Michael.Gay Cleanup of old maps in SubwaySequencer project #jira UE-30633 Change 2978636 on 2016/05/16 by Jamie.Dale Added guard against a crash navigating through a menu #jira UE-30698 Change 2978611 on 2016/05/16 by Lee.Clark PS4 - Fix RenderTargetOutputFormat using the wrong output index for velocity rendering when using r.BasePassOutputsVelocity=True #jira UE-30133 Change 2978596 on 2016/05/16 by Allan.Bentham Extend iOS metal Z bias offset to all iOS (metal+gles) depth only shaders. #jira UE-27530 Change 2978566 on 2016/05/16 by Jamie.Dale Downgraded some checks to ensures and added more logging #jira UE-30613 Change2978399on 2016/05/16 by Keith.Judge Xbox One - Fix check() firing when we run out of occlusion buffer space. Also added occlusion query result caching (perf gain!). #jira UE-30581 Change 2978323 on 2016/05/16 by Jurre.deBaare Merge actor panel crashes when selecting a mesh component without static mesh #fix display 'No Static Mesh' when none is available #jira UE-30809 Change 2978322 on 2016/05/16 by Jurre.deBaare Issue with merging meshes resulting data saved across different LOD levels #fix use correct target LOD index for all source LODs #jira UE-30808 #lockdown Nick.Penwarden [CL 2999693 by Ben Marsh in Main branch]
1497 lines
52 KiB
C#
1497 lines
52 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)
|
|
{
|
|
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 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);
|
|
}
|
|
|
|
#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
|
|
}
|
|
}
|