//----------------------------------------------------------------------------- // Copyright (c) Microsoft Corporation. All rights reserved. //----------------------------------------------------------------------------- namespace System.Activities { using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Runtime.Serialization; using System.Activities.Runtime; using System.Runtime; using System.Diagnostics.CodeAnalysis; [DataContract] public class ExclusiveHandle : Handle { ReadOnlyCollection readOnlyBookmarkScopeCollection; List bookmarkScopes; ActivityInstance owningInstance; ActivityExecutor executor; ExclusiveHandleBookmarkList importantBookmarks; ExclusiveHandleBookmarkList unimportantBookmarks; bool bookmarkScopesListIsDefault; public ExclusiveHandle() { this.CanBeRemovedWithExecutingChildren = true; } public ReadOnlyCollection RegisteredBookmarkScopes { get { if (this.bookmarkScopes == null) { return new ReadOnlyCollection(new List()); } if (this.readOnlyBookmarkScopeCollection == null) { this.readOnlyBookmarkScopeCollection = new ReadOnlyCollection(this.bookmarkScopes); } return this.readOnlyBookmarkScopeCollection; } } [DataMember(EmitDefaultValue = false, Name = "bookmarkScopes")] internal List SerializedBookmarkScopes { get { return this.bookmarkScopes; } set { this.bookmarkScopes = value; } } [DataMember(Name = "owningInstance")] internal ActivityInstance SerializedOwningInstance { get { return this.owningInstance; } set { this.owningInstance = value; } } [DataMember(EmitDefaultValue = false, Name = "executor")] internal ActivityExecutor SerializedExecutor { get { return this.executor; } set { this.executor = value; } } [DataMember(EmitDefaultValue = false, Name = "importantBookmarks")] internal ExclusiveHandleBookmarkList SerializedImportantBookmarks { get { return this.importantBookmarks; } set { this.importantBookmarks = value; } } [DataMember(EmitDefaultValue = false, Name = "unimportantBookmarks")] internal ExclusiveHandleBookmarkList SerializedUnimportantBookmarks { get { return this.unimportantBookmarks; } set { this.unimportantBookmarks = value; } } [DataMember(EmitDefaultValue = false, Name = "bookmarkScopesListIsDefault")] internal bool SerializedBookmarkScopesListIsDefault { get { return this.bookmarkScopesListIsDefault; } set { this.bookmarkScopesListIsDefault = value; } } [SuppressMessage(FxCop.Category.Design, FxCop.Rule.ConsiderPassingBaseTypesAsParameters, Justification = "We are restricting the activities that can call this API.")] public void RegisterBookmarkScope(NativeActivityContext context, BookmarkScopeHandle bookmarkScopeHandle) { if (context == null) { throw FxTrace.Exception.ArgumentNull("context"); } context.ThrowIfDisposed(); if (bookmarkScopeHandle == null) { throw FxTrace.Exception.ArgumentNull("bookmarkScopeHandle"); } if ((this.ImportantBookmarks != null && this.ImportantBookmarks.Count != 0) || (this.UnimportantBookmarks != null && this.UnimportantBookmarks.Count != 0)) { throw FxTrace.Exception.AsError(new InvalidOperationException(SR.ExclusiveHandleRegisterBookmarkScopeFailed)); } if (this.bookmarkScopesListIsDefault) { this.bookmarkScopesListIsDefault = false; this.bookmarkScopes.Clear(); } this.bookmarkScopes.Add(bookmarkScopeHandle); this.readOnlyBookmarkScopeCollection = null; } [SuppressMessage(FxCop.Category.Design, FxCop.Rule.ConsiderPassingBaseTypesAsParameters, Justification = "We are restricting the activities that can call this API.")] public void Reinitialize(NativeActivityContext context) { if (context == null) { throw FxTrace.Exception.ArgumentNull("context"); } context.ThrowIfDisposed(); if ((this.ImportantBookmarks != null && this.ImportantBookmarks.Count != 0) || (this.UnimportantBookmarks != null && this.UnimportantBookmarks.Count != 0)) { throw FxTrace.Exception.AsError(new InvalidOperationException(SR.ExclusiveHandleReinitializeFailed)); } this.bookmarkScopes.Clear(); this.readOnlyBookmarkScopeCollection = null; this.PerformDefaultRegistration(); } protected override void OnInitialize(HandleInitializationContext context) { this.owningInstance = context.OwningActivityInstance; this.executor = context.Executor; PerformDefaultRegistration(); } internal ExclusiveHandleBookmarkList ImportantBookmarks { get { return this.importantBookmarks; } set { this.importantBookmarks = value; } } internal ExclusiveHandleBookmarkList UnimportantBookmarks { get { return this.unimportantBookmarks; } set { this.unimportantBookmarks = value; } } internal void AddToImportantBookmarks(Bookmark bookmark) { if (this.ImportantBookmarks == null) { this.ImportantBookmarks = new ExclusiveHandleBookmarkList(); } Fx.Assert(!this.ImportantBookmarks.Contains(bookmark), "We shouldnt be here. We attempted to add the same bookmark"); this.ImportantBookmarks.Add(bookmark); if (bookmark.ExclusiveHandles == null) { bookmark.ExclusiveHandles = new ExclusiveHandleList(); } Fx.Assert(!bookmark.ExclusiveHandles.Contains(this), "We shouldnt be here. We attempted to add the bookmark to this exclusive handle already"); bookmark.ExclusiveHandles.Add(this); } internal void AddToUnimportantBookmarks(Bookmark bookmark) { if (this.UnimportantBookmarks == null) { this.UnimportantBookmarks = new ExclusiveHandleBookmarkList(); } Fx.Assert(!this.UnimportantBookmarks.Contains(bookmark), "We shouldnt be here. We attempted to add the same bookmark"); this.UnimportantBookmarks.Add(bookmark); if (bookmark.ExclusiveHandles == null) { bookmark.ExclusiveHandles = new ExclusiveHandleList(); } Fx.Assert(!bookmark.ExclusiveHandles.Contains(this), "We shouldnt be here. We attempted to add the bookmark to this exclusive handle already"); bookmark.ExclusiveHandles.Add(this); } internal void RemoveBookmark(Bookmark bookmark) { Fx.Assert((this.ImportantBookmarks != null && this.ImportantBookmarks.Contains(bookmark)) || (this.UnimportantBookmarks != null && this.UnimportantBookmarks.Contains(bookmark)), "Internal error"); if (this.ImportantBookmarks != null) { if (this.ImportantBookmarks.Contains(bookmark)) { this.ImportantBookmarks.Remove(bookmark); return; } } if (this.UnimportantBookmarks != null) { if (this.UnimportantBookmarks.Contains(bookmark)) { this.UnimportantBookmarks.Remove(bookmark); } } } void PerformDefaultRegistration() { if (this.bookmarkScopes == null) { this.bookmarkScopes = new List(); } //First register the default subinstance this.bookmarkScopes.Add(BookmarkScopeHandle.Default); // Note that we are starting the LocationEnvironment traversal from the current environment's Parent. We don't // want to include any BookmarkScopeHandles that are at the same scope level as the ExclusiveHandle. The ExclusiveHandle // should only be dependent on BookmarkScopeHandles that are higher in the scope tree. LocationEnvironment current = this.owningInstance.Environment; if (current != null) { for (current = current.Parent; current != null; current = current.Parent) { //don't bother continuing if at this level there are no handles if (!current.HasHandles) { continue; } // Look at the contained handles for the environment. List handles = current.Handles; if (handles != null) { int count = handles.Count; for (int i = 0; i < count; i++) { BookmarkScopeHandle scopeHandle = handles[i] as BookmarkScopeHandle; if (scopeHandle != null) { this.bookmarkScopes.Add(scopeHandle); } } } } } // Also need to look in the Executor for handles that may have been created without an environment. List executorHandles = this.executor.Handles; if (executorHandles != null) { int count = executorHandles.Count; for (int i = 0; i < count; i++) { BookmarkScopeHandle scopeHandle = executorHandles[i] as BookmarkScopeHandle; if (scopeHandle != null) { this.bookmarkScopes.Add(scopeHandle); } } } this.bookmarkScopesListIsDefault = true; } // Exclusive handle needs to track bookmarks such that it can tell the difference between two bookmarks in // different bookmark scopes with the same name. Since we always deal in terms of the internal bookmark // reference that we have, we can do an object.ReferenceEquals comparison to determine distinct bookmarks // without having to add some sort of "containing scope" property to Bookmark. [DataContract] internal class ExclusiveHandleBookmarkList { List bookmarks; public ExclusiveHandleBookmarkList() : base() { this.bookmarks = new List(); } public int Count { get { return this.bookmarks.Count; } } [DataMember(Name = "bookmarks")] internal List SerializedBookmarks { get { return this.bookmarks; } set { this.bookmarks = value; } } public void Add(Bookmark bookmark) { Fx.Assert(bookmark != null, "A valid bookmark is expected."); this.bookmarks.Add(bookmark); } public void Remove(Bookmark bookmark) { Fx.Assert(bookmark != null, "A valid bookmark is expected."); for (int i = 0; i < this.bookmarks.Count; i++) { if (object.ReferenceEquals(this.bookmarks[i], bookmark)) { this.bookmarks.RemoveAt(i); return; } } } public bool Contains(Bookmark bookmark) { Fx.Assert(bookmark != null, "A valid bookmark is expected."); for (int i = 0; i < this.bookmarks.Count; i++) { if (object.ReferenceEquals(this.bookmarks[i], bookmark)) { return true; } } return false; } } } }