2005-12-08 07:13:07 +00:00

178 lines
4.7 KiB
Java

/*
* Copyright (C) 2005 Alfresco, Inc.
*
* Licensed under the Mozilla Public License version 1.1
* with a permitted attribution clause. You may obtain a
* copy of the License at
*
* http://www.alfresco.org/legal/license.txt
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied. See the License for the specific
* language governing permissions and limitations under the
* License.
*/
package org.alfresco.filesys.smb.server;
import java.io.IOException;
import java.net.Socket;
import org.alfresco.filesys.netbios.RFCNetBIOSProtocol;
import org.alfresco.filesys.util.DataPacker;
/**
* NetBIOS Protocol Packet Handler Class
*/
public class NetBIOSPacketHandler extends PacketHandler
{
/**
* Class constructor
*
* @param sock Socket
* @exception IOException If a network error occurs
*/
public NetBIOSPacketHandler(Socket sock) throws IOException
{
super(sock, SMBSrvPacket.PROTOCOL_NETBIOS, "NetBIOS", "NB");
}
/**
* Read a packet from the input stream
*
* @param pkt SMBSrvPacket
* @return int
* @exception IOexception If a network error occurs
*/
public final int readPacket(SMBSrvPacket pkt) throws IOException
{
// Read the packet header
byte[] buf = pkt.getBuffer();
int len = 0;
while (len < RFCNetBIOSProtocol.HEADER_LEN && len != -1)
len = readPacket(buf, len, RFCNetBIOSProtocol.HEADER_LEN - len);
// Check if the connection has been closed, read length equals -1
if (len == -1)
return len;
// Check if we received a valid NetBIOS header
if (len < RFCNetBIOSProtocol.HEADER_LEN)
throw new IOException("Invalid NetBIOS header, len=" + len);
// Get the packet type from the header
int typ = (int) (buf[0] & 0xFF);
int flags = (int) buf[1];
int dlen = (int) DataPacker.getShort(buf, 2);
if ((flags & 0x01) != 0)
dlen += 0x10000;
// Check for a session keep alive type message
if (typ == RFCNetBIOSProtocol.SESSION_KEEPALIVE)
return 0;
// Check if the packet buffer is large enough to hold the data + header
if (buf.length < (dlen + RFCNetBIOSProtocol.HEADER_LEN))
{
// Allocate a new buffer to hold the data and copy the existing header
byte[] newBuf = new byte[dlen + RFCNetBIOSProtocol.HEADER_LEN];
for (int i = 0; i < 4; i++)
newBuf[i] = buf[i];
// Attach the new buffer to the SMB packet
pkt.setBuffer(newBuf);
buf = newBuf;
}
// Read the data part of the packet into the users buffer, this may take
// several reads
int offset = RFCNetBIOSProtocol.HEADER_LEN;
int totlen = offset;
while (dlen > 0)
{
// Read the data
len = readPacket(buf, offset, dlen);
// Check if the connection has been closed
if (len == -1)
return -1;
// Update the received length and remaining data length
totlen += len;
dlen -= len;
// Update the user buffer offset as more reads will be required
// to complete the data read
offset += len;
} // end while reading data
// Return the received packet length
return totlen;
}
/**
* Send a packet to the output stream
*
* @param pkt SMBSrvPacket
* @param len int
* @exception IOexception If a network error occurs
*/
public final void writePacket(SMBSrvPacket pkt, int len) throws IOException
{
// Fill in the NetBIOS message header, this is already allocated as
// part of the users buffer.
byte[] buf = pkt.getBuffer();
buf[0] = (byte) RFCNetBIOSProtocol.SESSION_MESSAGE;
buf[1] = (byte) 0;
if (len > 0xFFFF)
{
// Set the >64K flag
buf[1] = (byte) 0x01;
// Set the low word of the data length
DataPacker.putShort((short) (len & 0xFFFF), buf, 2);
}
else
{
// Set the data length
DataPacker.putShort((short) len, buf, 2);
}
// Output the data packet
int bufSiz = len + RFCNetBIOSProtocol.HEADER_LEN;
writePacket(buf, 0, bufSiz);
}
}