Files
simulation_core/tests/unit-tests/AdHocCompiler/UnsafePeripheral.cs

145 lines
5.3 KiB
C#

//
// Copyright (c) 2010-2026 Antmicro
//
// This file is licensed under the MIT License.
// Full license text is available in 'licenses/MIT.txt'.
//
using System;
using System.Linq;
using System.Threading;
using Antmicro.Renode.Core;
using Antmicro.Renode.Core.Structure.Registers;
using Antmicro.Renode.Utilities;
namespace Antmicro.Renode.Peripherals
{
public class UnsafePeripheral : BasicDoubleWordPeripheral, IKnownSize
{
public UnsafePeripheral(Machine machine) : base(machine)
{
Registers.Base.Define(this)
.WithValueField(0, 32, out value, FieldMode.Write, writeCallback: (_, val) =>
{
if(val.ToString() != IntToString((int)val))
{
// Modify the value to fail the test.
value.Value = val + 1;
}
// Create two arrays of the same length.
int length = 100;
byte[] byteArray1 = new byte[length];
byte[] byteArray2 = new byte[length];
// Fill byteArray1 with 0 - 99.
for(int i = 0; i < length; ++i)
{
byteArray1[i] = (byte)i;
}
// Copy the contents of byteArray1 to byteArray2.
Copy(byteArray1, 0, byteArray2, 0, length);
if(byteArray2[8] != byteArray1[8])
{
// Modify the value to fail the test.
value.Value = val + 1;
}
// Copy the contents of the last 10 elements of byteArray1 to the beginning of byteArray2.
// The offset specifies where the copying begins in the source array.
int offset = length - 10;
Copy(byteArray1, offset, byteArray2, 0, length - offset);
if(byteArray2[0] != byteArray1[90])
{
// Modify the value to fail the test.
value.Value = val + 1;
}
});
Registers.Multiplier.Define(this)
.WithValueField(0, 32, FieldMode.Read, valueProviderCallback: _ => value.Value * 2);
Registers.BitCounter.Define(this)
.WithValueField(0, 32, FieldMode.Read, valueProviderCallback: _ => (uint)BitHelper.GetBits(value.Value).Where(x => x).Select(x => 1).Sum(x => x));
}
// https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/unsafe-code#how-to-use-pointers-to-copy-an-array-of-bytes
private static unsafe void Copy(byte[] source, int sourceOffset, byte[] target,
int targetOffset, int count)
{
// If either array is not instantiated, you cannot complete the copy.
if((source == null) || (target == null))
{
throw new ArgumentException("source or target is null");
}
// If either offset, or the number of bytes to copy, is negative, you
// cannot complete the copy.
if((sourceOffset < 0) || (targetOffset < 0) || (count < 0))
{
throw new ArgumentException("offset or bytes to copy is negative");
}
// If the number of bytes from the offset to the end of the array is
// less than the number of bytes you want to copy, you cannot complete
// the copy.
if((source.Length - sourceOffset < count) ||
(target.Length - targetOffset < count))
{
throw new ArgumentException("offset to end of array is less than bytes to be copied");
}
// The following fixed statement pins the location of the source and
// target objects in memory so that they will not be moved by garbage
// collection.
fixed(byte* pSource = source, pTarget = target)
{
// Copy the specified number of bytes from source to target.
for(int i = 0; i < count; i++)
{
pTarget[targetOffset + i] = pSource[sourceOffset + i];
}
}
}
// https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/language-specification/unsafe-code#249-stack-allocation
private static string IntToString(int value)
{
if(value == int.MinValue)
{
return "-2147483648";
}
int n = value >= 0 ? value : -value;
unsafe
{
char* buffer = stackalloc char[16];
char* p = buffer + 16;
do
{
*--p = (char)(n % 10 + '0');
n /= 10;
} while (n != 0);
if(value < 0)
{
*--p = '-';
}
return new string(p, 0, (int)(buffer + 16 - p));
}
}
public long Size { get { return 0x100; } }
private IValueRegisterField value;
private enum Registers : long
{
Base = 0x0,
Multiplier = 0x04,
BitCounter = 0x08,
}
}
}