-
-
Notifications
You must be signed in to change notification settings - Fork 111
/
Copy pathAt24Base.cs
207 lines (179 loc) · 7.77 KB
/
At24Base.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System;
using System.Device.I2c;
namespace Iot.Device.At24cxx
{
/// <summary>
/// Base class for common functionality of the At24c serices of devices.
/// </summary>
public abstract class At24Base : IDisposable
{
/// <summary>
/// Default I2C address for At24cxx familly.
/// </summary>
public const byte DefaultI2cAddress = 0x50;
/// <summary>
/// The underlying I2C device used for communication.
/// </summary>
private readonly I2cDevice _i2cDevice;
/// <summary>
/// Gets the number of pages available on the device.
/// </summary>
public int PageCount { get; private set; }
/// <summary>
/// Gets the device page size (in bytes).
/// </summary>
public int PageSize { get; private set; }
/// <summary>
/// Gets the available memory on the device (in bytes).
/// </summary>
public int Size { get; private set; }
/// <summary>
/// Initializes a new instance of the <see cref="At24Base" /> class.
/// </summary>
/// <param name="i2cDevice">The I2C device to use for communication.</param>
/// <param name="pageSize">The device page size in bytes.</param>
/// <param name="pageCount">The number of pages on the device.</param>
/// <exception cref="ArgumentNullException">
/// Thrown when <paramref name="i2cDevice"/> is null.
/// </exception>
internal At24Base(I2cDevice i2cDevice, ushort pageSize, ushort pageCount)
{
if (i2cDevice == null)
{
throw new ArgumentNullException();
}
_i2cDevice = i2cDevice;
PageSize = pageSize;
PageCount = pageCount;
Size = pageSize * pageCount;
}
/// <summary>
/// Helper function to create a buffer pre-populated with the device address.
/// </summary>
/// <param name="address">The device address.</param>
/// <param name="dataLength">The length of data the buffer is required to hold.</param>
/// <returns>
/// A buffer with the device address occupying the first two bytes.
/// </returns>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="address"/> falls outside of the addressable range for this device.
/// </exception>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="dataLength"/> exceeds the page size for this device.
/// </exception>
protected byte[] CreateWriteBuffer(int address, int dataLength)
{
if (address < 0 || address > (Size - 1))
{
throw new ArgumentOutOfRangeException();
}
if (dataLength > PageSize)
{
throw new ArgumentOutOfRangeException();
}
// Create buffer with an additional two bytes to hold the address.
byte[] buffer = new byte[dataLength + 2];
// At24cxx requires the memory address to be in the first two bytes preceding data.
buffer[0] = (byte)((address >> 8) & 0xFF);
buffer[1] = (byte)(address & 0xFF);
return buffer;
}
/// <summary>
/// Reads a single byte from the device at the address following the last byte read or written.
/// </summary>
/// <returns>
/// The byte read from the device.
/// </returns>
public byte ReadByte()
{
return _i2cDevice.ReadByte();
}
/// <summary>
/// Reads a single byte from the device at the given address.
/// </summary>
/// <param name="address">The address to read from.</param>
/// <returns>
/// The byte read from the device.
/// </returns>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="address"/> falls outside of the addressable range for this device.
/// </exception>
public byte ReadByte(int address)
{
byte[] writeBuffer = CreateWriteBuffer(address, 0);
byte[] readBuffer = new byte[1];
_i2cDevice.WriteRead(writeBuffer, readBuffer);
return readBuffer[0];
}
/// <summary>
/// Reads multiple bytes from the device starting from the given address.
/// </summary>
/// <param name="address">The address to read from.</param>
/// <param name="length">The number of bytes to read.</param>
/// <returns>
/// The bytes read from the device.
/// </returns>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="address"/> falls outside of the addressable range for this device.
/// </exception>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when the length of data to be read is zero or will cause the data read to fall outside of the addressable range for this device.
/// </exception>
public byte[] Read(int address, int length)
{
if (length <= 0 || (address + length) > (Size - 1))
{
throw new ArgumentOutOfRangeException();
}
byte[] writeBuffer = CreateWriteBuffer(address, 0);
byte[] readBuffer = new byte[length];
_i2cDevice.WriteRead(writeBuffer, readBuffer);
return readBuffer;
}
/// <summary>
/// Writes a single byte to the device at the given address.
/// </summary>
/// <param name="address">The address to write to.</param>
/// <param name="value">The byte to be written into the device.</param>
/// <returns>
/// The number of bytes successfully written to the device.
/// </returns>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="address"/> falls outside of the addressable range for this device.
/// </exception>
public uint WriteByte(int address, byte value)
{
byte[] writeBuffer = CreateWriteBuffer(address, 1);
writeBuffer[2] = value;
I2cTransferResult result = _i2cDevice.Write(writeBuffer);
// Account for memory address that was sent to the device along with the data to be written
return result.BytesTransferred - 2;
}
/// <summary>
/// Writes multiple bytes to the device starting from the given address.
/// </summary>
/// <param name="address">The address to write to.</param>
/// <param name="data">The bytes to be written to the device.</param>
/// <returns>The number of bytes successfully written to the device.</returns>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when <paramref name="address"/> falls outside of the addressable range for this device.
/// </exception>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when length of data to be written exceeds the page size for this device.
/// </exception>
public uint Write(int address, byte[] data)
{
byte[] writeBuffer = CreateWriteBuffer(address, data.Length);
data.CopyTo(writeBuffer, 2);
I2cTransferResult result = _i2cDevice.Write(writeBuffer);
// Account for memory address that was sent to the device along with the data to be written
return result.BytesTransferred - 2;
}
/// <inheritdoc />
public void Dispose()
{
}
}
}