#firefox #npapi
#firefox #npapi
Вопрос:
Я пытаюсь реализовать плагин NPAPI с возможностями потоковой передачи (NP_SEEK NPN_RequestRead). Независимо от того, что я пытаюсь, логическое NPBool seekable
значение всегда равно 0 (false).
Я запускаю firefox (iceweasel в Debian) из командной строки в локальном файле:
$ iceweasel test1.html
Если я подключу gdb к плагину npapi, вот что я вижу:
(gdb)
#2 0x00007f7e9da54e14 in mozilla::plugins::BrowserStreamChild::StreamConstructed (this=0x7f7e925cf310, mimeType=..., seekable=<optimized out>, stype=<optimized out>)
at /tmp/buildd/iceweasel-24.6.0esr/dom/plugins/ipc/BrowserStreamChild.cpp:62
62 amp;mStream, seekable, stype);
(gdb)
#3 0x00007f7e9da5688e in mozilla::plugins::PluginInstanceChild::AnswerPBrowserStreamConstructor (this=<optimized out>, aActor=<optimized out>, url=...,
length=<optimized out>, lastmodified=<optimized out>, notifyData=<optimized out>, headers=..., mimeType=..., seekable=@0x7fff25ed51df: false, rv=0x7fff25ed51e0,
stype=0x7fff25ed51e2) at /tmp/buildd/iceweasel-24.6.0esr/dom/plugins/ipc/PluginInstanceChild.cpp:2285
2285 ->StreamConstructed(mimeType, seekable, stype);
(gdb) p seekable
$1 = (const bool amp;) @0x7fff25ed51df: false
(gdb) up
#4 0x00007f7e9da8f77f in mozilla::plugins::PPluginInstanceChild::OnCallReceived (this=0x7f7e925f2c00, __msg=..., __reply=@0x7fff25ed5470: 0x0)
at /tmp/buildd/iceweasel-24.6.0esr/build-xulrunner/ipc/ipdl/PPluginInstanceChild.cpp:2479
warning: Source file is more recent than executable.
2479 if ((!(AnswerPBrowserStreamConstructor(actor, url, length, lastmodified, notifyData, headers, mimeType, seekable, (amp;(rv)), (amp;(stype)))))) {
(gdb) list -
2469 if ((!(actor))) {
2470 return MsgValueError;
2471 }
2472 (actor)->mId = RegisterID(actor, (__handle).mId);
2473 (actor)->mManager = this;
2474 (actor)->mChannel = mChannel;
2475 (mManagedPBrowserStreamChild).InsertElementSorted(actor);
2476 (actor)->mState = mozilla::plugins::PBrowserStream::__Start;
2477
2478 int32_t __id = mId;
(gdb) list -
2459 FatalError("Error deserializing 'bool'");
2460 return MsgValueError;
2461 }
2462 (__msg).EndRead(__iter);
2463 if ((!(PPluginInstance::Transition(mState, Trigger(Trigger::Send, PPluginInstance::Msg_PBrowserStreamConstructor__ID), (amp;(mState)))))) {
2464 NS_WARNING("bad state transition!");
2465 }
2466 NPError rv;
2467 uint16_t stype;
2468 actor = AllocPBrowserStream(url, length, lastmodified, notifyData, headers, mimeType, seekable, (amp;(rv)), (amp;(stype)));
(gdb) list -
2449 }
2450 if ((!(Read((amp;(headers)), (amp;(__msg)), (amp;(__iter)))))) {
2451 FatalError("Error deserializing 'nsCString'");
2452 return MsgValueError;
2453 }
2454 if ((!(Read((amp;(mimeType)), (amp;(__msg)), (amp;(__iter)))))) {
2455 FatalError("Error deserializing 'nsCString'");
2456 return MsgValueError;
2457 }
2458 if ((!(Read((amp;(seekable)), (amp;(__msg)), (amp;(__iter)))))) {
(gdb) up
#5 0x00007f7e9da868f0 in mozilla::plugins::PPluginModuleChild::OnCallReceived (this=<optimized out>, __msg=..., __reply=@0x7fff25ed5470: 0x0)
at /tmp/buildd/iceweasel-24.6.0esr/build-xulrunner/ipc/ipdl/PPluginModuleChild.cpp:1023
warning: Source file is more recent than executable.
1023 return (__routed)->OnCallReceived(__msg, __reply);
(gdb) list -
1013 PPluginModuleChild::OnCallReceived(
1014 const Messageamp; __msg,
1015 Message*amp; __reply)
1016 {
1017 int32_t __route = (__msg).routing_id();
1018 if ((MSG_ROUTING_CONTROL) != (__route)) {
1019 ChannelListener* __routed = Lookup(__route);
1020 if ((!(__routed))) {
1021 return MsgRouteError;
1022 }
(gdb) bt
Если я скопирую test1.html
/var/www
, а затем укажу на http://localhost/test1.html
, все работает, как ожидалось.
Однако в документации упоминается
seekable
Boolean indicating whether the stream is seekable:
true: Seekable. Stream supports random access through calls to NPN_RequestRead (for example, local files or HTTP servers that support byte-range requests).
Ответ №1:
Документация откровенно лжет.
seekable
Флаг в вызовеNPP_NewStream
- происходит из
OnStartBinding
- который вызывает
nsPluginStreamListenerPeer::IsSeekable
- который просто возвращает
nsPluginStreamListenerPeer::mSeekable
.
- происходит из
- Устанавливается только время
mSeekable
true
, когда (источник)- Поток http (https, spdy)
- HTTP-ответ не имеет
Content-Encoding
- HTTP-ответ предоставляет
Content-Length
. - HTTP-ответ имеет
Accept-Ranges: bytes
(пропуск заголовка не поддерживается)
- Следовательно, для всех других типов потоков (включая
file://
) и HTTP-потоков, не соответствующих требованиямseekable
, флаг установлен всегдаfalse
.
Более того, NPN_RequestRead
реализован только для HTTP-потоков, но на самом деле не заботится seekable
и, кроме того, фактически не проверяет, возвращается ли сервер 206
.
Заключение
Вы можете использовать NP_SEEK
потоки только с http (https, spdy). Вот почему материал работает из http://localhost
, но не из локального файла ( file://
).