342 lines
7.2 KiB
C#
342 lines
7.2 KiB
C#
using BizHawk.Common.NumberExtensions;
|
|
using BizHawk.Common.StringExtensions;
|
|
using BizHawk.Emulation.Common;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Text;
|
|
|
|
namespace BizHawk.Client.Common
|
|
{
|
|
/// <summary>
|
|
/// This class holds a double word (32 bits) <see cref="Watch"/>
|
|
/// </summary>
|
|
public sealed class DWordWatch : Watch
|
|
{
|
|
#region Fields
|
|
|
|
private uint _value;
|
|
private uint _previous;
|
|
|
|
#endregion
|
|
|
|
#region cTor(s)
|
|
|
|
/// <summary>
|
|
/// Inialize a new instance of <see cref="DWordWatch"/>
|
|
/// </summary>
|
|
/// <param name="domain"><see cref="MemoryDomain"/> where you want to track</param>
|
|
/// <param name="address">The address you want to track</param>
|
|
/// <param name="type">How you you want to display the value See <see cref="DisplayType"/></param>
|
|
/// <param name="bigEndian">Specify the endianess. true for big endian</param>
|
|
/// <param name="note">A custom note about the <see cref="Watch"/></param>
|
|
/// <param name="value">Current value</param>
|
|
/// <param name="previous">Previous value</param>
|
|
/// <param name="changeCount">How many times value has changed</param>
|
|
/// <exception cref="ArgumentException">Occurs when a <see cref="DisplayType"/> is incompatible with <see cref="WatchSize.DWord"/></exception>
|
|
internal DWordWatch(MemoryDomain domain, long address, DisplayType type, bool bigEndian, string note, uint value, uint previous, int changeCount)
|
|
: base(domain, address, WatchSize.DWord, type, bigEndian, note)
|
|
{
|
|
if (value == 0)
|
|
{
|
|
this._value = GetDWord();
|
|
}
|
|
else
|
|
{
|
|
this._value = value;
|
|
}
|
|
this._previous = previous;
|
|
this._changecount = changeCount;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Methods
|
|
|
|
/// <summary>
|
|
/// Enumerate wich <see cref="DisplayType"/> are valid for a <see cref="DWordWatch"/>
|
|
/// </summary>
|
|
public static IEnumerable<DisplayType> ValidTypes
|
|
{
|
|
get
|
|
{
|
|
yield return DisplayType.Unsigned;
|
|
yield return DisplayType.Signed;
|
|
yield return DisplayType.Hex;
|
|
yield return DisplayType.Binary;
|
|
yield return DisplayType.FixedPoint_20_12;
|
|
yield return DisplayType.FixedPoint_16_16;
|
|
yield return DisplayType.Float;
|
|
}
|
|
}
|
|
|
|
#region Implements
|
|
|
|
/// <summary>
|
|
/// Get a list a <see cref="DisplayType"/> that can be used for this <see cref="DWordWatch"/>
|
|
/// </summary>
|
|
/// <returns>An enumartion that contains all valid <see cref="DisplayType"/></returns>
|
|
public override IEnumerable<DisplayType> AvailableTypes()
|
|
{
|
|
return ValidTypes;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reset the previous value; set it to the current one
|
|
/// </summary>
|
|
public override void ResetPrevious()
|
|
{
|
|
_previous = GetWord();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Try to sets the value into the <see cref="MemoryDomain"/>
|
|
/// at the current <see cref="Watch"/> address
|
|
/// </summary>
|
|
/// <param name="value">Value to set</param>
|
|
/// <returns>True if value successfully sets; othewise, false</returns>
|
|
public override bool Poke(string value)
|
|
{
|
|
try
|
|
{
|
|
uint val = 0;
|
|
switch (Type)
|
|
{
|
|
case DisplayType.Unsigned:
|
|
if (value.IsUnsigned())
|
|
{
|
|
val = (uint)int.Parse(value);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
case DisplayType.Signed:
|
|
if (value.IsSigned())
|
|
{
|
|
val = (uint)int.Parse(value);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
case DisplayType.Hex:
|
|
if (value.IsHex())
|
|
{
|
|
val = (uint)int.Parse(value, NumberStyles.HexNumber);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
case DisplayType.FixedPoint_20_12:
|
|
if (value.IsFixedPoint())
|
|
{
|
|
val = (uint)(int)(double.Parse(value) * 4096.0);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
case DisplayType.FixedPoint_16_16:
|
|
if (value.IsFixedPoint())
|
|
{
|
|
val = (uint)(int)(double.Parse(value) * 65536.0);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
case DisplayType.Float:
|
|
if (value.IsFloat())
|
|
{
|
|
var bytes = BitConverter.GetBytes(float.Parse(value));
|
|
val = BitConverter.ToUInt32(bytes, 0);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if (Global.CheatList.Contains(Domain, _address))
|
|
{
|
|
var cheat = Global.CheatList.FirstOrDefault(c => c.Address == _address && c.Domain == Domain);
|
|
if (cheat != (Cheat)null)
|
|
{
|
|
cheat.PokeValue((int)val);
|
|
PokeDWord(val);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
PokeDWord(val);
|
|
return true;
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update the Watch (read it from <see cref="MemoryDomain"/>
|
|
/// </summary>
|
|
public override void Update()
|
|
{
|
|
switch (Global.Config.RamWatchDefinePrevious)
|
|
{
|
|
case PreviousType.Original:
|
|
return;
|
|
case PreviousType.LastChange:
|
|
var temp = _value;
|
|
_value = GetDWord();
|
|
if (_value != temp)
|
|
{
|
|
_previous = _value;
|
|
_changecount++;
|
|
}
|
|
|
|
break;
|
|
case PreviousType.LastFrame:
|
|
_previous = _value;
|
|
_value = GetDWord();
|
|
if (_value != Previous)
|
|
{
|
|
_changecount++;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
#endregion Implements
|
|
|
|
//TODO: Implements IFormattable
|
|
public string FormatValue(uint val)
|
|
{
|
|
switch (Type)
|
|
{
|
|
default:
|
|
case DisplayType.Unsigned:
|
|
return val.ToString();
|
|
case DisplayType.Signed:
|
|
return ((int)val).ToString();
|
|
case DisplayType.Hex:
|
|
return val.ToHexString(8);
|
|
case DisplayType.FixedPoint_20_12:
|
|
return string.Format("{0:0.######}", val / 4096.0);
|
|
case DisplayType.FixedPoint_16_16:
|
|
return string.Format("{0:0.######}", val / 65536.0);
|
|
case DisplayType.Float:
|
|
var bytes = BitConverter.GetBytes(val);
|
|
var _float = BitConverter.ToSingle(bytes, 0);
|
|
//return string.Format("{0:0.######}", _float);
|
|
return _float.ToString(); // adelikat: decided that we like sci notation instead of spooky rounding
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Properties
|
|
|
|
#region Implements
|
|
|
|
/// <summary>
|
|
/// Get a string representation of difference
|
|
/// between current value and the previous one
|
|
/// </summary>
|
|
public override string Diff
|
|
{
|
|
get
|
|
{
|
|
return FormatValue(_previous - _value);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the maximum possible value
|
|
/// </summary>
|
|
public override uint MaxValue
|
|
{
|
|
get
|
|
{
|
|
return uint.MaxValue;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the current value
|
|
/// </summary>
|
|
public override int Value
|
|
{
|
|
get
|
|
{
|
|
return (int)GetDWord();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the current value
|
|
/// but with stuff I don't understand
|
|
/// </summary>
|
|
public override int ValueNoFreeze
|
|
{
|
|
get
|
|
{
|
|
return (int)GetDWord(true);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get a string representation of the current value
|
|
/// </summary>
|
|
public override string ValueString
|
|
{
|
|
get
|
|
{
|
|
return FormatValue(GetDWord());
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the previous value
|
|
/// </summary>
|
|
public override int Previous
|
|
{
|
|
get
|
|
{
|
|
return (int)_previous;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get a string representation of the previous value
|
|
/// </summary>
|
|
public override string PreviousStr
|
|
{
|
|
get
|
|
{
|
|
return FormatValue(_previous);
|
|
}
|
|
}
|
|
|
|
#endregion Implements
|
|
|
|
#endregion
|
|
}
|
|
}
|