2015-05-21 06:06:00 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#include "nsDNSServiceDiscovery.h"
|
|
|
|
#include "MDNSResponderOperator.h"
|
|
|
|
#include "nsICancelable.h"
|
2015-09-10 21:03:00 +03:00
|
|
|
#include "nsXULAppAPI.h"
|
2015-05-21 06:06:00 +03:00
|
|
|
#include "private/pprio.h"
|
|
|
|
|
2015-11-18 09:02:31 +03:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
#include <cutils/properties.h>
|
|
|
|
#endif // MOZ_WIDGET_GONK
|
|
|
|
|
2015-05-21 06:06:00 +03:00
|
|
|
namespace mozilla {
|
|
|
|
namespace net {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2015-09-10 21:03:00 +03:00
|
|
|
inline void
|
2015-05-21 06:06:00 +03:00
|
|
|
StartService()
|
|
|
|
{
|
2015-11-18 09:02:31 +03:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2015-05-21 06:06:00 +03:00
|
|
|
char value[PROPERTY_VALUE_MAX] = { '\0' };
|
|
|
|
property_get("init.svc.mdnsd", value, "");
|
|
|
|
|
|
|
|
if (strcmp(value, "running") == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
property_set("ctl.start", "mdnsd");
|
2015-11-18 09:02:31 +03:00
|
|
|
#endif // MOZ_WIDGET_GONK
|
2015-05-21 06:06:00 +03:00
|
|
|
}
|
|
|
|
|
2015-07-04 04:29:00 +03:00
|
|
|
inline void
|
2015-05-21 06:06:00 +03:00
|
|
|
StopService()
|
|
|
|
{
|
2015-11-18 09:02:31 +03:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2015-05-21 06:06:00 +03:00
|
|
|
char value[PROPERTY_VALUE_MAX] = { '\0' };
|
|
|
|
property_get("init.svc.mdnsd", value, "");
|
|
|
|
|
|
|
|
if (strcmp(value, "stopped") == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
property_set("ctl.stop", "mdnsd");
|
2015-11-18 09:02:31 +03:00
|
|
|
#endif // MOZ_WIDGET_GONK
|
2015-05-21 06:06:00 +03:00
|
|
|
}
|
|
|
|
|
2015-09-10 21:03:00 +03:00
|
|
|
class ServiceCounter
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static bool IsServiceRunning()
|
|
|
|
{
|
|
|
|
return !!sUseCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static uint32_t sUseCount;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
ServiceCounter()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!sUseCount++) {
|
|
|
|
StartService();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual ~ServiceCounter()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!--sUseCount) {
|
|
|
|
StopService();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
uint32_t ServiceCounter::sUseCount = 0;
|
|
|
|
|
2015-05-21 06:06:00 +03:00
|
|
|
class DiscoveryRequest final : public nsICancelable
|
2015-09-10 21:03:00 +03:00
|
|
|
, private ServiceCounter
|
2015-05-21 06:06:00 +03:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
NS_DECL_NSICANCELABLE
|
|
|
|
|
|
|
|
explicit DiscoveryRequest(nsDNSServiceDiscovery* aService,
|
|
|
|
nsIDNSServiceDiscoveryListener* aListener);
|
|
|
|
|
|
|
|
private:
|
|
|
|
virtual ~DiscoveryRequest() { Cancel(NS_OK); }
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<nsDNSServiceDiscovery> mService;
|
2015-05-21 06:06:00 +03:00
|
|
|
nsIDNSServiceDiscoveryListener* mListener;
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(DiscoveryRequest, nsICancelable)
|
|
|
|
|
|
|
|
DiscoveryRequest::DiscoveryRequest(nsDNSServiceDiscovery* aService,
|
|
|
|
nsIDNSServiceDiscoveryListener* aListener)
|
|
|
|
: mService(aService)
|
|
|
|
, mListener(aListener)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DiscoveryRequest::Cancel(nsresult aReason)
|
|
|
|
{
|
|
|
|
if (mService) {
|
|
|
|
mService->StopDiscovery(mListener);
|
|
|
|
}
|
|
|
|
|
|
|
|
mService = nullptr;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
class RegisterRequest final : public nsICancelable
|
2015-09-10 21:03:00 +03:00
|
|
|
, private ServiceCounter
|
2015-05-21 06:06:00 +03:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
NS_DECL_NSICANCELABLE
|
|
|
|
|
|
|
|
explicit RegisterRequest(nsDNSServiceDiscovery* aService,
|
|
|
|
nsIDNSRegistrationListener* aListener);
|
|
|
|
|
|
|
|
private:
|
|
|
|
virtual ~RegisterRequest() { Cancel(NS_OK); }
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<nsDNSServiceDiscovery> mService;
|
2015-05-21 06:06:00 +03:00
|
|
|
nsIDNSRegistrationListener* mListener;
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(RegisterRequest, nsICancelable)
|
|
|
|
|
|
|
|
RegisterRequest::RegisterRequest(nsDNSServiceDiscovery* aService,
|
|
|
|
nsIDNSRegistrationListener* aListener)
|
|
|
|
: mService(aService)
|
|
|
|
, mListener(aListener)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
RegisterRequest::Cancel(nsresult aReason)
|
|
|
|
{
|
|
|
|
if (mService) {
|
|
|
|
mService->UnregisterService(mListener);
|
|
|
|
}
|
|
|
|
|
|
|
|
mService = nullptr;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace anonymous
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(nsDNSServiceDiscovery, nsIDNSServiceDiscovery)
|
|
|
|
|
2015-09-02 08:23:00 +03:00
|
|
|
nsDNSServiceDiscovery::~nsDNSServiceDiscovery()
|
|
|
|
{
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
StopService();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-05-21 06:06:00 +03:00
|
|
|
nsresult
|
|
|
|
nsDNSServiceDiscovery::Init()
|
|
|
|
{
|
2015-09-10 21:03:00 +03:00
|
|
|
if (!XRE_IsParentProcess()) {
|
|
|
|
MOZ_ASSERT(false, "nsDNSServiceDiscovery can only be used in parent process");
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2015-05-21 06:06:00 +03:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsDNSServiceDiscovery::StartDiscovery(const nsACString& aServiceType,
|
|
|
|
nsIDNSServiceDiscoveryListener* aListener,
|
|
|
|
nsICancelable** aRetVal)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aRetVal);
|
|
|
|
|
|
|
|
nsresult rv;
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = StopDiscovery(aListener)))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2015-09-10 21:03:00 +03:00
|
|
|
nsCOMPtr<nsICancelable> req = new DiscoveryRequest(this, aListener);
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<BrowseOperator> browserOp = new BrowseOperator(aServiceType,
|
2015-05-21 06:06:00 +03:00
|
|
|
aListener);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = browserOp->Start()))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
mDiscoveryMap.Put(aListener, browserOp);
|
|
|
|
|
|
|
|
req.forget(aRetVal);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsDNSServiceDiscovery::StopDiscovery(nsIDNSServiceDiscoveryListener* aListener)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<BrowseOperator> browserOp;
|
2015-05-21 06:06:00 +03:00
|
|
|
if (!mDiscoveryMap.Get(aListener, getter_AddRefs(browserOp))) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
browserOp->Cancel(); // cancel non-started operation
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = browserOp->Stop()))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
mDiscoveryMap.Remove(aListener);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsDNSServiceDiscovery::RegisterService(nsIDNSServiceInfo* aServiceInfo,
|
|
|
|
nsIDNSRegistrationListener* aListener,
|
|
|
|
nsICancelable** aRetVal)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aRetVal);
|
|
|
|
|
|
|
|
nsresult rv;
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = UnregisterService(aListener)))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2015-09-10 21:03:00 +03:00
|
|
|
nsCOMPtr<nsICancelable> req = new RegisterRequest(this, aListener);
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<RegisterOperator> registerOp = new RegisterOperator(aServiceInfo,
|
2015-05-21 06:06:00 +03:00
|
|
|
aListener);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = registerOp->Start()))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
mRegisterMap.Put(aListener, registerOp);
|
|
|
|
|
|
|
|
req.forget(aRetVal);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsDNSServiceDiscovery::UnregisterService(nsIDNSRegistrationListener* aListener)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<RegisterOperator> registerOp;
|
2015-05-21 06:06:00 +03:00
|
|
|
if (!mRegisterMap.Get(aListener, getter_AddRefs(registerOp))) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
registerOp->Cancel(); // cancel non-started operation
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = registerOp->Stop()))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
mRegisterMap.Remove(aListener);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsDNSServiceDiscovery::ResolveService(nsIDNSServiceInfo* aServiceInfo,
|
|
|
|
nsIDNSServiceResolveListener* aListener)
|
|
|
|
{
|
2015-09-10 21:03:00 +03:00
|
|
|
if (!ServiceCounter::IsServiceRunning()) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2015-05-21 06:06:00 +03:00
|
|
|
nsresult rv;
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<ResolveOperator> resolveOp = new ResolveOperator(aServiceInfo,
|
2015-05-21 06:06:00 +03:00
|
|
|
aListener);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv = resolveOp->Start()))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace net
|
|
|
|
} // namespace mozilla
|