为什么我的文件描述符为蓝牙套接字为空?

时间:2015-09-15 19:15:13

标签: android sockets bluetooth

有人可以帮我弄清楚为什么我只能通过BluetoothServerSocket.accept()打开蓝牙插座的空文件描述符?

我的目标是通过蓝牙在两个设备之间流式传输视频,方法是将视频写入一侧的文件描述符,另一侧的文件描述符读取视频。我的蓝牙连接很好,我可以来回发送原始数据,但我只能在客户端获取文件描述符。在服务器端,使用相同的代码,我只能获得一个空文件描述符。在调试器中,我可以在服务器端看到 mySocket.mSocketIS.this $ 0.fd 的文件描述符,但我无法弄清楚如何访问它。有人可以帮忙吗?这是Android 4.4.2,这是我的代码:

首先是破碎的代码(服务器端):

    // Listen for an incoming Bluetooth connection
class AcceptThread extends Thread
{
    // Thread that accepts incoming bluetooth connections
    public AcceptThread()
    {
        try
        {
            // Open a listening server socket.  This is non-blocking
            btServerSocket = BA.listenUsingRfcommWithServiceRecord("ServerApp", videoUUID);
        } catch(IOException e){ btServerSocket = null; }
    } // AcceptThread()

    public void run()
    {
        BluetoothSocket btSocket = null;

        // Listen until exception or we have a socket
        while(true)
        {
            try
            {
                // Blocking call to accept an incoming connection.  To get out of this, call cancel() which closes the socket, causing .accept() to throw an exception
                btSocket = btServerSocket.accept();
                // If we get here, we're connected!  

                Field pfdField = btSocket.getClass().getDeclaredField("mPfd");
                pfdField.setAccessible(true);
                ParcelFileDescriptor pfd = (ParcelFileDescriptor) pfdField.get(btSocket);

                // >>> ERROR - pfd is null  <<<<   I can see a fd at mySocket.mSocketIS.this$0.fd;, but how do I access it? 


                FileDescriptor myFd = pfd.getFileDescriptor();  


              // ... blah blah...

现在工作代码(客户端):

    // Connect to a remote device as the client (we are the client)
class ConnectThread extends Thread
{
    // ctor
    // remoteUUID - The UUID of the remote device that we want to connect to
    public ConnectThread(BluetoothDevice btDevice, UUID remoteUUID)
    {
        // Get a BT socket to connect with the given BluetoothDevice
        try
        {
            // MY_UUID is the app's UUID string, also used by the server code
            btClientSocket = btDevice.createRfcommSocketToServiceRecord(remoteUUID);
        }catch(Exception e){ postUIMessage("ConnectThread exception: " + e.toString());    }
    } // ConnectThread ctor

    public void run()
    {
        // Cancel discovery because it will slow down the connection
        BA.cancelDiscovery();
        try
        {
            // Connect the device through the socket. This will block until it succeeds or throws an exception.  To get out, call cancel() below, which will cause .connect() to throw an exception.
            btClientSocket.connect();

            Field pfdField = btClientSocket.getClass().getDeclaredField("mPfd");
            pfdField.setAccessible(true);
            ParcelFileDescriptor pfd = (ParcelFileDescriptor) pfdField.get(btClientSocket);
            FileDescriptor myFd = pfd.getFileDescriptor();  // Pass this to Recorder.setOutputFile();

            // Yay myFd is good!  

1 个答案:

答案 0 :(得分:0)

我找到了关于这个问题的解决方案,我们也使用蓝牙作为服务器。我在BluetoothSocket的LocalSocket字段中找到了文件描述符。我的目标是获取文件并关闭它。

int mfd = 0;
        Field socketField = null;
        LocalSocket mSocket = null;

        try
        {
            socketField = btSocket.getClass().getDeclaredField("mSocket");
            socketField.setAccessible(true);

            mSocket = (LocalSocket)socketField.get(btSocket);
        }
        catch(Exception e)
        {
            Log ( "Exception getting mSocket in cleanCloseFix(): " + e.toString());
        }

        if(mSocket != null)
        {
            FileDescriptor fileDescriptor =
                    mSocket.getFileDescriptor();

            String in = fileDescriptor.toString();

            //regular expression to get filedescriptor index id
            Pattern p = Pattern.compile("\\[(.*?)\\]");
            Matcher m = p.matcher(in);

            while(m.find()) {
                Log ( "File Descriptor " + m.group(1));
                mfd = Integer.parseInt(m.group(1));
                break;
            }

            //Shutdown the socket properly
            mSocket.shutdownInput();
            mSocket.shutdownOutput();
            mSocket.close();

            mSocket = null;

            try { socketField.set(btSocket, mSocket); }
            catch(Exception e)
            {
                Log ("Exception setting mSocket = null in cleanCloseFix(): " + e.toString());
            }

            //Close the file descriptor when we have it from the Local Socket
            try {
                ParcelFileDescriptor parcelFileDescriptor = ParcelFileDescriptor.adoptFd(mfd);
                if (parcelFileDescriptor != null) {
                    parcelFileDescriptor.close();
                    Log ( "File descriptor close succeed : FD = " + mfd);
                }
            } catch (Exception ex) {
                Log ( "File descriptor close exception " + ex.getMessage());
            }
        }