// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "ppapi/cpp/file_io.h" #include // memcpy #include "ppapi/c/ppb_file_io.h" #include "ppapi/c/pp_errors.h" #include "ppapi/cpp/completion_callback.h" #include "ppapi/cpp/dev/resource_array_dev.h" #include "ppapi/cpp/file_ref.h" #include "ppapi/cpp/instance_handle.h" #include "ppapi/cpp/module.h" #include "ppapi/cpp/module_impl.h" namespace pp { namespace { template <> const char* interface_name() { return PPB_FILEIO_INTERFACE_1_0; } template <> const char* interface_name() { return PPB_FILEIO_INTERFACE_1_1; } } // namespace FileIO::FileIO() { } FileIO::FileIO(const InstanceHandle& instance) { if (has_interface()) { PassRefFromConstructor(get_interface()->Create( instance.pp_instance())); } else if (has_interface()) { PassRefFromConstructor(get_interface()->Create( instance.pp_instance())); } } FileIO::FileIO(const FileIO& other) : Resource(other) { } int32_t FileIO::Open(const FileRef& file_ref, int32_t open_flags, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Open( pp_resource(), file_ref.pp_resource(), open_flags, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Open( pp_resource(), file_ref.pp_resource(), open_flags, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Query(PP_FileInfo* result_buf, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Query( pp_resource(), result_buf, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Query( pp_resource(), result_buf, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Touch(PP_Time last_access_time, PP_Time last_modified_time, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Touch( pp_resource(), last_access_time, last_modified_time, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Touch( pp_resource(), last_access_time, last_modified_time, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Read(int64_t offset, char* buffer, int32_t bytes_to_read, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Read(pp_resource(), offset, buffer, bytes_to_read, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Read(pp_resource(), offset, buffer, bytes_to_read, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Read( int32_t offset, int32_t max_read_length, const CompletionCallbackWithOutput< std::vector >& cc) { if (has_interface()) { PP_ArrayOutput array_output = cc.output(); return get_interface()->ReadToArray(pp_resource(), offset, max_read_length, &array_output, cc.pp_completion_callback()); } else if (has_interface()) { // Data for our callback wrapper. The callback handler will delete it and // temp_buffer. CallbackData1_0* data = new CallbackData1_0; data->output = cc.output(); data->temp_buffer = max_read_length >= 0 ? new char[max_read_length] : NULL; data->original_callback = cc.pp_completion_callback(); // Actual returned bytes might not equals to max_read_length. We need to // read to a temporary buffer first and copy later to make sure the array // buffer has correct size. return get_interface()->Read( pp_resource(), offset, data->temp_buffer, max_read_length, PP_MakeCompletionCallback(&CallbackConverter, data)); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Write(int64_t offset, const char* buffer, int32_t bytes_to_write, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Write( pp_resource(), offset, buffer, bytes_to_write, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Write( pp_resource(), offset, buffer, bytes_to_write, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::SetLength(int64_t length, const CompletionCallback& cc) { if (has_interface()) { return get_interface()->SetLength( pp_resource(), length, cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->SetLength( pp_resource(), length, cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } int32_t FileIO::Flush(const CompletionCallback& cc) { if (has_interface()) { return get_interface()->Flush( pp_resource(), cc.pp_completion_callback()); } else if (has_interface()) { return get_interface()->Flush( pp_resource(), cc.pp_completion_callback()); } return cc.MayForce(PP_ERROR_NOINTERFACE); } void FileIO::Close() { if (has_interface()) get_interface()->Close(pp_resource()); else if (has_interface()) get_interface()->Close(pp_resource()); } // static void FileIO::CallbackConverter(void* user_data, int32_t result) { CallbackData1_0* data = static_cast(user_data); if (result >= 0) { // Copy to the destination buffer owned by the callback. char* buffer = static_cast(data->output.GetDataBuffer( data->output.user_data, result, sizeof(char))); memcpy(buffer, data->temp_buffer, result); delete[] data->temp_buffer; } // Now execute the original callback. PP_RunCompletionCallback(&data->original_callback, result); delete data; } } // namespace pp