/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * 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.
 */
#ifndef _DECAF_IO_INTERRUPTEDIOEXCEPTION_H
#define _DECAF_IO_INTERRUPTEDIOEXCEPTION_H

#include <decaf/lang/Exception.h>
#include <decaf/io/IOException.h>

namespace decaf{
namespace io{

    /*
     * Signals that an I/O exception of some sort has occurred.
     */
    class DECAF_API InterruptedIOException : public io::IOException {
    public:

        /**
         * Default Constructor
         */
        InterruptedIOException() throw() {}

        /**
         * Copy Constructor
         * @param ex the exception to copy
         */
        InterruptedIOException( const lang::Exception& ex ) throw()
        : IOException() {
            *(lang::Exception*)this = ex;
        }

        /**
         * Copy Constructor
         * @param ex the exception to copy, which is an instance of this type
         */
        InterruptedIOException( const InterruptedIOException& ex ) throw()
        : IOException() {
            *(lang::Exception*)this = ex;
        }

        /**
         * Constructor - Initializes the file name and line number where
         * this message occurred.  Sets the message to report, using an
         * optional list of arguments to parse into the message
         * @param file name where exception occurs
         * @param line number where the exception occurred.
         * @param cause The exception that was the cause for this one to be thrown.
         * @param message to report
         * @param list of primitives that are formatted into the message
         */
        InterruptedIOException( const char* file, const int lineNumber,
                                const std::exception* cause,
                                const char* msg, ... ) throw() : IOException( cause )
        {
            va_list vargs;
            va_start( vargs, msg );
            buildMessage( msg, vargs );

            // Set the first mark for this exception.
            setMark( file, lineNumber );
        }

        /**
         * Constructor
         * @param cause Pointer to the exception that caused this one to
         * be thrown, the object is cloned caller retains ownership.
         */
        InterruptedIOException( const std::exception* cause ) throw() : IOException( cause ) {}

        /**
         * Constructor
         * @param file name of the file were the exception occurred.
         * @param lineNumber line where the exception occurred
         * @param msg the message that was generated
         */
        InterruptedIOException( const char* file, const int lineNumber,
                                const char* msg, ... ) throw()
        : IOException() {
            va_list vargs;
            va_start( vargs, msg );
            buildMessage( msg, vargs );

            // Set the first mark for this exception.
            setMark( file, lineNumber );
        }

        /**
         * Clones this exception.  This is useful for cases where you need
         * to preserve the type of the original exception as well as the message.
         * All subclasses should override.
         */
        virtual InterruptedIOException* clone() const {
            return new IOException( *this );
        }

        virtual ~InterruptedIOException() throw() {}

    };

}}

#endif /*_DECAF_IO_INTERRUPTEDIOEXCEPTION_H*/
