IndexedDBFactory now ForceCloses databases.
[chromium-blink-merge.git] / content / child / npapi / plugin_instance.cc
bloba7e2b94ea8e0e5d83ae64214a1d48a15d98b9b4d
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "content/child/npapi/plugin_instance.h"
7 #include "base/bind.h"
8 #include "base/file_util.h"
9 #include "base/message_loop/message_loop.h"
10 #include "base/strings/string_number_conversions.h"
11 #include "base/strings/utf_string_conversions.h"
12 #include "build/build_config.h"
13 #include "content/child/npapi/plugin_host.h"
14 #include "content/child/npapi/plugin_lib.h"
15 #include "content/child/npapi/plugin_stream_url.h"
16 #include "content/child/npapi/plugin_string_stream.h"
17 #include "content/child/npapi/webplugin.h"
18 #include "content/child/npapi/webplugin_delegate.h"
19 #include "content/child/npapi/webplugin_resource_client.h"
20 #include "content/public/common/content_constants.h"
21 #include "net/base/escape.h"
23 #if defined(OS_MACOSX)
24 #include <ApplicationServices/ApplicationServices.h>
25 #endif
27 namespace content {
29 PluginInstance::PluginInstance(PluginLib* plugin, const std::string& mime_type)
30 : plugin_(plugin),
31 npp_(0),
32 host_(PluginHost::Singleton()),
33 npp_functions_(plugin->functions()),
34 window_handle_(0),
35 windowless_(false),
36 transparent_(true),
37 webplugin_(0),
38 mime_type_(mime_type),
39 get_notify_data_(0),
40 use_mozilla_user_agent_(false),
41 #if defined (OS_MACOSX)
42 #ifdef NP_NO_QUICKDRAW
43 drawing_model_(NPDrawingModelCoreGraphics),
44 #else
45 drawing_model_(NPDrawingModelQuickDraw),
46 #endif
47 #ifdef NP_NO_CARBON
48 event_model_(NPEventModelCocoa),
49 #else
50 event_model_(NPEventModelCarbon),
51 #endif
52 currently_handled_event_(NULL),
53 #endif
54 message_loop_(base::MessageLoop::current()),
55 load_manually_(false),
56 in_close_streams_(false),
57 next_timer_id_(1),
58 next_notify_id_(0),
59 next_range_request_id_(0),
60 handles_url_redirects_(false) {
61 npp_ = new NPP_t();
62 npp_->ndata = 0;
63 npp_->pdata = 0;
65 if (mime_type_ == kFlashPluginSwfMimeType)
66 transparent_ = false;
68 memset(&zero_padding_, 0, sizeof(zero_padding_));
69 DCHECK(message_loop_);
72 PluginInstance::~PluginInstance() {
73 CloseStreams();
75 if (npp_ != 0) {
76 delete npp_;
77 npp_ = 0;
80 if (plugin_.get())
81 plugin_->CloseInstance();
84 PluginStreamUrl* PluginInstance::CreateStream(unsigned long resource_id,
85 const GURL& url,
86 const std::string& mime_type,
87 int notify_id) {
89 bool notify;
90 void* notify_data;
91 GetNotifyData(notify_id, &notify, &notify_data);
92 PluginStreamUrl* stream = new PluginStreamUrl(
93 resource_id, url, this, notify, notify_data);
95 AddStream(stream);
96 return stream;
99 void PluginInstance::AddStream(PluginStream* stream) {
100 open_streams_.push_back(make_scoped_refptr(stream));
103 void PluginInstance::RemoveStream(PluginStream* stream) {
104 if (in_close_streams_)
105 return;
107 std::vector<scoped_refptr<PluginStream> >::iterator stream_index;
108 for (stream_index = open_streams_.begin();
109 stream_index != open_streams_.end(); ++stream_index) {
110 if (stream_index->get() == stream) {
111 open_streams_.erase(stream_index);
112 break;
117 bool PluginInstance::IsValidStream(const NPStream* stream) {
118 std::vector<scoped_refptr<PluginStream> >::iterator stream_index;
119 for (stream_index = open_streams_.begin();
120 stream_index != open_streams_.end(); ++stream_index) {
121 if ((*stream_index)->stream() == stream)
122 return true;
125 return false;
128 void PluginInstance::CloseStreams() {
129 in_close_streams_ = true;
130 for (unsigned int index = 0; index < open_streams_.size(); ++index) {
131 // Close all streams on the way down.
132 open_streams_[index]->Close(NPRES_USER_BREAK);
134 open_streams_.clear();
135 in_close_streams_ = false;
138 WebPluginResourceClient* PluginInstance::GetRangeRequest(
139 int id) {
140 PendingRangeRequestMap::iterator iter = pending_range_requests_.find(id);
141 if (iter == pending_range_requests_.end()) {
142 NOTREACHED();
143 return NULL;
146 WebPluginResourceClient* rv = iter->second->AsResourceClient();
147 pending_range_requests_.erase(iter);
148 return rv;
151 bool PluginInstance::Start(const GURL& url,
152 char** const param_names,
153 char** const param_values,
154 int param_count,
155 bool load_manually) {
156 load_manually_ = load_manually;
157 unsigned short mode = load_manually_ ? NP_FULL : NP_EMBED;
158 npp_->ndata = this;
160 NPError err = NPP_New(mode, param_count,
161 const_cast<char **>(param_names), const_cast<char **>(param_values));
163 if (err == NPERR_NO_ERROR) {
164 handles_url_redirects_ =
165 ((npp_functions_->version >= NPVERS_HAS_URL_REDIRECT_HANDLING) &&
166 (npp_functions_->urlredirectnotify));
168 return err == NPERR_NO_ERROR;
171 NPObject *PluginInstance::GetPluginScriptableObject() {
172 NPObject *value = NULL;
173 NPError error = NPP_GetValue(NPPVpluginScriptableNPObject, &value);
174 if (error != NPERR_NO_ERROR || value == NULL)
175 return NULL;
176 return value;
179 bool PluginInstance::GetFormValue(base::string16* value) {
180 // Plugins will allocate memory for the return value by using NPN_MemAlloc().
181 char *plugin_value = NULL;
182 NPError error = NPP_GetValue(NPPVformValue, &plugin_value);
183 if (error != NPERR_NO_ERROR || !plugin_value) {
184 return false;
186 // Assumes the result is UTF8 text, as Firefox does.
187 *value = base::UTF8ToUTF16(plugin_value);
188 host_->host_functions()->memfree(plugin_value);
189 return true;
192 // WebPluginLoadDelegate methods
193 void PluginInstance::DidFinishLoadWithReason(const GURL& url,
194 NPReason reason,
195 int notify_id) {
196 bool notify;
197 void* notify_data;
198 GetNotifyData(notify_id, &notify, &notify_data);
199 if (!notify) {
200 NOTREACHED();
201 return;
204 NPP_URLNotify(url.spec().c_str(), reason, notify_data);
207 unsigned PluginInstance::GetBackingTextureId() {
208 // By default the plugin instance is not backed by an OpenGL texture.
209 return 0;
212 // NPAPI methods
213 NPError PluginInstance::NPP_New(unsigned short mode,
214 short argc,
215 char* argn[],
216 char* argv[]) {
217 DCHECK(npp_functions_ != 0);
218 DCHECK(npp_functions_->newp != 0);
219 DCHECK(argc >= 0);
221 if (npp_functions_->newp != 0) {
222 return npp_functions_->newp(
223 (NPMIMEType)mime_type_.c_str(), npp_, mode, argc, argn, argv, NULL);
225 return NPERR_INVALID_FUNCTABLE_ERROR;
228 void PluginInstance::NPP_Destroy() {
229 DCHECK(npp_functions_ != 0);
230 DCHECK(npp_functions_->destroy != 0);
232 if (npp_functions_->destroy != 0) {
233 NPSavedData *savedData = 0;
234 npp_functions_->destroy(npp_, &savedData);
236 // TODO: Support savedData. Technically, these need to be
237 // saved on a per-URL basis, and then only passed
238 // to new instances of the plugin at the same URL.
239 // Sounds like a huge security risk. When we do support
240 // these, we should pass them back to the PluginLib
241 // to be stored there.
242 DCHECK(savedData == 0);
245 for (unsigned int file_index = 0; file_index < files_created_.size();
246 file_index++) {
247 base::DeleteFile(files_created_[file_index], false);
250 // Ensure that no timer callbacks are invoked after NPP_Destroy.
251 timers_.clear();
254 NPError PluginInstance::NPP_SetWindow(NPWindow* window) {
255 DCHECK(npp_functions_ != 0);
256 DCHECK(npp_functions_->setwindow != 0);
258 if (npp_functions_->setwindow != 0) {
259 return npp_functions_->setwindow(npp_, window);
261 return NPERR_INVALID_FUNCTABLE_ERROR;
264 NPError PluginInstance::NPP_NewStream(NPMIMEType type,
265 NPStream* stream,
266 NPBool seekable,
267 unsigned short* stype) {
268 DCHECK(npp_functions_ != 0);
269 DCHECK(npp_functions_->newstream != 0);
270 if (npp_functions_->newstream != 0) {
271 return npp_functions_->newstream(npp_, type, stream, seekable, stype);
273 return NPERR_INVALID_FUNCTABLE_ERROR;
276 NPError PluginInstance::NPP_DestroyStream(NPStream* stream, NPReason reason) {
277 DCHECK(npp_functions_ != 0);
278 DCHECK(npp_functions_->destroystream != 0);
280 if (stream == NULL || !IsValidStream(stream) || (stream->ndata == NULL))
281 return NPERR_INVALID_INSTANCE_ERROR;
283 if (npp_functions_->destroystream != 0) {
284 NPError result = npp_functions_->destroystream(npp_, stream, reason);
285 stream->ndata = NULL;
286 return result;
288 return NPERR_INVALID_FUNCTABLE_ERROR;
291 int PluginInstance::NPP_WriteReady(NPStream* stream) {
292 DCHECK(npp_functions_ != 0);
293 DCHECK(npp_functions_->writeready != 0);
294 if (npp_functions_->writeready != 0) {
295 return npp_functions_->writeready(npp_, stream);
297 return 0;
300 int PluginInstance::NPP_Write(NPStream* stream,
301 int offset,
302 int len,
303 void* buffer) {
304 DCHECK(npp_functions_ != 0);
305 DCHECK(npp_functions_->write != 0);
306 if (npp_functions_->write != 0) {
307 return npp_functions_->write(npp_, stream, offset, len, buffer);
309 return 0;
312 void PluginInstance::NPP_StreamAsFile(NPStream* stream, const char* fname) {
313 DCHECK(npp_functions_ != 0);
314 DCHECK(npp_functions_->asfile != 0);
315 if (npp_functions_->asfile != 0) {
316 npp_functions_->asfile(npp_, stream, fname);
319 // Creating a temporary FilePath instance on the stack as the explicit
320 // FilePath constructor with StringType as an argument causes a compiler
321 // error when invoked via vector push back.
322 base::FilePath file_name = base::FilePath::FromUTF8Unsafe(fname);
323 files_created_.push_back(file_name);
326 void PluginInstance::NPP_URLNotify(const char* url,
327 NPReason reason,
328 void* notifyData) {
329 DCHECK(npp_functions_ != 0);
330 DCHECK(npp_functions_->urlnotify != 0);
331 if (npp_functions_->urlnotify != 0) {
332 npp_functions_->urlnotify(npp_, url, reason, notifyData);
336 NPError PluginInstance::NPP_GetValue(NPPVariable variable, void* value) {
337 DCHECK(npp_functions_ != 0);
338 // getvalue is NULL for Shockwave
339 if (npp_functions_->getvalue != 0) {
340 return npp_functions_->getvalue(npp_, variable, value);
342 return NPERR_INVALID_FUNCTABLE_ERROR;
345 NPError PluginInstance::NPP_SetValue(NPNVariable variable, void* value) {
346 DCHECK(npp_functions_ != 0);
347 if (npp_functions_->setvalue != 0) {
348 return npp_functions_->setvalue(npp_, variable, value);
350 return NPERR_INVALID_FUNCTABLE_ERROR;
353 short PluginInstance::NPP_HandleEvent(void* event) {
354 DCHECK(npp_functions_ != 0);
355 DCHECK(npp_functions_->event != 0);
356 if (npp_functions_->event != 0) {
357 return npp_functions_->event(npp_, (void*)event);
359 return false;
362 bool PluginInstance::NPP_Print(NPPrint* platform_print) {
363 DCHECK(npp_functions_ != 0);
364 if (npp_functions_->print != 0) {
365 npp_functions_->print(npp_, platform_print);
366 return true;
368 return false;
371 void PluginInstance::NPP_URLRedirectNotify(const char* url, int32_t status,
372 void* notify_data) {
373 DCHECK(npp_functions_ != 0);
374 if (npp_functions_->urlredirectnotify != 0) {
375 npp_functions_->urlredirectnotify(npp_, url, status, notify_data);
379 void PluginInstance::SendJavaScriptStream(const GURL& url,
380 const std::string& result,
381 bool success,
382 int notify_id) {
383 bool notify;
384 void* notify_data;
385 GetNotifyData(notify_id, &notify, &notify_data);
387 if (success) {
388 PluginStringStream *stream =
389 new PluginStringStream(this, url, notify, notify_data);
390 AddStream(stream);
391 stream->SendToPlugin(result, "text/html");
392 } else {
393 // NOTE: Sending an empty stream here will crash MacroMedia
394 // Flash 9. Just send the URL Notify.
395 if (notify)
396 NPP_URLNotify(url.spec().c_str(), NPRES_DONE, notify_data);
400 void PluginInstance::DidReceiveManualResponse(const GURL& url,
401 const std::string& mime_type,
402 const std::string& headers,
403 uint32 expected_length,
404 uint32 last_modified) {
405 DCHECK(load_manually_);
407 plugin_data_stream_ = CreateStream(-1, url, mime_type, 0);
408 plugin_data_stream_->DidReceiveResponse(mime_type, headers, expected_length,
409 last_modified, true);
412 void PluginInstance::DidReceiveManualData(const char* buffer, int length) {
413 DCHECK(load_manually_);
414 if (plugin_data_stream_.get() != NULL) {
415 plugin_data_stream_->DidReceiveData(buffer, length, 0);
419 void PluginInstance::DidFinishManualLoading() {
420 DCHECK(load_manually_);
421 if (plugin_data_stream_.get() != NULL) {
422 plugin_data_stream_->DidFinishLoading(plugin_data_stream_->ResourceId());
423 plugin_data_stream_->Close(NPRES_DONE);
424 plugin_data_stream_ = NULL;
428 void PluginInstance::DidManualLoadFail() {
429 DCHECK(load_manually_);
430 if (plugin_data_stream_.get() != NULL) {
431 plugin_data_stream_->DidFail(plugin_data_stream_->ResourceId());
432 plugin_data_stream_ = NULL;
436 void PluginInstance::PluginThreadAsyncCall(void (*func)(void*),
437 void* user_data) {
438 message_loop_->PostTask(
439 FROM_HERE, base::Bind(&PluginInstance::OnPluginThreadAsyncCall, this,
440 func, user_data));
443 void PluginInstance::OnPluginThreadAsyncCall(void (*func)(void*),
444 void* user_data) {
445 // Do not invoke the callback if NPP_Destroy has already been invoked.
446 if (webplugin_)
447 func(user_data);
450 uint32 PluginInstance::ScheduleTimer(uint32 interval,
451 NPBool repeat,
452 void (*func)(NPP id, uint32 timer_id)) {
453 // Use next timer id.
454 uint32 timer_id;
455 timer_id = next_timer_id_;
456 ++next_timer_id_;
457 DCHECK(next_timer_id_ != 0);
459 // Record timer interval and repeat.
460 TimerInfo info;
461 info.interval = interval;
462 info.repeat = repeat ? true : false;
463 timers_[timer_id] = info;
465 // Schedule the callback.
466 base::MessageLoop::current()->PostDelayedTask(
467 FROM_HERE,
468 base::Bind(&PluginInstance::OnTimerCall, this, func, npp_, timer_id),
469 base::TimeDelta::FromMilliseconds(interval));
470 return timer_id;
473 void PluginInstance::UnscheduleTimer(uint32 timer_id) {
474 // Remove info about the timer.
475 TimerMap::iterator it = timers_.find(timer_id);
476 if (it != timers_.end())
477 timers_.erase(it);
480 #if !defined(OS_MACOSX)
481 NPError PluginInstance::PopUpContextMenu(NPMenu* menu) {
482 NOTIMPLEMENTED();
483 return NPERR_GENERIC_ERROR;
485 #endif
487 void PluginInstance::OnTimerCall(void (*func)(NPP id, uint32 timer_id),
488 NPP id,
489 uint32 timer_id) {
490 // Do not invoke callback if the timer has been unscheduled.
491 TimerMap::iterator it = timers_.find(timer_id);
492 if (it == timers_.end())
493 return;
495 // Get all information about the timer before invoking the callback. The
496 // callback might unschedule the timer.
497 TimerInfo info = it->second;
499 func(id, timer_id);
501 // If the timer was unscheduled by the callback, just free up the timer id.
502 if (timers_.find(timer_id) == timers_.end())
503 return;
505 // Reschedule repeating timers after invoking the callback so callback is not
506 // re-entered if it pumps the message loop.
507 if (info.repeat) {
508 base::MessageLoop::current()->PostDelayedTask(
509 FROM_HERE,
510 base::Bind(&PluginInstance::OnTimerCall, this, func, npp_, timer_id),
511 base::TimeDelta::FromMilliseconds(info.interval));
512 } else {
513 timers_.erase(it);
517 void PluginInstance::PushPopupsEnabledState(bool enabled) {
518 popups_enabled_stack_.push(enabled);
521 void PluginInstance::PopPopupsEnabledState() {
522 popups_enabled_stack_.pop();
525 void PluginInstance::RequestRead(NPStream* stream, NPByteRange* range_list) {
526 std::string range_info = "bytes=";
528 while (range_list) {
529 range_info += base::IntToString(range_list->offset);
530 range_info.push_back('-');
531 range_info +=
532 base::IntToString(range_list->offset + range_list->length - 1);
533 range_list = range_list->next;
534 if (range_list)
535 range_info.push_back(',');
538 if (plugin_data_stream_.get()) {
539 if (plugin_data_stream_->stream() == stream) {
540 webplugin_->CancelDocumentLoad();
541 plugin_data_stream_ = NULL;
545 // The lifetime of a NPStream instance depends on the PluginStream instance
546 // which owns it. When a plugin invokes NPN_RequestRead on a seekable stream,
547 // we don't want to create a new stream when the corresponding response is
548 // received. We send over a cookie which represents the PluginStream
549 // instance which is sent back from the renderer when the response is
550 // received.
551 std::vector<scoped_refptr<PluginStream> >::iterator stream_index;
552 for (stream_index = open_streams_.begin();
553 stream_index != open_streams_.end(); ++stream_index) {
554 PluginStream* plugin_stream = stream_index->get();
555 if (plugin_stream->stream() == stream) {
556 // A stream becomes seekable the first time NPN_RequestRead
557 // is called on it.
558 plugin_stream->set_seekable(true);
560 pending_range_requests_[++next_range_request_id_] = plugin_stream;
561 webplugin_->InitiateHTTPRangeRequest(
562 stream->url, range_info.c_str(), next_range_request_id_);
563 return;
566 NOTREACHED();
569 void PluginInstance::RequestURL(const char* url,
570 const char* method,
571 const char* target,
572 const char* buf,
573 unsigned int len,
574 bool notify,
575 void* notify_data) {
576 int notify_id = 0;
577 if (notify) {
578 notify_id = ++next_notify_id_;
579 pending_requests_[notify_id] = notify_data;
582 webplugin_->HandleURLRequest(
583 url, method, target, buf, len, notify_id, popups_allowed(),
584 notify ? handles_url_redirects_ : false);
587 bool PluginInstance::ConvertPoint(double source_x, double source_y,
588 NPCoordinateSpace source_space,
589 double* dest_x, double* dest_y,
590 NPCoordinateSpace dest_space) {
591 #if defined(OS_MACOSX)
592 CGRect main_display_bounds = CGDisplayBounds(CGMainDisplayID());
594 double flipped_screen_x = source_x;
595 double flipped_screen_y = source_y;
596 switch(source_space) {
597 case NPCoordinateSpacePlugin:
598 flipped_screen_x += plugin_origin_.x();
599 flipped_screen_y += plugin_origin_.y();
600 break;
601 case NPCoordinateSpaceWindow:
602 flipped_screen_x += containing_window_frame_.x();
603 flipped_screen_y = containing_window_frame_.height() - source_y +
604 containing_window_frame_.y();
605 break;
606 case NPCoordinateSpaceFlippedWindow:
607 flipped_screen_x += containing_window_frame_.x();
608 flipped_screen_y += containing_window_frame_.y();
609 break;
610 case NPCoordinateSpaceScreen:
611 flipped_screen_y = main_display_bounds.size.height - flipped_screen_y;
612 break;
613 case NPCoordinateSpaceFlippedScreen:
614 break;
615 default:
616 NOTREACHED();
617 return false;
620 double target_x = flipped_screen_x;
621 double target_y = flipped_screen_y;
622 switch(dest_space) {
623 case NPCoordinateSpacePlugin:
624 target_x -= plugin_origin_.x();
625 target_y -= plugin_origin_.y();
626 break;
627 case NPCoordinateSpaceWindow:
628 target_x -= containing_window_frame_.x();
629 target_y -= containing_window_frame_.y();
630 target_y = containing_window_frame_.height() - target_y;
631 break;
632 case NPCoordinateSpaceFlippedWindow:
633 target_x -= containing_window_frame_.x();
634 target_y -= containing_window_frame_.y();
635 break;
636 case NPCoordinateSpaceScreen:
637 target_y = main_display_bounds.size.height - flipped_screen_y;
638 break;
639 case NPCoordinateSpaceFlippedScreen:
640 break;
641 default:
642 NOTREACHED();
643 return false;
646 if (dest_x)
647 *dest_x = target_x;
648 if (dest_y)
649 *dest_y = target_y;
650 return true;
651 #else
652 NOTIMPLEMENTED();
653 return false;
654 #endif
657 void PluginInstance::GetNotifyData(int notify_id,
658 bool* notify,
659 void** notify_data) {
660 PendingRequestMap::iterator iter = pending_requests_.find(notify_id);
661 if (iter != pending_requests_.end()) {
662 *notify = true;
663 *notify_data = iter->second;
664 pending_requests_.erase(iter);
665 } else {
666 *notify = false;
667 *notify_data = NULL;
671 void PluginInstance::URLRedirectResponse(bool allow, void* notify_data) {
672 // The notify_data passed in allows us to identify the matching stream.
673 std::vector<scoped_refptr<PluginStream> >::iterator stream_index;
674 for (stream_index = open_streams_.begin();
675 stream_index != open_streams_.end(); ++stream_index) {
676 PluginStream* plugin_stream = stream_index->get();
677 if (plugin_stream->notify_data() == notify_data) {
678 PluginStreamUrl* plugin_stream_url =
679 static_cast<PluginStreamUrl*>(plugin_stream);
680 plugin_stream_url->URLRedirectResponse(allow);
681 break;
686 } // namespace content