1/*
2 * Copyright (c) 2015 The Android Open Source Project
3 * Copyright (C) 2015 Samsung LSI
4 * Copyright (c) 2008-2009, Motorola, Inc.
5 *
6 * All rights reserved.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions are met:
10 *
11 * - Redistributions of source code must retain the above copyright notice,
12 * this list of conditions and the following disclaimer.
13 *
14 * - Redistributions in binary form must reproduce the above copyright notice,
15 * this list of conditions and the following disclaimer in the documentation
16 * and/or other materials provided with the distribution.
17 *
18 * - Neither the name of the Motorola, Inc. nor the names of its contributors
19 * may be used to endorse or promote products derived from this software
20 * without specific prior written permission.
21 *
22 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
23 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
26 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
27 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
28 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
29 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
30 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
31 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32 * POSSIBILITY OF SUCH DAMAGE.
33 */
34
35package javax.obex;
36
37import java.io.IOException;
38import java.io.InputStream;
39import java.io.OutputStream;
40import java.io.DataInputStream;
41import java.io.DataOutputStream;
42import java.io.ByteArrayOutputStream;
43
44import android.util.Log;
45
46/**
47 * This class implements the <code>Operation</code> interface. It will read and
48 * write data via puts and gets.
49 * @hide
50 */
51public final class ClientOperation implements Operation, BaseStream {
52
53    private static final String TAG = "ClientOperation";
54
55    private static final boolean V = ObexHelper.VDBG;
56
57    private ClientSession mParent;
58
59    private boolean mInputOpen;
60
61    private PrivateInputStream mPrivateInput;
62
63    private boolean mPrivateInputOpen;
64
65    private PrivateOutputStream mPrivateOutput;
66
67    private boolean mPrivateOutputOpen;
68
69    private String mExceptionMessage;
70
71    private int mMaxPacketSize;
72
73    private boolean mOperationDone;
74
75    private boolean mGetOperation;
76
77    private boolean mGetFinalFlag;
78
79    private HeaderSet mRequestHeader;
80
81    private HeaderSet mReplyHeader;
82
83    private boolean mEndOfBodySent;
84
85    private boolean mSendBodyHeader = true;
86    // A latch - when triggered, there is not way back ;-)
87    private boolean mSrmActive = false;
88
89    // Assume SRM disabled - until support is confirmed
90    // by the server
91    private boolean mSrmEnabled = false;
92    // keep waiting until final-bit is received in request
93    // to handle the case where the SRM enable header is in
94    // a different OBEX packet than the SRMP header.
95    private boolean mSrmWaitingForRemote = true;
96
97
98    /**
99     * Creates new OperationImpl to read and write data to a server
100     * @param maxSize the maximum packet size
101     * @param p the parent to this object
102     * @param type <code>true</code> if this is a get request;
103     *        <code>false</code. if this is a put request
104     * @param header the header to set in the initial request
105     * @throws IOException if the an IO error occurred
106     */
107    public ClientOperation(int maxSize, ClientSession p, HeaderSet header, boolean type)
108            throws IOException {
109
110        mParent = p;
111        mEndOfBodySent = false;
112        mInputOpen = true;
113        mOperationDone = false;
114        mMaxPacketSize = maxSize;
115        mGetOperation = type;
116        mGetFinalFlag = false;
117
118        mPrivateInputOpen = false;
119        mPrivateOutputOpen = false;
120        mPrivateInput = null;
121        mPrivateOutput = null;
122
123        mReplyHeader = new HeaderSet();
124
125        mRequestHeader = new HeaderSet();
126
127        int[] headerList = header.getHeaderList();
128
129        if (headerList != null) {
130
131            for (int i = 0; i < headerList.length; i++) {
132                mRequestHeader.setHeader(headerList[i], header.getHeader(headerList[i]));
133            }
134        }
135
136        if ((header).mAuthChall != null) {
137            mRequestHeader.mAuthChall = new byte[(header).mAuthChall.length];
138            System.arraycopy((header).mAuthChall, 0, mRequestHeader.mAuthChall, 0,
139                    (header).mAuthChall.length);
140        }
141
142        if ((header).mAuthResp != null) {
143            mRequestHeader.mAuthResp = new byte[(header).mAuthResp.length];
144            System.arraycopy((header).mAuthResp, 0, mRequestHeader.mAuthResp, 0,
145                    (header).mAuthResp.length);
146
147        }
148
149        if ((header).mConnectionID != null) {
150            mRequestHeader.mConnectionID = new byte[4];
151            System.arraycopy((header).mConnectionID, 0, mRequestHeader.mConnectionID, 0,
152                    4);
153
154        }
155    }
156
157    /**
158     * Allows to set flag which will force GET to be always sent as single packet request with
159     * final flag set. This is to improve compatibility with some profiles, i.e. PBAP which
160     * require requests to be sent this way.
161     */
162    public void setGetFinalFlag(boolean flag) {
163        mGetFinalFlag = flag;
164    }
165
166    /**
167     * Sends an ABORT message to the server. By calling this method, the
168     * corresponding input and output streams will be closed along with this
169     * object.
170     * @throws IOException if the transaction has already ended or if an OBEX
171     *         server called this method
172     */
173    public synchronized void abort() throws IOException {
174        ensureOpen();
175        //no compatible with sun-ri
176        if ((mOperationDone) && (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE)) {
177            throw new IOException("Operation has already ended");
178        }
179
180        mExceptionMessage = "Operation aborted";
181        if ((!mOperationDone) && (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
182            mOperationDone = true;
183            /*
184             * Since we are not sending any headers or returning any headers then
185             * we just need to write and read the same bytes
186             */
187            mParent.sendRequest(ObexHelper.OBEX_OPCODE_ABORT, null, mReplyHeader, null, false);
188
189            if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_OK) {
190                throw new IOException("Invalid response code from server");
191            }
192
193            mExceptionMessage = null;
194        }
195
196        close();
197    }
198
199    /**
200     * Retrieves the response code retrieved from the server. Response codes are
201     * defined in the <code>ResponseCodes</code> interface.
202     * @return the response code retrieved from the server
203     * @throws IOException if an error occurred in the transport layer during
204     *         the transaction; if this method is called on a
205     *         <code>HeaderSet</code> object created by calling
206     *         <code>createHeaderSet</code> in a <code>ClientSession</code>
207     *         object
208     */
209    public synchronized int getResponseCode() throws IOException {
210        //avoid dup validateConnection
211        if ((mReplyHeader.responseCode == -1)
212                || (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
213            validateConnection();
214        }
215
216        return mReplyHeader.responseCode;
217    }
218
219    /**
220     * This method will always return <code>null</code>
221     * @return <code>null</code>
222     */
223    public String getEncoding() {
224        return null;
225    }
226
227    /**
228     * Returns the type of content that the resource connected to is providing.
229     * E.g. if the connection is via HTTP, then the value of the content-type
230     * header field is returned.
231     * @return the content type of the resource that the URL references, or
232     *         <code>null</code> if not known
233     */
234    public String getType() {
235        try {
236            return (String)mReplyHeader.getHeader(HeaderSet.TYPE);
237        } catch (IOException e) {
238            if(V) Log.d(TAG, "Exception occured - returning null",e);
239            return null;
240        }
241    }
242
243    /**
244     * Returns the length of the content which is being provided. E.g. if the
245     * connection is via HTTP, then the value of the content-length header field
246     * is returned.
247     * @return the content length of the resource that this connection's URL
248     *         references, or -1 if the content length is not known
249     */
250    public long getLength() {
251        try {
252            Long temp = (Long)mReplyHeader.getHeader(HeaderSet.LENGTH);
253
254            if (temp == null) {
255                return -1;
256            } else {
257                return temp.longValue();
258            }
259        } catch (IOException e) {
260            if(V) Log.d(TAG,"Exception occured - returning -1",e);
261            return -1;
262        }
263    }
264
265    /**
266     * Open and return an input stream for a connection.
267     * @return an input stream
268     * @throws IOException if an I/O error occurs
269     */
270    public InputStream openInputStream() throws IOException {
271
272        ensureOpen();
273
274        if (mPrivateInputOpen)
275            throw new IOException("no more input streams available");
276        if (mGetOperation) {
277            // send the GET request here
278            validateConnection();
279        } else {
280            if (mPrivateInput == null) {
281                mPrivateInput = new PrivateInputStream(this);
282            }
283        }
284
285        mPrivateInputOpen = true;
286
287        return mPrivateInput;
288    }
289
290    /**
291     * Open and return a data input stream for a connection.
292     * @return an input stream
293     * @throws IOException if an I/O error occurs
294     */
295    public DataInputStream openDataInputStream() throws IOException {
296        return new DataInputStream(openInputStream());
297    }
298
299    /**
300     * Open and return an output stream for a connection.
301     * @return an output stream
302     * @throws IOException if an I/O error occurs
303     */
304    public OutputStream openOutputStream() throws IOException {
305
306        ensureOpen();
307        ensureNotDone();
308
309        if (mPrivateOutputOpen)
310            throw new IOException("no more output streams available");
311
312        if (mPrivateOutput == null) {
313            // there are 3 bytes operation headers and 3 bytes body headers //
314            mPrivateOutput = new PrivateOutputStream(this, getMaxPacketSize());
315        }
316
317        mPrivateOutputOpen = true;
318
319        return mPrivateOutput;
320    }
321
322    public int getMaxPacketSize() {
323        return mMaxPacketSize - 6 - getHeaderLength();
324    }
325
326    public int getHeaderLength() {
327        // OPP may need it
328        byte[] headerArray = ObexHelper.createHeader(mRequestHeader, false);
329        return headerArray.length;
330    }
331
332    /**
333     * Open and return a data output stream for a connection.
334     * @return an output stream
335     * @throws IOException if an I/O error occurs
336     */
337    public DataOutputStream openDataOutputStream() throws IOException {
338        return new DataOutputStream(openOutputStream());
339    }
340
341    /**
342     * Closes the connection and ends the transaction
343     * @throws IOException if the operation has already ended or is closed
344     */
345    public void close() throws IOException {
346        mInputOpen = false;
347        mPrivateInputOpen = false;
348        mPrivateOutputOpen = false;
349        mParent.setRequestInactive();
350    }
351
352    /**
353     * Returns the headers that have been received during the operation.
354     * Modifying the object returned has no effect on the headers that are sent
355     * or retrieved.
356     * @return the headers received during this <code>Operation</code>
357     * @throws IOException if this <code>Operation</code> has been closed
358     */
359    public HeaderSet getReceivedHeader() throws IOException {
360        ensureOpen();
361
362        return mReplyHeader;
363    }
364
365    /**
366     * Specifies the headers that should be sent in the next OBEX message that
367     * is sent.
368     * @param headers the headers to send in the next message
369     * @throws IOException if this <code>Operation</code> has been closed or the
370     *         transaction has ended and no further messages will be exchanged
371     * @throws IllegalArgumentException if <code>headers</code> was not created
372     *         by a call to <code>ServerRequestHandler.createHeaderSet()</code>
373     * @throws NullPointerException if <code>headers</code> is <code>null</code>
374     */
375    public void sendHeaders(HeaderSet headers) throws IOException {
376        ensureOpen();
377        if (mOperationDone) {
378            throw new IOException("Operation has already exchanged all data");
379        }
380
381        if (headers == null) {
382            throw new IOException("Headers may not be null");
383        }
384
385        int[] headerList = headers.getHeaderList();
386        if (headerList != null) {
387            for (int i = 0; i < headerList.length; i++) {
388                mRequestHeader.setHeader(headerList[i], headers.getHeader(headerList[i]));
389            }
390        }
391    }
392
393    /**
394     * Verifies that additional information may be sent. In other words, the
395     * operation is not done.
396     * @throws IOException if the operation is completed
397     */
398    public void ensureNotDone() throws IOException {
399        if (mOperationDone) {
400            throw new IOException("Operation has completed");
401        }
402    }
403
404    /**
405     * Verifies that the connection is open and no exceptions should be thrown.
406     * @throws IOException if an exception needs to be thrown
407     */
408    public void ensureOpen() throws IOException {
409        mParent.ensureOpen();
410
411        if (mExceptionMessage != null) {
412            throw new IOException(mExceptionMessage);
413        }
414        if (!mInputOpen) {
415            throw new IOException("Operation has already ended");
416        }
417    }
418
419    /**
420     * Verifies that the connection is open and the proper data has been read.
421     * @throws IOException if an IO error occurs
422     */
423    private void validateConnection() throws IOException {
424        ensureOpen();
425
426        // to sure only one privateInput object exist.
427        if (mPrivateInput == null) {
428            startProcessing();
429        }
430    }
431
432    /**
433     * Sends a request to the client of the specified type.
434     * This function will enable SRM and set SRM active if the server
435     * response allows this.
436     * @param opCode the request code to send to the client
437     * @return <code>true</code> if there is more data to send;
438     *         <code>false</code> if there is no more data to send
439     * @throws IOException if an IO error occurs
440     */
441    private boolean sendRequest(int opCode) throws IOException {
442        boolean returnValue = false;
443        ByteArrayOutputStream out = new ByteArrayOutputStream();
444        int bodyLength = -1;
445        byte[] headerArray = ObexHelper.createHeader(mRequestHeader, true);
446        if (mPrivateOutput != null) {
447            bodyLength = mPrivateOutput.size();
448        }
449
450        /*
451         * Determine if there is space to add a body request.  At present
452         * this method checks to see if there is room for at least a 17
453         * byte body header.  This number needs to be at least 6 so that
454         * there is room for the header ID and length and the reply ID and
455         * length, but it is a waste of resources if we can't send much of
456         * the body.
457         */
458        final int MINIMUM_BODY_LENGTH = 3;
459        if ((ObexHelper.BASE_PACKET_LENGTH + headerArray.length + MINIMUM_BODY_LENGTH)
460                > mMaxPacketSize) {
461            int end = 0;
462            int start = 0;
463            // split & send the headerArray in multiple packets.
464
465            while (end != headerArray.length) {
466                //split the headerArray
467
468                end = ObexHelper.findHeaderEnd(headerArray, start, mMaxPacketSize
469                        - ObexHelper.BASE_PACKET_LENGTH);
470                // can not split
471                if (end == -1) {
472                    mOperationDone = true;
473                    abort();
474                    mExceptionMessage = "Header larger then can be sent in a packet";
475                    mInputOpen = false;
476
477                    if (mPrivateInput != null) {
478                        mPrivateInput.close();
479                    }
480
481                    if (mPrivateOutput != null) {
482                        mPrivateOutput.close();
483                    }
484                    throw new IOException("OBEX Packet exceeds max packet size");
485                }
486
487                byte[] sendHeader = new byte[end - start];
488                System.arraycopy(headerArray, start, sendHeader, 0, sendHeader.length);
489                if (!mParent.sendRequest(opCode, sendHeader, mReplyHeader, mPrivateInput, false)) {
490                    return false;
491                }
492
493                if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE) {
494                    return false;
495                }
496
497                start = end;
498            }
499
500            // Enable SRM if it should be enabled
501            checkForSrm();
502
503            if (bodyLength > 0) {
504                return true;
505            } else {
506                return false;
507            }
508        } else {
509            /* All headers will fit into a single package */
510            if(mSendBodyHeader == false) {
511                /* As we are not to send any body data, set the FINAL_BIT */
512                opCode |= ObexHelper.OBEX_OPCODE_FINAL_BIT_MASK;
513            }
514            out.write(headerArray);
515        }
516
517        if (bodyLength > 0) {
518            /*
519             * Determine if we can send the whole body or just part of
520             * the body.  Remember that there is the 3 bytes for the
521             * response message and 3 bytes for the header ID and length
522             */
523            if (bodyLength > (mMaxPacketSize - headerArray.length - 6)) {
524                returnValue = true;
525
526                bodyLength = mMaxPacketSize - headerArray.length - 6;
527            }
528
529            byte[] body = mPrivateOutput.readBytes(bodyLength);
530
531            /*
532             * Since this is a put request if the final bit is set or
533             * the output stream is closed we need to send the 0x49
534             * (End of Body) otherwise, we need to send 0x48 (Body)
535             */
536            if ((mPrivateOutput.isClosed()) && (!returnValue) && (!mEndOfBodySent)
537                    && ((opCode & ObexHelper.OBEX_OPCODE_FINAL_BIT_MASK) != 0)) {
538                out.write(HeaderSet.END_OF_BODY);
539                mEndOfBodySent = true;
540            } else {
541                out.write(HeaderSet.BODY);
542            }
543
544            bodyLength += 3;
545            out.write((byte)(bodyLength >> 8));
546            out.write((byte)bodyLength);
547
548            if (body != null) {
549                out.write(body);
550            }
551        }
552
553        if (mPrivateOutputOpen && bodyLength <= 0 && !mEndOfBodySent) {
554            // only 0x82 or 0x83 can send 0x49
555            if ((opCode & ObexHelper.OBEX_OPCODE_FINAL_BIT_MASK) == 0) {
556                out.write(HeaderSet.BODY);
557            } else {
558                out.write(HeaderSet.END_OF_BODY);
559                mEndOfBodySent = true;
560            }
561
562            bodyLength = 3;
563            out.write((byte)(bodyLength >> 8));
564            out.write((byte)bodyLength);
565        }
566
567        if (out.size() == 0) {
568            if (!mParent.sendRequest(opCode, null, mReplyHeader, mPrivateInput, mSrmActive)) {
569                return false;
570            }
571            // Enable SRM if it should be enabled
572            checkForSrm();
573            return returnValue;
574        }
575        if ((out.size() > 0)
576                && (!mParent.sendRequest(opCode, out.toByteArray(),
577                        mReplyHeader, mPrivateInput, mSrmActive))) {
578            return false;
579        }
580        // Enable SRM if it should be enabled
581        checkForSrm();
582
583        // send all of the output data in 0x48,
584        // send 0x49 with empty body
585        if ((mPrivateOutput != null) && (mPrivateOutput.size() > 0))
586            returnValue = true;
587
588        return returnValue;
589    }
590
591    private void checkForSrm() throws IOException {
592        Byte srmMode = (Byte)mReplyHeader.getHeader(HeaderSet.SINGLE_RESPONSE_MODE);
593        if(mParent.isSrmSupported() == true && srmMode != null
594                && srmMode == ObexHelper.OBEX_SRM_ENABLE) {
595            mSrmEnabled = true;
596        }
597        /**
598         * Call this only when a complete obex packet have been received.
599         * (This is not optimal, but the current design is not really suited to
600         * the way SRM is specified.)
601         * The BT usage of SRM is not really safe - it assumes that the SRMP will fit
602         * into every OBEX packet, hence if another header occupies the entire packet,
603         * the scheme will not work - unlikely though.
604         */
605        if(mSrmEnabled) {
606            mSrmWaitingForRemote = false;
607            Byte srmp = (Byte)mReplyHeader.getHeader(HeaderSet.SINGLE_RESPONSE_MODE_PARAMETER);
608            if(srmp != null && srmp == ObexHelper.OBEX_SRMP_WAIT) {
609                mSrmWaitingForRemote = true;
610                // Clear the wait header, as the absence of the header in the next packet
611                // indicates don't wait anymore.
612                mReplyHeader.setHeader(HeaderSet.SINGLE_RESPONSE_MODE_PARAMETER, null);
613            }
614        }
615        if((mSrmWaitingForRemote == false) && (mSrmEnabled == true)) {
616            mSrmActive = true;
617        }
618    }
619
620    /**
621     * This method starts the processing thread results. It will send the
622     * initial request. If the response takes more then one packet, a thread
623     * will be started to handle additional requests
624     * @throws IOException if an IO error occurs
625     */
626    private synchronized void startProcessing() throws IOException {
627
628        if (mPrivateInput == null) {
629            mPrivateInput = new PrivateInputStream(this);
630        }
631        boolean more = true;
632
633        if (mGetOperation) {
634            if (!mOperationDone) {
635                mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
636                while ((more) && (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
637                    more = sendRequest(ObexHelper.OBEX_OPCODE_GET);
638                }
639                // For GET we need to loop until all headers have been sent,
640                // And then we wait for the first continue package with the
641                // reply.
642                if (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE) {
643                    mParent.sendRequest(ObexHelper.OBEX_OPCODE_GET_FINAL,
644                            null, mReplyHeader, mPrivateInput, mSrmActive);
645                }
646                if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE) {
647                    mOperationDone = true;
648                } else {
649                    checkForSrm();
650                }
651            }
652        } else {
653            // PUT operation
654            if (!mOperationDone) {
655                mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
656                while ((more) && (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
657                    more = sendRequest(ObexHelper.OBEX_OPCODE_PUT);
658                }
659            }
660
661            if (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE) {
662                mParent.sendRequest(ObexHelper.OBEX_OPCODE_PUT_FINAL,
663                        null, mReplyHeader, mPrivateInput, mSrmActive);
664            }
665
666            if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE) {
667                mOperationDone = true;
668            }
669        }
670    }
671
672    /**
673     * Continues the operation since there is no data to read.
674     * @param sendEmpty <code>true</code> if the operation should send an empty
675     *        packet or not send anything if there is no data to send
676     * @param inStream <code>true</code> if the stream is input stream or is
677     *        output stream
678     * @throws IOException if an IO error occurs
679     */
680    public synchronized boolean continueOperation(boolean sendEmpty, boolean inStream)
681            throws IOException {
682
683        // One path to the first put operation - the other one does not need to
684        // handle SRM, as all will fit into one packet.
685
686        if (mGetOperation) {
687            if ((inStream) && (!mOperationDone)) {
688                // to deal with inputstream in get operation
689                mParent.sendRequest(ObexHelper.OBEX_OPCODE_GET_FINAL,
690                        null, mReplyHeader, mPrivateInput, mSrmActive);
691                /*
692                  * Determine if that was not the last packet in the operation
693                  */
694                if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE) {
695                    mOperationDone = true;
696                } else {
697                    checkForSrm();
698                }
699
700                return true;
701
702            } else if ((!inStream) && (!mOperationDone)) {
703                // to deal with outputstream in get operation
704
705                if (mPrivateInput == null) {
706                    mPrivateInput = new PrivateInputStream(this);
707                }
708                sendRequest(ObexHelper.OBEX_OPCODE_GET);
709                return true;
710
711            } else if (mOperationDone) {
712                return false;
713            }
714
715        } else {
716            // PUT operation
717            if ((!inStream) && (!mOperationDone)) {
718                // to deal with outputstream in put operation
719                if (mReplyHeader.responseCode == -1) {
720                    mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
721                }
722                sendRequest(ObexHelper.OBEX_OPCODE_PUT);
723                return true;
724            } else if ((inStream) && (!mOperationDone)) {
725                // How to deal with inputstream  in put operation ?
726                return false;
727
728            } else if (mOperationDone) {
729                return false;
730            }
731
732        }
733        return false;
734    }
735
736    /**
737     * Called when the output or input stream is closed.
738     * @param inStream <code>true</code> if the input stream is closed;
739     *        <code>false</code> if the output stream is closed
740     * @throws IOException if an IO error occurs
741     */
742    public void streamClosed(boolean inStream) throws IOException {
743        if (!mGetOperation) {
744            if ((!inStream) && (!mOperationDone)) {
745                // to deal with outputstream in put operation
746
747                boolean more = true;
748
749                if ((mPrivateOutput != null) && (mPrivateOutput.size() <= 0)) {
750                    byte[] headerArray = ObexHelper.createHeader(mRequestHeader, false);
751                    if (headerArray.length <= 0)
752                        more = false;
753                }
754                // If have not sent any data so send  all now
755                if (mReplyHeader.responseCode == -1) {
756                    mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
757                }
758
759                while ((more) && (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
760                    more = sendRequest(ObexHelper.OBEX_OPCODE_PUT);
761                }
762
763                /*
764                 * According to the IrOBEX specification, after the final put, you
765                 * only have a single reply to send.  so we don't need the while
766                 * loop.
767                 */
768                while (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE) {
769
770                    sendRequest(ObexHelper.OBEX_OPCODE_PUT_FINAL);
771                }
772                mOperationDone = true;
773            } else if ((inStream) && (mOperationDone)) {
774                // how to deal with input stream in put stream ?
775                mOperationDone = true;
776            }
777        } else {
778            if ((inStream) && (!mOperationDone)) {
779
780                // to deal with inputstream in get operation
781                // Have not sent any data so send it all now
782
783                if (mReplyHeader.responseCode == -1) {
784                    mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
785                }
786
787                while (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE && !mOperationDone) {
788                    if (!sendRequest(ObexHelper.OBEX_OPCODE_GET_FINAL)) {
789                        break;
790                    }
791                }
792                while (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE && !mOperationDone) {
793                    mParent.sendRequest(ObexHelper.OBEX_OPCODE_GET_FINAL, null,
794                            mReplyHeader, mPrivateInput, false);
795                    // Regardless of the SRM state, wait for the response.
796                }
797                mOperationDone = true;
798            } else if ((!inStream) && (!mOperationDone)) {
799                // to deal with outputstream in get operation
800                // part of the data may have been sent in continueOperation.
801
802                boolean more = true;
803
804                if ((mPrivateOutput != null) && (mPrivateOutput.size() <= 0)) {
805                    byte[] headerArray = ObexHelper.createHeader(mRequestHeader, false);
806                    if (headerArray.length <= 0)
807                        more = false;
808                }
809
810                if (mPrivateInput == null) {
811                    mPrivateInput = new PrivateInputStream(this);
812                }
813                if ((mPrivateOutput != null) && (mPrivateOutput.size() <= 0))
814                    more = false;
815
816                mReplyHeader.responseCode = ResponseCodes.OBEX_HTTP_CONTINUE;
817                while ((more) && (mReplyHeader.responseCode == ResponseCodes.OBEX_HTTP_CONTINUE)) {
818                    more = sendRequest(ObexHelper.OBEX_OPCODE_GET);
819                }
820                sendRequest(ObexHelper.OBEX_OPCODE_GET_FINAL);
821                //                parent.sendRequest(0x83, null, replyHeaders, privateInput);
822                if (mReplyHeader.responseCode != ResponseCodes.OBEX_HTTP_CONTINUE) {
823                    mOperationDone = true;
824                }
825            }
826        }
827    }
828
829    public void noBodyHeader(){
830        mSendBodyHeader = false;
831    }
832}
833