Skip to content

Commit f327038

Browse files
committed
added many safeguards for thread safety but ultimately failed, committing changes before switching to an alternative solution
1 parent 2213474 commit f327038

File tree

7 files changed

+122
-95
lines changed

7 files changed

+122
-95
lines changed

GPUPrefSwitcher/AppEntry.cs

+5-6
Original file line numberDiff line numberDiff line change
@@ -94,21 +94,20 @@ public override readonly int GetHashCode()
9494
hashCode = hashCode * -1521134295 + EqualityComparer<string>.Default.GetHashCode(AppName);
9595
hashCode = hashCode * -1521134295 + EnableSwitcher.GetHashCode();
9696
hashCode = hashCode * -1521134295 + EnableFileSwapper.GetHashCode();
97-
hashCode = hashCode * -1521134295 + GetArrHash(FileSwapperPaths);
98-
hashCode = hashCode * -1521134295 + GetArrHash(FileSwapperPaths);
97+
hashCode = hashCode * -1521134295 + GetStrArrHash(FileSwapperPaths);
9998
hashCode = hashCode * -1521134295 + GPUPrefOnBattery.GetHashCode();
10099
hashCode = hashCode * -1521134295 + GPUPrefPluggedIn.GetHashCode();
101100
hashCode = hashCode * -1521134295 + RunOnBatteryPath.GetHashCode();
102101
hashCode = hashCode * -1521134295 + RunPluggedInPath.GetHashCode();
103102
hashCode = hashCode * -1521134295 + PendingAddToRegistry.GetHashCode();
104103
hashCode = hashCode * -1521134295 + SeenInRegistry.GetHashCode();
105-
hashCode = hashCode * -1521134295 + GetArrHash(from s in SwapperStates select s.ToString()); //a bit hacky but it should work
104+
hashCode = hashCode * -1521134295 + GetStrArrHash(from s in SwapperStates select s.ToString()); //a bit hacky but it should work
106105

107106
//TODO: need for AppName
108107
return hashCode;
109108
}
110109

111-
public static int GetArrHash(IEnumerable<object> objs)
110+
public static int GetStrArrHash(IEnumerable<string> strs)
112111
{
113112
int hash = -335392656;
114113
/*
@@ -117,9 +116,9 @@ public static int GetArrHash(IEnumerable<object> objs)
117116
hash = hash * -130699793 + objs[i].GetHashCode();
118117
}
119118
*/
120-
foreach(object obj in objs)
119+
foreach(string s in strs)
121120
{
122-
hash = hash * -130699793 + obj.GetHashCode();
121+
hash = hash * -130699793 + strs.GetHashCode();
123122
}
124123
return hash;
125124
}

GPUPrefSwitcher/AppEntrySaveHandler.cs

+96-71
Original file line numberDiff line numberDiff line change
@@ -31,11 +31,11 @@ private List<AppEntry> prevAppEntries
3131
private List<AppEntry> prevAppEntries_;
3232
*/
3333

34-
public List<AppEntry> CurrentAppEntries
34+
public IReadOnlyList<AppEntry> CurrentAppEntries
3535
{
3636
get
3737
{
38-
return currentAppEntries;
38+
return currentAppEntries.AsReadOnly();
3939
}
4040
}
4141
private List<AppEntry> currentAppEntries;
@@ -94,46 +94,54 @@ public AppEntrySaveHandler()
9494
} finally { semaphoreSlim.Release(); }
9595

9696

97-
Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
98-
Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
97+
//Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
98+
//Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
9999
}
100100

101-
public void ChangeAppEntryByPath(string path, AppEntry updatedAppEntry)
101+
public void ChangeAppEntryByPathAndSave(string path, AppEntry updatedAppEntry)
102102
{
103103
try
104104
{
105+
Logger.inst.Log($"ChangeAppEntryByPath waiting on a semaphore ({path})");
105106
semaphoreSlim.Wait();
107+
Logger.inst.Log($"ChangeAppEntryByPath entered a semaphore ({path}).");
106108

107-
Logger.inst.Log($"prev: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
109+
Logger.inst.Log($"Attempting to change \n {currentAppEntries.Single(x=>x.AppPath==path)}\n to \n {updatedAppEntry}");
108110

109-
int index = CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path));
110-
111+
//Logger.inst.Log($"prev: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
111112

112-
/* //for-loop alternative, but the above should throw an error with an obvious enough meaning
113-
int index = -1;
114-
for (int i = 0; i < CurrentAppEntries.Count; i++)
115-
{
116-
AppEntry appEntry = CurrentAppEntries[i];
117-
if (appEntry.AppPath == path)
118-
{
119-
index = i; break;
120-
}
121-
}
113+
int index = currentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path));
122114

123-
if(index<0)
124-
throw new AppEntrySaverException($"UpdateAppEntry: No AppEntry with path {path} was found");
125-
*/
115+
currentAppEntries[index] = updatedAppEntry;
126116

127-
CurrentAppEntries[index] = updatedAppEntry;
117+
SaveAppEntryChanges_Internal();
128118

129-
Logger.inst.Log($"new: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
119+
//Logger.inst.Log($"new: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
130120
} finally
131121
{
132-
Logger.inst.Log("ChangeAppEntryByPath released a semaphore.");
122+
Logger.inst.Log($"ChangeAppEntryByPath released a semaphore ({path})");
133123
semaphoreSlim.Release();
134124
}
135125
}
136126

127+
public int RemoveAll(Predicate<AppEntry> predicate)
128+
{
129+
return currentAppEntries.RemoveAll(predicate);
130+
}
131+
132+
public void AddAppEntryAndSave(AppEntry appEntry)
133+
{
134+
try
135+
{
136+
semaphoreSlim.Wait();
137+
138+
currentAppEntries.Add(appEntry);
139+
140+
SaveAppEntryChanges_Internal();
141+
142+
}
143+
finally { semaphoreSlim.Release(); }
144+
}
137145
/*
138146
* DECISION: Replacement or overwrite?
139147
* Per-Entry Replacement (SELECTED):
@@ -147,71 +155,88 @@ public void ChangeAppEntryByPath(string path, AppEntry updatedAppEntry)
147155
*
148156
* UPDATE: Only per-entry eplacement is viable because the XML can still contain stuff that isn't in the registry!!
149157
*/
158+
/// <summary>
159+
/// NOT THREAD SAFE
160+
/// </summary>
150161
public void SaveAppEntryChanges()
151162
{
152163
try
153164
{
154165
semaphoreSlim.Wait();
155166

156-
//Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
157-
//Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
158-
List<AppEntry> differences = new();
159-
differences.AddRange(currentAppEntries.Where(entry => NotSameOrInPrevAppEntries(entry)));
160-
161-
List<string> existingAppPaths = new List<string>(from appEntry in PreferencesXML.GetAppEntries() select appEntry.AppPath);
162-
List<AppEntry> needToAdd = new();
163-
needToAdd.AddRange(currentAppEntries.Where(entry => !existingAppPaths.Contains(entry.AppPath)));
164-
165-
//remove appentries whose *paths* exist in the prev but not the current
166-
List<AppEntry> needToRemoveFromXML = new();
167-
foreach (AppEntry entry in prevAppEntries)
168-
{
169-
if (!currentAppEntries.Exists(a => a.AppPath == entry.AppPath))
170-
{
171-
needToRemoveFromXML.Add(entry);
172-
}
173-
}
167+
SaveAppEntryChanges_Internal();
168+
169+
} finally
170+
{
171+
semaphoreSlim.Release();
172+
}
173+
}
174174

175-
//new AppEntries should be added first so that ModifyAppEntry can actually find the AppEntry with the path
176-
foreach (AppEntry appEntry in needToAdd)
175+
private void SaveAppEntryChanges_Internal()
176+
{
177+
//Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
178+
//Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
179+
List<AppEntry> differences = new();
180+
differences.AddRange(currentAppEntries.Where(entry => NotSameOrInPrevAppEntries(entry)));
181+
182+
List<string> existingAppPaths = new List<string>(from appEntry in PreferencesXML.GetAppEntries() select appEntry.AppPath);
183+
List<AppEntry> needToAdd = new();
184+
needToAdd.AddRange(currentAppEntries.Where(entry => !existingAppPaths.Contains(entry.AppPath)));
185+
186+
//remove appentries whose *paths* exist in the prev but not the current
187+
List<AppEntry> needToRemoveFromXML = new();
188+
foreach (AppEntry entry in prevAppEntries)
189+
{
190+
if (!currentAppEntries.Exists(a => a.AppPath == entry.AppPath))
177191
{
178-
PreferencesXML.AddAppEntryAndSave(appEntry);
192+
needToRemoveFromXML.Add(entry);
179193
}
194+
}
180195

181-
foreach (AppEntry appEntry in differences)
182-
{
183-
PreferencesXML.ModifyAppEntryAndSave(appEntry.AppPath, appEntry);
184-
}
196+
//new AppEntries should be added first so that ModifyAppEntry can actually find the AppEntry with the path
197+
foreach (AppEntry appEntry in needToAdd)
198+
{
199+
PreferencesXML.AddAppEntryAndSave(appEntry);
200+
}
185201

186-
foreach (AppEntry appEntry in needToRemoveFromXML)
187-
{
188-
bool success = PreferencesXML.TryDeleteAppEntryAndSave(appEntry.AppPath);
189-
if (!success) //this would probably only ever fail if AppEntries were deleted externally whilst the GUI or Service was still running
190-
throw new XMLHelperException($"DeleteAppEntry: AppEntry with the specified path not found in data store: {appEntry.AppPath}");
191-
}
202+
foreach (AppEntry appEntry in differences)
203+
{
204+
PreferencesXML.ModifyAppEntryAndSave(appEntry.AppPath, appEntry);
205+
}
192206

193-
prevAppEntries = DeepCopyAppEntries(currentAppEntries); //update the saved entries
207+
foreach (AppEntry appEntry in needToRemoveFromXML)
208+
{
209+
bool success = PreferencesXML.TryDeleteAppEntryAndSave(appEntry.AppPath);
210+
if (!success) //this would probably only ever fail if AppEntries were deleted externally whilst the GUI or Service was still running
211+
throw new XMLHelperException($"DeleteAppEntry: AppEntry with the specified path not found in data store: {appEntry.AppPath}");
212+
}
194213

195-
//Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
196-
//Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
214+
prevAppEntries = DeepCopyAppEntries(currentAppEntries); //update the saved entries
197215

198-
Logger.inst.Log($"Concluded saving. Differences: {differences.Count} Added: {needToAdd.Count} Removed: {needToRemoveFromXML.Count}");
216+
//Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
217+
//Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
199218

200-
bool NotSameOrInPrevAppEntries(AppEntry appEntry)
219+
Logger.inst.Log($"Concluded saving. Differences: {differences.Count} Added: {needToAdd.Count} Removed: {needToRemoveFromXML.Count}");
220+
if(differences.Count > 0)
221+
{
222+
Logger.inst.Log("Changed:");
223+
foreach(AppEntry appEntry in differences)
201224
{
202-
/*
203-
if (appEntry.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe")
204-
{
205-
Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
206-
Logger.inst.Log(CurrentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
207-
Logger.inst.Log($"{appEntry} not in prev app entries: {!prevAppEntries.Contains(appEntry)}");
208-
}
209-
*/
210-
return !prevAppEntries.Contains(appEntry); //Contains uses Equals() which is implemented in AppEntry
225+
Logger.inst.Log(appEntry.ToString());
211226
}
212-
} finally
227+
}
228+
229+
bool NotSameOrInPrevAppEntries(AppEntry appEntry)
213230
{
214-
semaphoreSlim.Release();
231+
/*
232+
if (appEntry.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe")
233+
{
234+
Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
235+
Logger.inst.Log(CurrentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
236+
Logger.inst.Log($"{appEntry} not in prev app entries: {!prevAppEntries.Contains(appEntry)}");
237+
}
238+
*/
239+
return !prevAppEntries.Contains(appEntry); //Contains uses Equals() which is implemented in AppEntry
215240
}
216241
}
217242

GPUPrefSwitcher/AppLogger.cs

+1
Original file line numberDiff line numberDiff line change
@@ -229,6 +229,7 @@ public void Log(string str, int logLevel = 1000)
229229
{
230230
semaphoreSlim_Standard.AvailableWaitHandle.WaitOne();//block if there's still a write in progress
231231
_ = WriteAsyncInternal_Standard(toWrite);
232+
//WriteAsyncInternal_Standard(toWrite).Wait();
232233
}
233234

234235
StandardLogBuffer.Add(toWrite);

GPUPrefSwitcher/FileSwapper.cs

+5-4
Original file line numberDiff line numberDiff line change
@@ -258,9 +258,10 @@ private async Task InitiateSingleFileSwap(FileSwapPathTask fileSwapPathTask, int
258258
AppEntry modified = AppEntry; //struct copy
259259
modified.SwapperStates[swapPathIndex] = PowerLineStatus.Offline;
260260

261+
Logger.inst.Log($"Saving SwapPath state for SwapPath {swapPath} for app {AppEntry.AppPath}");
261262
//await new Task(new Action( () => preferencesXML.ModifyAppEntryAndSave(AppEntry.AppPath, modified)));
262-
AppEntrySaveHandler.ChangeAppEntryByPath(AppEntry.AppPath, modified);
263-
AppEntrySaveHandler.SaveAppEntryChanges();
263+
AppEntrySaveHandler.ChangeAppEntryByPathAndSave(AppEntry.AppPath, modified);
264+
//AppEntrySaveHandler.SaveAppEntryChanges();
264265

265266
string s3 = $"Saved SwapPath state for SwapPath {swapPath} for app {AppEntry.AppPath}";
266267
Logger.inst.Log(s3);
@@ -306,8 +307,8 @@ private async Task InitiateSingleFileSwap(FileSwapPathTask fileSwapPathTask, int
306307
AppEntry modified = AppEntry; //struct copy
307308
modified.SwapperStates[swapPathIndex] = PowerLineStatus.Online;
308309

309-
AppEntrySaveHandler.ChangeAppEntryByPath(AppEntry.AppPath, modified);
310-
AppEntrySaveHandler.SaveAppEntryChanges();
310+
AppEntrySaveHandler.ChangeAppEntryByPathAndSave(AppEntry.AppPath, modified);
311+
//AppEntrySaveHandler.SaveAppEntryChanges();
311312
string s3 = $"Saved SwapPath state for SwapPath {swapPath} for app {AppEntry.AppPath}";
312313
Logger.inst.Log(s3);
313314
} else

GPUPrefSwitcher/PreferencesXML.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -409,6 +409,7 @@ internal void ModifyAppEntryAndSave(string path, AppEntry newAppEntry) //TODO fi
409409
string newSeenInRegistry = newAppEntry.SeenInRegistry.ToString().ToLower();
410410

411411
string[] newFileSwapperPaths = newAppEntry.FileSwapperPaths;
412+
PowerLineStatus[] newSwapperStates = newAppEntry.SwapperStates;
412413

413414
if (newAppPath == null)
414415
throw new InvalidOperationException("AppEntry passed in had a null AppPath value");
@@ -481,7 +482,7 @@ internal void ModifyAppEntryAndSave(string path, AppEntry newAppEntry) //TODO fi
481482

482483
XmlElement xmlElement = xmlDocument.CreateElement(XML_SWAP_PATH);
483484

484-
xmlElement.SetAttribute(XML_ATTR_SWAPPATHSTATUS, PowerLineStatusConversions.PowerLineStatusToOfflineOrOnline(newAppEntry.SwapperStates[i]));//TODO: does this gauruntee order?
485+
xmlElement.SetAttribute(XML_ATTR_SWAPPATHSTATUS, PowerLineStatusConversions.PowerLineStatusToOfflineOrOnline(newSwapperStates[i]));//TODO: does this gauruntee order?
485486

486487
xmlElement.InnerText = newFileSwapperPath;
487488

0 commit comments

Comments
 (0)