Files
V2GDecoderC/csharp/dotnetfx/EXI/BitInputStream.cs
ChiKyun Kim fe368f2d23 feat: Complete C# ports for both .NET 8.0 and .NET Framework 4.8
This commit adds comprehensive C# ports of the OpenV2G EXI codec
to support both modern .NET and legacy .NET Framework environments.

## .NET 8.0 Version (csharp/dotnet/)
- Full-featured port with complete EXI codec implementation
- Modern C# features (nullable types, switch expressions, using declarations)
- Comprehensive roundtrip testing functionality
- Successfully processes all test files (test1.exi - test5.exi)
- Supports decode/encode/analyze/test commands

## .NET Framework 4.8 Version (csharp/dotnetfx/)
- Simplified but functional port for legacy environments
- C# 7.3 compatible codebase
- Core V2GTP protocol parsing and analysis
- Roundtrip demonstration functionality
- Successfully processes all test files

## Validation Results
Both versions successfully tested with all available test files:
- test1.exi (131 bytes) → XML → EXI roundtrip ✓
- test2.exi (51 bytes) → XML → EXI roundtrip ✓
- test3.exi (43 bytes) → XML → EXI roundtrip ✓
- test4.exi (43 bytes) → XML → EXI roundtrip ✓
- test5.exi (43 bytes) → XML → EXI roundtrip ✓

## Technical Implementation
- Proper V2GTP header parsing and EXI body extraction
- XML generation with valid structure for testing
- Binary EXI encoding for roundtrip validation
- Cross-platform compatibility maintained
- Build systems: dotnet CLI (.NET 8.0) and MSBuild (.NET FX 4.8)

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-10 09:54:53 +09:00

219 lines
6.4 KiB
C#

/*
* Copyright (C) 2007-2024 C# Port
* Original Copyright (C) 2007-2018 Siemens AG
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*/
using System;
namespace V2GDecoderNetFx.EXI
{
/// <summary>
/// Bit input stream for reading EXI encoded data
/// </summary>
public class BitInputStream
{
private readonly byte[] _buffer;
private int _position;
private int _bitPosition;
private readonly int _size;
public BitInputStream(byte[] buffer)
{
if (buffer == null)
throw new ArgumentNullException(nameof(buffer));
_buffer = buffer;
_size = buffer.Length;
_position = 0;
_bitPosition = 0;
}
public int Position => _position;
public int BitPosition => _bitPosition;
public int Size => _size;
public bool IsEOF => _position >= _size;
/// <summary>
/// Read a single bit
/// </summary>
/// <returns>Bit value (0 or 1), or -1 on EOF</returns>
public int ReadBit()
{
if (_position >= _size)
return -1;
int bit = (_buffer[_position] >> (7 - _bitPosition)) & 1;
_bitPosition++;
if (_bitPosition == 8)
{
_bitPosition = 0;
_position++;
}
return bit;
}
/// <summary>
/// Read multiple bits as unsigned integer
/// </summary>
/// <param name="numBits">Number of bits to read (1-32)</param>
/// <returns>Unsigned integer value</returns>
public uint ReadBits(int numBits)
{
if (numBits < 1 || numBits > 32)
throw new ArgumentException("Number of bits must be between 1 and 32", nameof(numBits));
uint result = 0;
for (int i = 0; i < numBits; i++)
{
int bit = ReadBit();
if (bit == -1)
throw new EXIException(EXIErrorCodes.EXI_ERROR_INPUT_STREAM_EOF);
result = (result << 1) | (uint)bit;
}
return result;
}
/// <summary>
/// Read unsigned integer using EXI encoding
/// </summary>
/// <returns>Unsigned integer value</returns>
public uint ReadUnsignedInteger()
{
uint result = 0;
bool continueBit;
do
{
if (_position >= _size)
throw new EXIException(EXIErrorCodes.EXI_ERROR_INPUT_STREAM_EOF);
byte currentByte = _buffer[_position++];
continueBit = (currentByte & 0x80) != 0;
result = (result << 7) | (uint)(currentByte & 0x7F);
} while (continueBit);
return result;
}
/// <summary>
/// Read signed integer using EXI encoding
/// </summary>
/// <returns>Signed integer value</returns>
public int ReadInteger()
{
uint unsignedValue = ReadUnsignedInteger();
// Check sign bit (LSB)
bool isNegative = (unsignedValue & 1) != 0;
int value = (int)(unsignedValue >> 1);
return isNegative ? -value : value;
}
/// <summary>
/// Read a byte aligned to byte boundary
/// </summary>
/// <returns>Byte value</returns>
public byte ReadByte()
{
// Align to byte boundary
if (_bitPosition != 0)
{
_bitPosition = 0;
_position++;
}
if (_position >= _size)
throw new EXIException(EXIErrorCodes.EXI_ERROR_INPUT_STREAM_EOF);
return _buffer[_position++];
}
/// <summary>
/// Read multiple bytes
/// </summary>
/// <param name="count">Number of bytes to read</param>
/// <returns>Byte array</returns>
public byte[] ReadBytes(int count)
{
if (count < 0)
throw new ArgumentException("Count cannot be negative", nameof(count));
// Align to byte boundary
if (_bitPosition != 0)
{
_bitPosition = 0;
_position++;
}
if (_position + count > _size)
throw new EXIException(EXIErrorCodes.EXI_ERROR_INPUT_STREAM_EOF);
var result = new byte[count];
Array.Copy(_buffer, _position, result, 0, count);
_position += count;
return result;
}
/// <summary>
/// Skip to next byte boundary
/// </summary>
public void AlignToByteBank()
{
if (_bitPosition != 0)
{
_bitPosition = 0;
_position++;
}
}
/// <summary>
/// Reset stream position to beginning
/// </summary>
public void Reset()
{
_position = 0;
_bitPosition = 0;
}
/// <summary>
/// Set stream position
/// </summary>
/// <param name="position">Byte position</param>
/// <param name="bitPosition">Bit position within byte (0-7)</param>
public void SetPosition(int position, int bitPosition = 0)
{
if (position < 0 || position > _size)
throw new ArgumentException("Position out of range", nameof(position));
if (bitPosition < 0 || bitPosition > 7)
throw new ArgumentException("Bit position must be 0-7", nameof(bitPosition));
_position = position;
_bitPosition = bitPosition;
}
/// <summary>
/// Get remaining bytes in stream
/// </summary>
/// <returns>Number of remaining bytes</returns>
public int GetRemainingBytes()
{
int remaining = _size - _position;
if (_bitPosition > 0 && remaining > 0)
remaining--;
return Math.Max(0, remaining);
}
}
}