aboutsummaryrefslogtreecommitdiff
path: root/gnu
diff options
context:
space:
mode:
Diffstat (limited to 'gnu')
-rw-r--r--gnu/packages/gnuzilla.scm26
-rw-r--r--gnu/packages/patches/icecat-CVE-2015-4477.patch37
-rw-r--r--gnu/packages/patches/icecat-CVE-2015-7207.patch1140
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt01.patch356
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt02.patch58
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt03.patch60
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt04.patch53
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt05.patch32
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1952-pt06.patch103
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1954.patch32
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1960.patch55
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1961.patch33
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1962.patch107
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1964.patch54
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1965.patch44
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1966.patch36
-rw-r--r--gnu/packages/patches/icecat-CVE-2016-1974.patch530
-rw-r--r--gnu/packages/patches/icecat-bug-1248851.patch37
-rw-r--r--gnu/packages/patches/icecat-update-graphite2-pt2.patch861
-rw-r--r--gnu/packages/patches/icecat-update-graphite2.patch9988
20 files changed, 4 insertions, 13638 deletions
diff --git a/gnu/packages/gnuzilla.scm b/gnu/packages/gnuzilla.scm
index b4892d77cd..1ae97b256b 100644
--- a/gnu/packages/gnuzilla.scm
+++ b/gnu/packages/gnuzilla.scm
@@ -3,6 +3,7 @@
;;; Copyright © 2013, 2014, 2015 Ludovic Courtès <ludo@gnu.org>
;;; Copyright © 2014, 2015, 2016 Mark H Weaver <mhw@netris.org>
;;; Copyright © 2015 Sou Bunnbu <iyzsong@gmail.com>
+;;; Copyright © 2016 Efraim Flashner <efraim@flashner.co.il>
;;;
;;; This file is part of GNU Guix.
;;;
@@ -277,7 +278,7 @@ standards.")
(define-public icecat
(package
(name "icecat")
- (version "38.6.0-gnu1")
+ (version "38.7.1-gnu1")
(source
(origin
(method url-fetch)
@@ -286,29 +287,10 @@ standards.")
name "-" version ".tar.bz2"))
(sha256
(base32
- "0bd4k5cwr8ynscaxffvj2x3kgky3dmjq0qhpcb931l98bh0103lx"))
+ "1wdmd6hasra36g86ha1dw8sl7a5mvr7c4jbjx4zyg9629y5gqr8g"))
(patches (map search-patch
'("icecat-avoid-bundled-includes.patch"
- "icecat-re-enable-DHE-cipher-suites.patch"
- "icecat-update-graphite2.patch"
- "icecat-update-graphite2-pt2.patch"
- "icecat-CVE-2015-4477.patch"
- "icecat-CVE-2015-7207.patch"
- "icecat-CVE-2016-1952-pt01.patch"
- "icecat-CVE-2016-1952-pt02.patch"
- "icecat-CVE-2016-1952-pt03.patch"
- "icecat-CVE-2016-1952-pt04.patch"
- "icecat-CVE-2016-1952-pt05.patch"
- "icecat-CVE-2016-1952-pt06.patch"
- "icecat-CVE-2016-1954.patch"
- "icecat-CVE-2016-1960.patch"
- "icecat-CVE-2016-1961.patch"
- "icecat-CVE-2016-1962.patch"
- "icecat-CVE-2016-1964.patch"
- "icecat-CVE-2016-1965.patch"
- "icecat-CVE-2016-1966.patch"
- "icecat-CVE-2016-1974.patch"
- "icecat-bug-1248851.patch")))
+ "icecat-re-enable-DHE-cipher-suites.patch")))
(modules '((guix build utils)))
(snippet
'(begin
diff --git a/gnu/packages/patches/icecat-CVE-2015-4477.patch b/gnu/packages/patches/icecat-CVE-2015-4477.patch
deleted file mode 100644
index c010c5ecec..0000000000
--- a/gnu/packages/patches/icecat-CVE-2015-4477.patch
+++ /dev/null
@@ -1,37 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/beae8783b8c2
-
-# HG changeset patch
-# User Paul Adenot <paul@paul.cx>
-# Date 1456422965 0
-# Node ID beae8783b8c2c672da12a95c70ae663cbd0d5016
-# Parent 3a606f8182c82480f8f350b622ab55a170ec1eb6
-Bug 1179484. r=roc
-
-MozReview-Commit-ID: HNaYLyMe3sM
-
-diff --git a/dom/media/webaudio/MediaStreamAudioDestinationNode.cpp b/dom/media/webaudio/MediaStreamAudioDestinationNode.cpp
---- a/dom/media/webaudio/MediaStreamAudioDestinationNode.cpp
-+++ b/dom/media/webaudio/MediaStreamAudioDestinationNode.cpp
-@@ -69,16 +69,20 @@ MediaStreamAudioDestinationNode::MediaSt
- ChannelInterpretation::Speakers)
- , mDOMStream(DOMAudioNodeMediaStream::CreateTrackUnionStream(GetOwner(),
- this))
- {
- TrackUnionStream* tus = static_cast<TrackUnionStream*>(mDOMStream->GetStream());
- MOZ_ASSERT(tus == mDOMStream->GetStream()->AsProcessedStream());
- tus->SetTrackIDFilter(FilterAudioNodeStreamTrack);
-
-+ if (aContext->Graph() != tus->Graph()) {
-+ return;
-+ }
-+
- MediaStreamDestinationEngine* engine = new MediaStreamDestinationEngine(this, tus);
- mStream = aContext->Graph()->CreateAudioNodeStream(engine, MediaStreamGraph::INTERNAL_STREAM);
- mPort = tus->AllocateInputPort(mStream, 0);
-
- nsIDocument* doc = aContext->GetParentObject()->GetExtantDoc();
- if (doc) {
- mDOMStream->CombineWithPrincipal(doc->NodePrincipal());
- }
-
diff --git a/gnu/packages/patches/icecat-CVE-2015-7207.patch b/gnu/packages/patches/icecat-CVE-2015-7207.patch
deleted file mode 100644
index db5fc6ce66..0000000000
--- a/gnu/packages/patches/icecat-CVE-2015-7207.patch
+++ /dev/null
@@ -1,1140 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/532544c91db7
-
-# HG changeset patch
-# User Dragana Damjanovic <dd.mozilla@gmail.com>
-# Date 1456962626 28800
-# Node ID 532544c91db7f13c39be1b7b7c4461cd03126e9c
-# Parent f4220254d5bd0851a439467da39ba431e0ce2804
-Bug 1185256 - Save originURI to the history. r=bz ba=ritu
-
-MozReview-Commit-ID: Lvh9C84RQUc
-
-diff --git a/docshell/base/nsDocShell.cpp b/docshell/base/nsDocShell.cpp
---- a/docshell/base/nsDocShell.cpp
-+++ b/docshell/base/nsDocShell.cpp
-@@ -1020,16 +1020,17 @@ nsDocShell::DestroyChildren()
- //*****************************************************************************
- // nsDocShell::nsISupports
- //*****************************************************************************
-
- NS_IMPL_ADDREF_INHERITED(nsDocShell, nsDocLoader)
- NS_IMPL_RELEASE_INHERITED(nsDocShell, nsDocLoader)
-
- NS_INTERFACE_MAP_BEGIN(nsDocShell)
-+ NS_INTERFACE_MAP_ENTRY(nsIDocShell_ESR38_2)
- NS_INTERFACE_MAP_ENTRY(nsIDocShell_ESR38)
- NS_INTERFACE_MAP_ENTRY(nsIDocShell)
- NS_INTERFACE_MAP_ENTRY(nsIDocShellTreeItem)
- NS_INTERFACE_MAP_ENTRY(nsIWebNavigation)
- NS_INTERFACE_MAP_ENTRY(nsIBaseWindow)
- NS_INTERFACE_MAP_ENTRY(nsIScrollable)
- NS_INTERFACE_MAP_ENTRY(nsITextScroll)
- NS_INTERFACE_MAP_ENTRY(nsIDocCharset)
-@@ -1372,16 +1373,17 @@ nsDocShell::LoadURI(nsIURI* aURI,
- return NS_OK; // JS may not handle returning of an error code
- }
-
- if (DoAppRedirectIfNeeded(aURI, aLoadInfo, aFirstParty)) {
- return NS_OK;
- }
-
- nsCOMPtr<nsIURI> referrer;
-+ nsCOMPtr<nsIURI> originalURI;
- nsCOMPtr<nsIInputStream> postStream;
- nsCOMPtr<nsIInputStream> headersStream;
- nsCOMPtr<nsISupports> owner;
- bool inheritOwner = false;
- bool ownerIsExplicit = false;
- bool sendReferrer = true;
- uint32_t referrerPolicy = mozilla::net::RP_Default;
- bool isSrcdoc = false;
-@@ -1398,16 +1400,20 @@ nsDocShell::LoadURI(nsIURI* aURI,
- if (!StartupTimeline::HasRecord(StartupTimeline::FIRST_LOAD_URI) &&
- mItemType == typeContent && !NS_IsAboutBlank(aURI)) {
- StartupTimeline::RecordOnce(StartupTimeline::FIRST_LOAD_URI);
- }
-
- // Extract the info from the DocShellLoadInfo struct...
- if (aLoadInfo) {
- aLoadInfo->GetReferrer(getter_AddRefs(referrer));
-+ nsCOMPtr<nsIDocShellLoadInfo_ESR38> liESR38 = do_QueryInterface(aLoadInfo);
-+ if (liESR38) {
-+ liESR38->GetOriginalURI(getter_AddRefs(originalURI));
-+ }
-
- nsDocShellInfoLoadType lt = nsIDocShellLoadInfo::loadNormal;
- aLoadInfo->GetLoadType(&lt);
- // Get the appropriate loadType from nsIDocShellLoadInfo type
- loadType = ConvertDocShellLoadInfoToLoadType(lt);
-
- aLoadInfo->GetOwner(getter_AddRefs(owner));
- aLoadInfo->GetInheritOwner(&inheritOwner);
-@@ -1652,34 +1658,35 @@ nsDocShell::LoadURI(nsIURI* aURI,
- if (aLoadFlags & LOAD_FLAGS_FORCE_ALLOW_COOKIES) {
- flags |= INTERNAL_LOAD_FLAGS_FORCE_ALLOW_COOKIES;
- }
-
- if (isSrcdoc) {
- flags |= INTERNAL_LOAD_FLAGS_IS_SRCDOC;
- }
-
-- return InternalLoad(aURI,
-- referrer,
-- referrerPolicy,
-- owner,
-- flags,
-- target.get(),
-- nullptr, // No type hint
-- NullString(), // No forced download
-- postStream,
-- headersStream,
-- loadType,
-- nullptr, // No SHEntry
-- aFirstParty,
-- srcdoc,
-- sourceDocShell,
-- baseURI,
-- nullptr, // No nsIDocShell
-- nullptr); // No nsIRequest
-+ return InternalLoad2(aURI,
-+ originalURI,
-+ referrer,
-+ referrerPolicy,
-+ owner,
-+ flags,
-+ target.get(),
-+ nullptr, // No type hint
-+ NullString(), // No forced download
-+ postStream,
-+ headersStream,
-+ loadType,
-+ nullptr, // No SHEntry
-+ aFirstParty,
-+ srcdoc,
-+ sourceDocShell,
-+ baseURI,
-+ nullptr, // No nsIDocShell
-+ nullptr); // No nsIRequest
- }
-
- NS_IMETHODIMP
- nsDocShell::LoadStream(nsIInputStream* aStream, nsIURI* aURI,
- const nsACString& aContentType,
- const nsACString& aContentCharset,
- nsIDocShellLoadInfo* aLoadInfo)
- {
-@@ -5398,21 +5405,21 @@ nsDocShell::LoadErrorPage(nsIURI* aURI,
- // end of the URL, so append it last.
- errorPageUrl.AppendLiteral("&d=");
- errorPageUrl.AppendASCII(escapedDescription.get());
-
- nsCOMPtr<nsIURI> errorPageURI;
- rv = NS_NewURI(getter_AddRefs(errorPageURI), errorPageUrl);
- NS_ENSURE_SUCCESS(rv, rv);
-
-- return InternalLoad(errorPageURI, nullptr, mozilla::net::RP_Default,
-- nullptr, INTERNAL_LOAD_FLAGS_INHERIT_OWNER, nullptr,
-- nullptr, NullString(), nullptr, nullptr, LOAD_ERROR_PAGE,
-- nullptr, true, NullString(), this, nullptr, nullptr,
-- nullptr);
-+ return InternalLoad2(errorPageURI, nullptr, nullptr, mozilla::net::RP_Default,
-+ nullptr, INTERNAL_LOAD_FLAGS_INHERIT_OWNER, nullptr,
-+ nullptr, NullString(), nullptr, nullptr, LOAD_ERROR_PAGE,
-+ nullptr, true, NullString(), this, nullptr, nullptr,
-+ nullptr);
- }
-
- NS_IMETHODIMP
- nsDocShell::Reload(uint32_t aReloadFlags)
- {
- if (!IsNavigationAllowed()) {
- return NS_OK; // JS may not handle returning of an error code
- }
-@@ -5448,44 +5455,54 @@ nsDocShell::Reload(uint32_t aReloadFlags
- nsCOMPtr<nsIDocument> doc(GetDocument());
-
- // Do not inherit owner from document
- uint32_t flags = INTERNAL_LOAD_FLAGS_NONE;
- nsAutoString srcdoc;
- nsIPrincipal* principal = nullptr;
- nsAutoString contentTypeHint;
- nsCOMPtr<nsIURI> baseURI;
-+ nsCOMPtr<nsIURI> originalURI;
- if (doc) {
- principal = doc->NodePrincipal();
- doc->GetContentType(contentTypeHint);
-
- if (doc->IsSrcdocDocument()) {
- doc->GetSrcdocData(srcdoc);
- flags |= INTERNAL_LOAD_FLAGS_IS_SRCDOC;
- baseURI = doc->GetBaseURI();
- }
-- }
-- rv = InternalLoad(mCurrentURI,
-- mReferrerURI,
-- mReferrerPolicy,
-- principal,
-- flags,
-- nullptr, // No window target
-- NS_LossyConvertUTF16toASCII(contentTypeHint).get(),
-- NullString(), // No forced download
-- nullptr, // No post data
-- nullptr, // No headers data
-- loadType, // Load type
-- nullptr, // No SHEntry
-- true,
-- srcdoc, // srcdoc argument for iframe
-- this, // For reloads we are the source
-- baseURI,
-- nullptr, // No nsIDocShell
-- nullptr); // No nsIRequest
-+ nsCOMPtr<nsIChannel> chan = doc->GetChannel();
-+ if (chan) {
-+ nsCOMPtr<nsIHttpChannel> httpChan(do_QueryInterface(chan));
-+ if (httpChan) {
-+ httpChan->GetOriginalURI(getter_AddRefs(originalURI));
-+ }
-+ }
-+ }
-+
-+ rv = InternalLoad2(mCurrentURI,
-+ originalURI,
-+ mReferrerURI,
-+ mReferrerPolicy,
-+ principal,
-+ flags,
-+ nullptr, // No window target
-+ NS_LossyConvertUTF16toASCII(contentTypeHint).get(),
-+ NullString(), // No forced download
-+ nullptr, // No post data
-+ nullptr, // No headers data
-+ loadType, // Load type
-+ nullptr, // No SHEntry
-+ true,
-+ srcdoc, // srcdoc argument for iframe
-+ this, // For reloads we are the source
-+ baseURI,
-+ nullptr, // No nsIDocShell
-+ nullptr); // No nsIRequest
- }
-
- return rv;
- }
-
- NS_IMETHODIMP
- nsDocShell::Stop(uint32_t aStopFlags)
- {
-@@ -9463,27 +9480,28 @@ CopyFavicon(nsIURI* aOldURI, nsIURI* aNe
- #endif
- }
-
- } // anonymous namespace
-
- class InternalLoadEvent : public nsRunnable
- {
- public:
-- InternalLoadEvent(nsDocShell* aDocShell, nsIURI* aURI,
-+ InternalLoadEvent(nsDocShell* aDocShell, nsIURI* aURI, nsIURI* aOriginalURI,
- nsIURI* aReferrer, uint32_t aReferrerPolicy,
- nsISupports* aOwner, uint32_t aFlags,
- const char* aTypeHint, nsIInputStream* aPostData,
- nsIInputStream* aHeadersData, uint32_t aLoadType,
- nsISHEntry* aSHEntry, bool aFirstParty,
- const nsAString& aSrcdoc, nsIDocShell* aSourceDocShell,
- nsIURI* aBaseURI)
- : mSrcdoc(aSrcdoc)
- , mDocShell(aDocShell)
- , mURI(aURI)
-+ , mOriginalURI(aOriginalURI)
- , mReferrer(aReferrer)
- , mReferrerPolicy(aReferrerPolicy)
- , mOwner(aOwner)
- , mPostData(aPostData)
- , mHeadersData(aHeadersData)
- , mSHEntry(aSHEntry)
- , mFlags(aFlags)
- , mLoadType(aLoadType)
-@@ -9494,34 +9512,36 @@ public:
- // Make sure to keep null things null as needed
- if (aTypeHint) {
- mTypeHint = aTypeHint;
- }
- }
-
- NS_IMETHOD Run()
- {
-- return mDocShell->InternalLoad(mURI, mReferrer,
-- mReferrerPolicy,
-- mOwner, mFlags,
-- nullptr, mTypeHint.get(),
-- NullString(), mPostData, mHeadersData,
-- mLoadType, mSHEntry, mFirstParty,
-- mSrcdoc, mSourceDocShell, mBaseURI,
-- nullptr, nullptr);
-+ return mDocShell->InternalLoad2(mURI, mOriginalURI,
-+ mReferrer,
-+ mReferrerPolicy,
-+ mOwner, mFlags,
-+ nullptr, mTypeHint.get(),
-+ NullString(), mPostData, mHeadersData,
-+ mLoadType, mSHEntry, mFirstParty,
-+ mSrcdoc, mSourceDocShell, mBaseURI,
-+ nullptr, nullptr);
- }
-
- private:
- // Use IDL strings so .get() returns null by default
- nsXPIDLString mWindowTarget;
- nsXPIDLCString mTypeHint;
- nsString mSrcdoc;
-
- nsRefPtr<nsDocShell> mDocShell;
- nsCOMPtr<nsIURI> mURI;
-+ nsCOMPtr<nsIURI> mOriginalURI;
- nsCOMPtr<nsIURI> mReferrer;
- uint32_t mReferrerPolicy;
- nsCOMPtr<nsISupports> mOwner;
- nsCOMPtr<nsIInputStream> mPostData;
- nsCOMPtr<nsIInputStream> mHeadersData;
- nsCOMPtr<nsISHEntry> mSHEntry;
- uint32_t mFlags;
- uint32_t mLoadType;
-@@ -9584,16 +9604,43 @@ nsDocShell::InternalLoad(nsIURI* aURI,
- nsISHEntry* aSHEntry,
- bool aFirstParty,
- const nsAString& aSrcdoc,
- nsIDocShell* aSourceDocShell,
- nsIURI* aBaseURI,
- nsIDocShell** aDocShell,
- nsIRequest** aRequest)
- {
-+ return InternalLoad2(aURI, nullptr, aReferrer, aReferrerPolicy, aOwner,
-+ aFlags, aWindowTarget, aTypeHint, aFileName, aPostData,
-+ aHeadersData, aLoadType, aSHEntry, aFirstParty, aSrcdoc,
-+ aSourceDocShell, aBaseURI, aDocShell, aRequest);
-+}
-+
-+NS_IMETHODIMP
-+nsDocShell::InternalLoad2(nsIURI* aURI,
-+ nsIURI* aOriginalURI,
-+ nsIURI* aReferrer,
-+ uint32_t aReferrerPolicy,
-+ nsISupports* aOwner,
-+ uint32_t aFlags,
-+ const char16_t* aWindowTarget,
-+ const char* aTypeHint,
-+ const nsAString& aFileName,
-+ nsIInputStream* aPostData,
-+ nsIInputStream* aHeadersData,
-+ uint32_t aLoadType,
-+ nsISHEntry* aSHEntry,
-+ bool aFirstParty,
-+ const nsAString& aSrcdoc,
-+ nsIDocShell* aSourceDocShell,
-+ nsIURI* aBaseURI,
-+ nsIDocShell** aDocShell,
-+ nsIRequest** aRequest)
-+{
- nsresult rv = NS_OK;
- mOriginalUriString.Truncate();
-
- #ifdef PR_LOGGING
- if (gDocShellLeakLog && PR_LOG_TEST(gDocShellLeakLog, PR_LOG_DEBUG)) {
- nsAutoCString spec;
- if (aURI) {
- aURI->GetSpec(spec);
-@@ -9831,34 +9878,58 @@ nsDocShell::InternalLoad(nsIURI* aURI,
- targetDocShell = do_QueryInterface(webNav);
- }
-
- //
- // Transfer the load to the target DocShell... Pass nullptr as the
- // window target name from to prevent recursive retargeting!
- //
- if (NS_SUCCEEDED(rv) && targetDocShell) {
-- rv = targetDocShell->InternalLoad(aURI,
-- aReferrer,
-- aReferrerPolicy,
-- owner,
-- aFlags,
-- nullptr, // No window target
-- aTypeHint,
-- NullString(), // No forced download
-- aPostData,
-- aHeadersData,
-- aLoadType,
-- aSHEntry,
-- aFirstParty,
-- aSrcdoc,
-- aSourceDocShell,
-- aBaseURI,
-- aDocShell,
-- aRequest);
-+ nsCOMPtr<nsIDocShell_ESR38_2> dsESR38 = do_QueryInterface(targetDocShell);
-+ if (dsESR38) {
-+ rv = dsESR38->InternalLoad2(aURI,
-+ aOriginalURI,
-+ aReferrer,
-+ aReferrerPolicy,
-+ owner,
-+ aFlags,
-+ nullptr, // No window target
-+ aTypeHint,
-+ NullString(), // No forced download
-+ aPostData,
-+ aHeadersData,
-+ aLoadType,
-+ aSHEntry,
-+ aFirstParty,
-+ aSrcdoc,
-+ aSourceDocShell,
-+ aBaseURI,
-+ aDocShell,
-+ aRequest);
-+ } else {
-+ rv = targetDocShell->InternalLoad(aURI,
-+ aReferrer,
-+ aReferrerPolicy,
-+ owner,
-+ aFlags,
-+ nullptr, // No window target
-+ aTypeHint,
-+ NullString(), // No forced download
-+ aPostData,
-+ aHeadersData,
-+ aLoadType,
-+ aSHEntry,
-+ aFirstParty,
-+ aSrcdoc,
-+ aSourceDocShell,
-+ aBaseURI,
-+ aDocShell,
-+ aRequest);
-+ }
-+
- if (rv == NS_ERROR_NO_CONTENT) {
- // XXXbz except we never reach this code!
- if (isNewWindow) {
- //
- // At this point, a new window has been created, but the
- // URI did not have any data associated with it...
- //
- // So, the best we can do, is to tear down the new window
-@@ -9913,17 +9984,17 @@ nsDocShell::InternalLoad(nsIURI* aURI,
- // the unload event also a replace load, so we don't
- // create extra history entries.
- if (LOAD_TYPE_HAS_FLAGS(aLoadType, LOAD_FLAGS_REPLACE_HISTORY)) {
- mLoadType = LOAD_NORMAL_REPLACE;
- }
-
- // Do this asynchronously
- nsCOMPtr<nsIRunnable> ev =
-- new InternalLoadEvent(this, aURI, aReferrer,
-+ new InternalLoadEvent(this, aURI, aOriginalURI, aReferrer,
- aReferrerPolicy, aOwner, aFlags,
- aTypeHint, aPostData, aHeadersData,
- aLoadType, aSHEntry, aFirstParty, aSrcdoc,
- aSourceDocShell, aBaseURI);
- return NS_DispatchToCurrentThread(ev);
- }
-
- // Just ignore this load attempt
-@@ -10371,17 +10442,17 @@ nsDocShell::InternalLoad(nsIURI* aURI,
- }
-
- net::PredictorLearn(aURI, nullptr,
- nsINetworkPredictor::LEARN_LOAD_TOPLEVEL, this);
- net::PredictorPredict(aURI, nullptr,
- nsINetworkPredictor::PREDICT_LOAD, this, nullptr);
-
- nsCOMPtr<nsIRequest> req;
-- rv = DoURILoad(aURI, aReferrer,
-+ rv = DoURILoad(aURI, aOriginalURI, aReferrer,
- !(aFlags & INTERNAL_LOAD_FLAGS_DONT_SEND_REFERRER),
- aReferrerPolicy,
- owner, aTypeHint, aFileName, aPostData, aHeadersData,
- aFirstParty, aDocShell, getter_AddRefs(req),
- (aFlags & INTERNAL_LOAD_FLAGS_FIRST_LOAD) != 0,
- (aFlags & INTERNAL_LOAD_FLAGS_BYPASS_CLASSIFIER) != 0,
- (aFlags & INTERNAL_LOAD_FLAGS_FORCE_ALLOW_COOKIES) != 0,
- srcdoc, aBaseURI, contentType);
-@@ -10445,16 +10516,17 @@ nsDocShell::GetInheritedPrincipal(bool a
- return docPrincipal;
- }
-
- return nullptr;
- }
-
- nsresult
- nsDocShell::DoURILoad(nsIURI* aURI,
-+ nsIURI* aOriginalURI,
- nsIURI* aReferrerURI,
- bool aSendReferrer,
- uint32_t aReferrerPolicy,
- nsISupports* aOwner,
- const char* aTypeHint,
- const nsAString& aFileName,
- nsIInputStream* aPostData,
- nsIInputStream* aHeadersData,
-@@ -10652,17 +10724,22 @@ nsDocShell::DoURILoad(nsIURI* aURI,
- }
-
- // Make sure to give the caller a channel if we managed to create one
- // This is important for correct error page/session history interaction
- if (aRequest) {
- NS_ADDREF(*aRequest = channel);
- }
-
-- channel->SetOriginalURI(aURI);
-+ if (aOriginalURI) {
-+ channel->SetOriginalURI(aOriginalURI);
-+ } else {
-+ channel->SetOriginalURI(aURI);
-+ }
-+
- if (aTypeHint && *aTypeHint) {
- channel->SetContentType(nsDependentCString(aTypeHint));
- mContentTypeHint = aTypeHint;
- } else {
- mContentTypeHint.Truncate();
- }
-
- if (!aFileName.IsVoid()) {
-@@ -11624,16 +11701,20 @@ nsDocShell::AddState(JS::Handle<JS::Valu
-
- // AddToSessionHistory may not modify mOSHE. In case it doesn't,
- // we'll just set mOSHE here.
- mOSHE = newSHEntry;
-
- } else {
- newSHEntry = mOSHE;
- newSHEntry->SetURI(newURI);
-+ nsCOMPtr<nsISHEntry_ESR38> entryESR38 = do_QueryInterface(newSHEntry);
-+ if (entryESR38) {
-+ entryESR38->SetOriginalURI(newURI);
-+ }
- }
-
- // Step 4: Modify new/original session history entry and clear its POST
- // data, if there is any.
- newSHEntry->SetStateData(scContainer);
- newSHEntry->SetPostData(nullptr);
-
- // If this push/replaceState changed the document's current URI and the new
-@@ -11816,16 +11897,17 @@ nsDocShell::AddToSessionHistory(nsIURI*
-
- if (!entry) {
- return NS_ERROR_OUT_OF_MEMORY;
- }
- }
-
- // Get the post data & referrer
- nsCOMPtr<nsIInputStream> inputStream;
-+ nsCOMPtr<nsIURI> originalURI;
- nsCOMPtr<nsIURI> referrerURI;
- uint32_t referrerPolicy = mozilla::net::RP_Default;
- nsCOMPtr<nsISupports> cacheKey;
- nsCOMPtr<nsISupports> owner = aOwner;
- bool expired = false;
- bool discardLayoutState = false;
- nsCOMPtr<nsICachingChannel> cacheChannel;
- if (aChannel) {
-@@ -11843,16 +11925,17 @@ nsDocShell::AddToSessionHistory(nsIURI*
- if (!httpChannel) {
- GetHttpChannel(aChannel, getter_AddRefs(httpChannel));
- }
- if (httpChannel) {
- nsCOMPtr<nsIUploadChannel> uploadChannel(do_QueryInterface(httpChannel));
- if (uploadChannel) {
- uploadChannel->GetUploadStream(getter_AddRefs(inputStream));
- }
-+ httpChannel->GetOriginalURI(getter_AddRefs(originalURI));
- httpChannel->GetReferrer(getter_AddRefs(referrerURI));
- httpChannel->GetReferrerPolicy(&referrerPolicy);
-
- discardLayoutState = ShouldDiscardLayoutState(httpChannel);
- }
- aChannel->GetOwner(getter_AddRefs(owner));
- if (!owner) {
- nsCOMPtr<nsILoadInfo> loadInfo;
-@@ -11875,16 +11958,21 @@ nsDocShell::AddToSessionHistory(nsIURI*
- EmptyString(), // Title
- inputStream, // Post data stream
- nullptr, // LayoutHistory state
- cacheKey, // CacheKey
- mContentTypeHint, // Content-type
- owner, // Channel or provided owner
- mHistoryID,
- mDynamicallyCreated);
-+
-+ nsCOMPtr<nsISHEntry_ESR38> entryESR38 = do_QueryInterface(entry);
-+ if (entryESR38) {
-+ entryESR38->SetOriginalURI(originalURI);
-+ }
- entry->SetReferrerURI(referrerURI);
- entry->SetReferrerPolicy(referrerPolicy);
- nsCOMPtr<nsIInputStreamChannel> inStrmChan = do_QueryInterface(aChannel);
- if (inStrmChan) {
- bool isSrcdocChannel;
- inStrmChan->GetIsSrcdocChannel(&isSrcdocChannel);
- if (isSrcdocChannel) {
- nsAutoString srcdoc;
-@@ -11976,25 +12064,32 @@ nsDocShell::AddToSessionHistory(nsIURI*
- nsresult
- nsDocShell::LoadHistoryEntry(nsISHEntry* aEntry, uint32_t aLoadType)
- {
- if (!IsNavigationAllowed()) {
- return NS_OK;
- }
-
- nsCOMPtr<nsIURI> uri;
-+ nsCOMPtr<nsIURI> originalURI;
- nsCOMPtr<nsIInputStream> postData;
- nsCOMPtr<nsIURI> referrerURI;
- uint32_t referrerPolicy;
- nsAutoCString contentType;
- nsCOMPtr<nsISupports> owner;
-
- NS_ENSURE_TRUE(aEntry, NS_ERROR_FAILURE);
-
- NS_ENSURE_SUCCESS(aEntry->GetURI(getter_AddRefs(uri)), NS_ERROR_FAILURE);
-+
-+ nsCOMPtr<nsISHEntry_ESR38> entryESR38 = do_QueryInterface(aEntry);
-+ if (entryESR38) {
-+ NS_ENSURE_SUCCESS(entryESR38->GetOriginalURI(getter_AddRefs(originalURI)),
-+ NS_ERROR_FAILURE);
-+ }
- NS_ENSURE_SUCCESS(aEntry->GetReferrerURI(getter_AddRefs(referrerURI)),
- NS_ERROR_FAILURE);
- NS_ENSURE_SUCCESS(aEntry->GetReferrerPolicy(&referrerPolicy),
- NS_ERROR_FAILURE);
- NS_ENSURE_SUCCESS(aEntry->GetPostData(getter_AddRefs(postData)),
- NS_ERROR_FAILURE);
- NS_ENSURE_SUCCESS(aEntry->GetContentType(contentType), NS_ERROR_FAILURE);
- NS_ENSURE_SUCCESS(aEntry->GetOwner(getter_AddRefs(owner)), NS_ERROR_FAILURE);
-@@ -12064,34 +12159,35 @@ nsDocShell::LoadHistoryEntry(nsISHEntry*
- } else {
- srcdoc = NullString();
- }
-
- // Passing nullptr as aSourceDocShell gives the same behaviour as before
- // aSourceDocShell was introduced. According to spec we should be passing
- // the source browsing context that was used when the history entry was
- // first created. bug 947716 has been created to address this issue.
-- rv = InternalLoad(uri,
-- referrerURI,
-- referrerPolicy,
-- owner,
-- flags,
-- nullptr, // No window target
-- contentType.get(), // Type hint
-- NullString(), // No forced file download
-- postData, // Post data stream
-- nullptr, // No headers stream
-- aLoadType, // Load type
-- aEntry, // SHEntry
-- true,
-- srcdoc,
-- nullptr, // Source docshell, see comment above
-- baseURI,
-- nullptr, // No nsIDocShell
-- nullptr); // No nsIRequest
-+ rv = InternalLoad2(uri,
-+ originalURI,
-+ referrerURI,
-+ referrerPolicy,
-+ owner,
-+ flags,
-+ nullptr, // No window target
-+ contentType.get(), // Type hint
-+ NullString(), // No forced file download
-+ postData, // Post data stream
-+ nullptr, // No headers stream
-+ aLoadType, // Load type
-+ aEntry, // SHEntry
-+ true,
-+ srcdoc,
-+ nullptr, // Source docshell, see comment above
-+ baseURI,
-+ nullptr, // No nsIDocShell
-+ nullptr); // No nsIRequest
- return rv;
- }
-
- NS_IMETHODIMP
- nsDocShell::GetShouldSaveLayoutState(bool* aShould)
- {
- *aShould = false;
- if (mOSHE) {
-@@ -13527,35 +13623,36 @@ nsDocShell::OnLinkClickSync(nsIContent*
- // with it under InternalLoad; we do _not_ want to change the URI
- // our caller passed in.
- nsCOMPtr<nsIURI> clonedURI;
- aURI->Clone(getter_AddRefs(clonedURI));
- if (!clonedURI) {
- return NS_ERROR_OUT_OF_MEMORY;
- }
-
-- nsresult rv = InternalLoad(clonedURI, // New URI
-- referer, // Referer URI
-- refererPolicy, // Referer policy
-- aContent->NodePrincipal(), // Owner is our node's
-- // principal
-- flags,
-- target.get(), // Window target
-- NS_LossyConvertUTF16toASCII(typeHint).get(),
-- aFileName, // Download as file
-- aPostDataStream, // Post data stream
-- aHeadersDataStream, // Headers stream
-- LOAD_LINK, // Load type
-- nullptr, // No SHEntry
-- true, // first party site
-- NullString(), // No srcdoc
-- this, // We are the source
-- nullptr, // baseURI not needed
-- aDocShell, // DocShell out-param
-- aRequest); // Request out-param
-+ nsresult rv = InternalLoad2(clonedURI, // New URI
-+ nullptr, // Original URI
-+ referer, // Referer URI
-+ refererPolicy, // Referer policy
-+ aContent->NodePrincipal(), // Owner is our node's
-+ // principal
-+ flags,
-+ target.get(), // Window target
-+ NS_LossyConvertUTF16toASCII(typeHint).get(),
-+ aFileName, // Download as file
-+ aPostDataStream, // Post data stream
-+ aHeadersDataStream, // Headers stream
-+ LOAD_LINK, // Load type
-+ nullptr, // No SHEntry
-+ true, // first party site
-+ NullString(), // No srcdoc
-+ this, // We are the source
-+ nullptr, // baseURI not needed
-+ aDocShell, // DocShell out-param
-+ aRequest); // Request out-param
- if (NS_SUCCEEDED(rv)) {
- DispatchPings(aContent, aURI, referer, refererPolicy);
- }
- return rv;
- }
-
- NS_IMETHODIMP
- nsDocShell::OnOverLink(nsIContent* aContent,
-diff --git a/docshell/base/nsDocShell.h b/docshell/base/nsDocShell.h
---- a/docshell/base/nsDocShell.h
-+++ b/docshell/base/nsDocShell.h
-@@ -132,17 +132,17 @@ enum eCharsetReloadState
- };
-
- //*****************************************************************************
- //*** nsDocShell
- //*****************************************************************************
-
- class nsDocShell final
- : public nsDocLoader
-- , public nsIDocShell_ESR38
-+ , public nsIDocShell_ESR38_2
- , public nsIWebNavigation
- , public nsIBaseWindow
- , public nsIScrollable
- , public nsITextScroll
- , public nsIDocCharset
- , public nsIContentViewerContainer
- , public nsIRefreshURI
- , public nsIWebProgressListener
-@@ -164,16 +164,17 @@ public:
- nsDocShell();
-
- NS_DECL_AND_IMPL_ZEROING_OPERATOR_NEW
-
- virtual nsresult Init() override;
-
- NS_DECL_ISUPPORTS_INHERITED
-
-+ NS_DECL_NSIDOCSHELL_ESR38_2
- NS_DECL_NSIDOCSHELL_ESR38
- NS_DECL_NSIDOCSHELL
- NS_DECL_NSIDOCSHELLTREEITEM
- NS_DECL_NSIWEBNAVIGATION
- NS_DECL_NSIBASEWINDOW
- NS_DECL_NSISCROLLABLE
- NS_DECL_NSITEXTSCROLL
- NS_DECL_NSIDOCCHARSET
-@@ -312,17 +313,20 @@ protected:
- // at the parent.
- nsIPrincipal* GetInheritedPrincipal(bool aConsiderCurrentDocument);
-
- // Actually open a channel and perform a URI load. Note: whatever owner is
- // passed to this function will be set on the channel. Callers who wish to
- // not have an owner on the channel should just pass null.
- // If aSrcdoc is not void, the load will be considered as a srcdoc load,
- // and the contents of aSrcdoc will be loaded instead of aURI.
-+ // aOriginalURI will be set as the originalURI on the channel that does the
-+ // load. If aOriginalURI is null, aURI will be set as the originalURI.
- nsresult DoURILoad(nsIURI* aURI,
-+ nsIURI* aOriginalURI,
- nsIURI* aReferrer,
- bool aSendReferrer,
- uint32_t aReferrerPolicy,
- nsISupports* aOwner,
- const char* aTypeHint,
- const nsAString& aFileName,
- nsIInputStream* aPostData,
- nsIInputStream* aHeadersData,
-diff --git a/docshell/base/nsDocShellLoadInfo.cpp b/docshell/base/nsDocShellLoadInfo.cpp
---- a/docshell/base/nsDocShellLoadInfo.cpp
-+++ b/docshell/base/nsDocShellLoadInfo.cpp
-@@ -34,16 +34,17 @@ nsDocShellLoadInfo::~nsDocShellLoadInfo(
- // nsDocShellLoadInfo::nsISupports
- //*****************************************************************************
-
- NS_IMPL_ADDREF(nsDocShellLoadInfo)
- NS_IMPL_RELEASE(nsDocShellLoadInfo)
-
- NS_INTERFACE_MAP_BEGIN(nsDocShellLoadInfo)
- NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDocShellLoadInfo)
-+ NS_INTERFACE_MAP_ENTRY(nsIDocShellLoadInfo_ESR38)
- NS_INTERFACE_MAP_ENTRY(nsIDocShellLoadInfo)
- NS_INTERFACE_MAP_END
-
- //*****************************************************************************
- // nsDocShellLoadInfo::nsIDocShellLoadInfo
- //*****************************************************************************
-
- NS_IMETHODIMP
-@@ -59,16 +60,33 @@ nsDocShellLoadInfo::GetReferrer(nsIURI**
- NS_IMETHODIMP
- nsDocShellLoadInfo::SetReferrer(nsIURI* aReferrer)
- {
- mReferrer = aReferrer;
- return NS_OK;
- }
-
- NS_IMETHODIMP
-+nsDocShellLoadInfo::GetOriginalURI(nsIURI** aOriginalURI)
-+{
-+ NS_ENSURE_ARG_POINTER(aOriginalURI);
-+
-+ *aOriginalURI = mOriginalURI;
-+ NS_IF_ADDREF(*aOriginalURI);
-+ return NS_OK;
-+}
-+
-+NS_IMETHODIMP
-+nsDocShellLoadInfo::SetOriginalURI(nsIURI* aOriginalURI)
-+{
-+ mOriginalURI = aOriginalURI;
-+ return NS_OK;
-+}
-+
-+NS_IMETHODIMP
- nsDocShellLoadInfo::GetOwner(nsISupports** aOwner)
- {
- NS_ENSURE_ARG_POINTER(aOwner);
-
- *aOwner = mOwner;
- NS_IF_ADDREF(*aOwner);
- return NS_OK;
- }
-diff --git a/docshell/base/nsDocShellLoadInfo.h b/docshell/base/nsDocShellLoadInfo.h
---- a/docshell/base/nsDocShellLoadInfo.h
-+++ b/docshell/base/nsDocShellLoadInfo.h
-@@ -14,29 +14,31 @@
- // Interfaces Needed
- #include "nsIDocShellLoadInfo.h"
-
- class nsIInputStream;
- class nsISHEntry;
- class nsIURI;
- class nsIDocShell;
-
--class nsDocShellLoadInfo : public nsIDocShellLoadInfo
-+class nsDocShellLoadInfo : public nsIDocShellLoadInfo_ESR38
- {
- public:
- nsDocShellLoadInfo();
-
- NS_DECL_ISUPPORTS
-+ NS_DECL_NSIDOCSHELLLOADINFO_ESR38
- NS_DECL_NSIDOCSHELLLOADINFO
-
- protected:
- virtual ~nsDocShellLoadInfo();
-
- protected:
- nsCOMPtr<nsIURI> mReferrer;
-+ nsCOMPtr<nsIURI> mOriginalURI;
- nsCOMPtr<nsISupports> mOwner;
- bool mInheritOwner;
- bool mOwnerIsExplicit;
- bool mSendReferrer;
- nsDocShellInfoReferrerPolicy mReferrerPolicy;
- nsDocShellInfoLoadType mLoadType;
- nsCOMPtr<nsISHEntry> mSHEntry;
- nsString mTarget;
-diff --git a/docshell/base/nsIDocShell.idl b/docshell/base/nsIDocShell.idl
---- a/docshell/base/nsIDocShell.idl
-+++ b/docshell/base/nsIDocShell.idl
-@@ -1059,8 +1059,66 @@ interface nsIDocShell : nsIDocShellTreeI
- interface nsIDocShell_ESR38 : nsIDocShell
- {
- /**
- * True if new child docshells should allow content retargeting.
- * Setting allowContentRetargeting also overwrites this value.
- */
- [infallible] attribute boolean allowContentRetargetingOnChildren;
- };
-+
-+[scriptable, builtinclass, uuid(607604b6-8fe0-4d2c-8a6c-44f5f31a6e02)]
-+interface nsIDocShell_ESR38_2 : nsIDocShell_ESR38
-+{
-+ /**
-+ * Loads the given URI. This method is identical to loadURI(...) except
-+ * that its parameter list is broken out instead of being packaged inside
-+ * of an nsIDocShellLoadInfo object...
-+ *
-+ * @param aURI - The URI to load.
-+ * @param aOriginalURI - The URI to set as the originalURI on the channel
-+ * that does the load. If null, aURI will be set as
-+ * the originalURI.
-+ * @param aReferrer - Referring URI
-+ * @param aReferrerPolicy - Referrer policy
-+ * @param aOwner - Owner (security principal)
-+ * @param aInheritOwner - Flag indicating whether the owner of the current
-+ * document should be inherited if aOwner is null.
-+ * @param aStopActiveDoc - Flag indicating whether loading the current
-+ * document should be stopped.
-+ * @param aWindowTarget - Window target for the load.
-+ * @param aTypeHint - A hint as to the content-type of the resulting
-+ * data. May be null or empty if no hint.
-+ * @param aFileName - Non-null when the link should be downloaded as
-+ the given filename.
-+ * @param aPostDataStream - Post data stream (if POSTing)
-+ * @param aHeadersStream - Stream containing "extra" request headers...
-+ * @param aLoadFlags - Flags to modify load behaviour. Flags are defined
-+ * in nsIWebNavigation.
-+ * @param aSHEntry - Active Session History entry (if loading from SH)
-+ * @param aSrcdoc When INTERNAL_LOAD_FLAGS_IS_SRCDOC is set, the
-+ * contents of this parameter will be loaded instead
-+ * of aURI.
-+ * @param aSourceDocShell - The source browsing context for the navigation.
-+ * @param aBaseURI - The base URI to be used for the load. Set in
-+ * srcdoc loads as it cannot otherwise be inferred
-+ * in certain situations such as view-source.
-+ */
-+ [noscript]void internalLoad2(in nsIURI aURI,
-+ in nsIURI aOriginalURI,
-+ in nsIURI aReferrer,
-+ in unsigned long aReferrerPolicy,
-+ in nsISupports aOwner,
-+ in uint32_t aFlags,
-+ in wstring aWindowTarget,
-+ in string aTypeHint,
-+ in AString aFileName,
-+ in nsIInputStream aPostDataStream,
-+ in nsIInputStream aHeadersStream,
-+ in unsigned long aLoadFlags,
-+ in nsISHEntry aSHEntry,
-+ in boolean firstParty,
-+ in AString aSrcdoc,
-+ in nsIDocShell aSourceDocShell,
-+ in nsIURI aBaseURI,
-+ out nsIDocShell aDocShell,
-+ out nsIRequest aRequest);
-+};
-diff --git a/docshell/base/nsIDocShellLoadInfo.idl b/docshell/base/nsIDocShellLoadInfo.idl
---- a/docshell/base/nsIDocShellLoadInfo.idl
-+++ b/docshell/base/nsIDocShellLoadInfo.idl
-@@ -106,8 +106,17 @@ interface nsIDocShellLoadInfo : nsISuppo
- attribute nsIDocShell sourceDocShell;
-
- /**
- * Used for srcdoc loads to give view-source knowledge of the load's base
- * URI as this information isn't embedded in the load's URI.
- */
- attribute nsIURI baseURI;
- };
-+
-+[scriptable, uuid(9d3bc466-5efe-414d-ae8b-3830b45877bb)]
-+interface nsIDocShellLoadInfo_ESR38 : nsIDocShellLoadInfo
-+{
-+ /**
-+ * The originalURI to be passed to nsIDocShell.internalLoad. May be null.
-+ */
-+ attribute nsIURI originalURI;
-+};
-diff --git a/docshell/shistory/public/nsISHEntry.idl b/docshell/shistory/public/nsISHEntry.idl
---- a/docshell/shistory/public/nsISHEntry.idl
-+++ b/docshell/shistory/public/nsISHEntry.idl
-@@ -319,8 +319,18 @@ interface nsISHEntryInternal : nsISuppor
- #define NS_SHENTRY_CID \
- {0xbfd1a791, 0xad9f, 0x11d3, {0xbd, 0xc7, 0x0, 0x50, 0x4, 0xa, 0x9b, 0x44}}
-
- #define NS_SHENTRY_CONTRACTID \
- "@mozilla.org/browser/session-history-entry;1"
-
- %}
-
-+[scriptable, uuid(e45ab6ef-3485-449c-b91c-0846b2bf6faf)]
-+interface nsISHEntry_ESR38 : nsISHEntry
-+{
-+ /**
-+ * A readonly property that returns the original URI of the current entry.
-+ * If an entry is the result of a redirect this attribute holds original
-+ * URI. The object returned is of type nsIURI
-+ */
-+ attribute nsIURI originalURI;
-+};
-diff --git a/docshell/shistory/src/nsSHEntry.cpp b/docshell/shistory/src/nsSHEntry.cpp
---- a/docshell/shistory/src/nsSHEntry.cpp
-+++ b/docshell/shistory/src/nsSHEntry.cpp
-@@ -38,16 +38,17 @@ nsSHEntry::nsSHEntry()
- , mIsSrcdocEntry(false)
- {
- mShared = new nsSHEntryShared();
- }
-
- nsSHEntry::nsSHEntry(const nsSHEntry &other)
- : mShared(other.mShared)
- , mURI(other.mURI)
-+ , mOriginalURI(other.mOriginalURI)
- , mReferrerURI(other.mReferrerURI)
- , mReferrerPolicy(other.mReferrerPolicy)
- , mTitle(other.mTitle)
- , mPostData(other.mPostData)
- , mLoadType(0) // XXX why not copy?
- , mID(other.mID)
- , mScrollPositionX(0) // XXX why not copy?
- , mScrollPositionY(0) // XXX why not copy?
-@@ -74,17 +75,17 @@ nsSHEntry::~nsSHEntry()
- // Null out the mParent pointers on all our kids.
- mChildren.EnumerateForwards(ClearParentPtr, nullptr);
- }
-
- //*****************************************************************************
- // nsSHEntry: nsISupports
- //*****************************************************************************
-
--NS_IMPL_ISUPPORTS(nsSHEntry, nsISHContainer, nsISHEntry, nsISHEntryInternal)
-+NS_IMPL_ISUPPORTS(nsSHEntry, nsISHContainer, nsISHEntry_ESR38, nsISHEntry, nsISHEntryInternal)
-
- //*****************************************************************************
- // nsSHEntry: nsISHEntry
- //*****************************************************************************
-
- NS_IMETHODIMP nsSHEntry::SetScrollPosition(int32_t x, int32_t y)
- {
- mScrollPositionX = x;
-@@ -119,16 +120,29 @@ NS_IMETHODIMP nsSHEntry::GetURI(nsIURI**
- }
-
- NS_IMETHODIMP nsSHEntry::SetURI(nsIURI* aURI)
- {
- mURI = aURI;
- return NS_OK;
- }
-
-+NS_IMETHODIMP nsSHEntry::GetOriginalURI(nsIURI** aOriginalURI)
-+{
-+ *aOriginalURI = mOriginalURI;
-+ NS_IF_ADDREF(*aOriginalURI);
-+ return NS_OK;
-+}
-+
-+NS_IMETHODIMP nsSHEntry::SetOriginalURI(nsIURI* aOriginalURI)
-+{
-+ mOriginalURI = aOriginalURI;
-+ return NS_OK;
-+}
-+
- NS_IMETHODIMP nsSHEntry::GetReferrerURI(nsIURI **aReferrerURI)
- {
- *aReferrerURI = mReferrerURI;
- NS_IF_ADDREF(*aReferrerURI);
- return NS_OK;
- }
-
- NS_IMETHODIMP nsSHEntry::SetReferrerURI(nsIURI *aReferrerURI)
-diff --git a/docshell/shistory/src/nsSHEntry.h b/docshell/shistory/src/nsSHEntry.h
---- a/docshell/shistory/src/nsSHEntry.h
-+++ b/docshell/shistory/src/nsSHEntry.h
-@@ -17,25 +17,26 @@
- // Interfaces needed
- #include "nsISHEntry.h"
- #include "nsISHContainer.h"
-
- class nsSHEntryShared;
- class nsIInputStream;
- class nsIURI;
-
--class nsSHEntry final : public nsISHEntry,
-+class nsSHEntry final : public nsISHEntry_ESR38,
- public nsISHContainer,
- public nsISHEntryInternal
- {
- public:
- nsSHEntry();
- nsSHEntry(const nsSHEntry &other);
-
- NS_DECL_ISUPPORTS
-+ NS_DECL_NSISHENTRY_ESR38
- NS_DECL_NSISHENTRY
- NS_DECL_NSISHENTRYINTERNAL
- NS_DECL_NSISHCONTAINER
-
- void DropPresentationState();
-
- static nsresult Startup();
- static void Shutdown();
-@@ -44,16 +45,17 @@ private:
- ~nsSHEntry();
-
- // We share the state in here with other SHEntries which correspond to the
- // same document.
- nsRefPtr<nsSHEntryShared> mShared;
-
- // See nsSHEntry.idl for comments on these members.
- nsCOMPtr<nsIURI> mURI;
-+ nsCOMPtr<nsIURI> mOriginalURI;
- nsCOMPtr<nsIURI> mReferrerURI;
- uint32_t mReferrerPolicy;
- nsString mTitle;
- nsCOMPtr<nsIInputStream> mPostData;
- uint32_t mLoadType;
- uint32_t mID;
- int32_t mScrollPositionX;
- int32_t mScrollPositionY;
-diff --git a/docshell/shistory/src/nsSHistory.cpp b/docshell/shistory/src/nsSHistory.cpp
---- a/docshell/shistory/src/nsSHistory.cpp
-+++ b/docshell/shistory/src/nsSHistory.cpp
-@@ -1779,16 +1779,26 @@ nsSHistory::InitiateLoad(nsISHEntry * aF
- * so that proper loadType is maintained through out a frameset
- */
- aFrameEntry->SetLoadType(aLoadType);
- aFrameDS->CreateLoadInfo (getter_AddRefs(loadInfo));
-
- loadInfo->SetLoadType(aLoadType);
- loadInfo->SetSHEntry(aFrameEntry);
-
-+ nsCOMPtr<nsIURI> originalURI;
-+ nsCOMPtr<nsISHEntry_ESR38> feESR38 = do_QueryInterface(aFrameEntry);
-+ if (feESR38) {
-+ feESR38->GetOriginalURI(getter_AddRefs(originalURI));
-+ }
-+ nsCOMPtr<nsIDocShellLoadInfo_ESR38> liESR38 = do_QueryInterface(loadInfo);
-+ if (liESR38) {
-+ liESR38->SetOriginalURI(originalURI);
-+ }
-+
- nsCOMPtr<nsIURI> nextURI;
- aFrameEntry->GetURI(getter_AddRefs(nextURI));
- // Time to initiate a document load
- return aFrameDS->LoadURI(nextURI, loadInfo, nsIWebNavigation::LOAD_FLAGS_NONE, false);
-
- }
-
-
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt01.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt01.patch
deleted file mode 100644
index 2b711b1761..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt01.patch
+++ /dev/null
@@ -1,356 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/c1d67bd4c993
-
-# HG changeset patch
-# User Timothy Nikkel <tnikkel@gmail.com>
-# Date 1454023801 21600
-# Node ID c1d67bd4c993b9e344c68954e6f0392c82b81e38
-# Parent 530559abe159d3c23f078d673d30ff03d9c244e2
-Bug 1224979 - Check if we compute usable filters for the downscaler, and if not put the downscaler in error state so it's not used. r=edwin, a=al
-
-diff --git a/image/Downscaler.cpp b/image/Downscaler.cpp
-new file mode 100644
---- /dev/null
-+++ b/image/Downscaler.cpp
-@@ -0,0 +1,340 @@
-+/* -*- 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 "Downscaler.h"
-+
-+#include <algorithm>
-+#include <ctime>
-+#include "gfxPrefs.h"
-+#include "image_operations.h"
-+#include "mozilla/SSE.h"
-+#include "convolver.h"
-+#include "skia/include/core/SkTypes.h"
-+
-+using std::max;
-+using std::swap;
-+
-+namespace mozilla {
-+namespace image {
-+
-+Downscaler::Downscaler(const nsIntSize& aTargetSize)
-+ : mTargetSize(aTargetSize)
-+ , mOutputBuffer(nullptr)
-+ , mXFilter(MakeUnique<skia::ConvolutionFilter1D>())
-+ , mYFilter(MakeUnique<skia::ConvolutionFilter1D>())
-+ , mWindowCapacity(0)
-+ , mHasAlpha(true)
-+ , mFlipVertically(false)
-+{
-+ MOZ_ASSERT(gfxPrefs::ImageDownscaleDuringDecodeEnabled(),
-+ "Downscaling even though downscale-during-decode is disabled?");
-+ MOZ_ASSERT(mTargetSize.width > 0 && mTargetSize.height > 0,
-+ "Invalid target size");
-+}
-+
-+Downscaler::~Downscaler()
-+{
-+ ReleaseWindow();
-+}
-+
-+void
-+Downscaler::ReleaseWindow()
-+{
-+ if (!mWindow) {
-+ return;
-+ }
-+
-+ for (int32_t i = 0; i < mWindowCapacity; ++i) {
-+ delete[] mWindow[i];
-+ }
-+
-+ mWindow = nullptr;
-+ mWindowCapacity = 0;
-+}
-+
-+nsresult
-+Downscaler::BeginFrame(const nsIntSize& aOriginalSize,
-+ const Maybe<nsIntRect>& aFrameRect,
-+ uint8_t* aOutputBuffer,
-+ bool aHasAlpha,
-+ bool aFlipVertically /* = false */)
-+{
-+ MOZ_ASSERT(aOutputBuffer);
-+ MOZ_ASSERT(mTargetSize != aOriginalSize,
-+ "Created a downscaler, but not downscaling?");
-+ MOZ_ASSERT(mTargetSize.width <= aOriginalSize.width,
-+ "Created a downscaler, but width is larger");
-+ MOZ_ASSERT(mTargetSize.height <= aOriginalSize.height,
-+ "Created a downscaler, but height is larger");
-+ MOZ_ASSERT(aOriginalSize.width > 0 && aOriginalSize.height > 0,
-+ "Invalid original size");
-+
-+ mFrameRect = aFrameRect.valueOr(nsIntRect(nsIntPoint(), aOriginalSize));
-+ MOZ_ASSERT(mFrameRect.x >= 0 && mFrameRect.y >= 0 &&
-+ mFrameRect.width >= 0 && mFrameRect.height >= 0,
-+ "Frame rect must have non-negative components");
-+ MOZ_ASSERT(nsIntRect(0, 0, aOriginalSize.width, aOriginalSize.height)
-+ .Contains(mFrameRect),
-+ "Frame rect must fit inside image");
-+ MOZ_ASSERT_IF(!nsIntRect(0, 0, aOriginalSize.width, aOriginalSize.height)
-+ .IsEqualEdges(mFrameRect),
-+ aHasAlpha);
-+
-+ mOriginalSize = aOriginalSize;
-+ mScale = gfxSize(double(mOriginalSize.width) / mTargetSize.width,
-+ double(mOriginalSize.height) / mTargetSize.height);
-+ mOutputBuffer = aOutputBuffer;
-+ mHasAlpha = aHasAlpha;
-+ mFlipVertically = aFlipVertically;
-+
-+ ReleaseWindow();
-+
-+ auto resizeMethod = skia::ImageOperations::RESIZE_LANCZOS3;
-+
-+ skia::resize::ComputeFilters(resizeMethod,
-+ mOriginalSize.width, mTargetSize.width,
-+ 0, mTargetSize.width,
-+ mXFilter.get());
-+
-+ if (mXFilter->max_filter() <= 0 || mXFilter->num_values() != mTargetSize.width) {
-+ NS_WARNING("Failed to compute filters for image downscaling");
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-+
-+ skia::resize::ComputeFilters(resizeMethod,
-+ mOriginalSize.height, mTargetSize.height,
-+ 0, mTargetSize.height,
-+ mYFilter.get());
-+
-+ if (mYFilter->max_filter() <= 0 || mYFilter->num_values() != mTargetSize.height) {
-+ NS_WARNING("Failed to compute filters for image downscaling");
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-+
-+ // Allocate the buffer, which contains scanlines of the original image.
-+ // pad by 15 to handle overreads by the simd code
-+ size_t bufferLen = mOriginalSize.width * sizeof(uint32_t) + 15;
-+ mRowBuffer.reset(new (fallible) uint8_t[bufferLen]);
-+ if (MOZ_UNLIKELY(!mRowBuffer)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-+
-+ // Zero buffer to keep valgrind happy.
-+ memset(mRowBuffer.get(), 0, bufferLen);
-+
-+ // Allocate the window, which contains horizontally downscaled scanlines. (We
-+ // can store scanlines which are already downscale because our downscaling
-+ // filter is separable.)
-+ mWindowCapacity = mYFilter->max_filter();
-+ mWindow.reset(new (fallible) uint8_t*[mWindowCapacity]);
-+ if (MOZ_UNLIKELY(!mWindow)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-+
-+ bool anyAllocationFailed = false;
-+ // pad by 15 to handle overreads by the simd code
-+ const int rowSize = mTargetSize.width * sizeof(uint32_t) + 15;
-+ for (int32_t i = 0; i < mWindowCapacity; ++i) {
-+ mWindow[i] = new (fallible) uint8_t[rowSize];
-+ anyAllocationFailed = anyAllocationFailed || mWindow[i] == nullptr;
-+ }
-+
-+ if (MOZ_UNLIKELY(anyAllocationFailed)) {
-+ // We intentionally iterate through the entire array even if an allocation
-+ // fails, to ensure that all the pointers in it are either valid or nullptr.
-+ // That in turn ensures that ReleaseWindow() can clean up correctly.
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-+
-+ ResetForNextProgressivePass();
-+
-+ return NS_OK;
-+}
-+
-+void
-+Downscaler::SkipToRow(int32_t aRow)
-+{
-+ if (mCurrentInLine < aRow) {
-+ ClearRow();
-+ do {
-+ CommitRow();
-+ } while (mCurrentInLine < aRow);
-+ }
-+}
-+
-+void
-+Downscaler::ResetForNextProgressivePass()
-+{
-+ mPrevInvalidatedLine = 0;
-+ mCurrentOutLine = 0;
-+ mCurrentInLine = 0;
-+ mLinesInBuffer = 0;
-+
-+ if (mFrameRect.IsEmpty()) {
-+ // Our frame rect is zero size; commit rows until the end of the image.
-+ SkipToRow(mOriginalSize.height - 1);
-+ } else {
-+ // If we have a vertical offset, commit rows to shift us past it.
-+ SkipToRow(mFrameRect.y);
-+ }
-+}
-+
-+static void
-+GetFilterOffsetAndLength(UniquePtr<skia::ConvolutionFilter1D>& aFilter,
-+ int32_t aOutputImagePosition,
-+ int32_t* aFilterOffsetOut,
-+ int32_t* aFilterLengthOut)
-+{
-+ MOZ_ASSERT(aOutputImagePosition < aFilter->num_values());
-+ aFilter->FilterForValue(aOutputImagePosition,
-+ aFilterOffsetOut,
-+ aFilterLengthOut);
-+}
-+
-+void
-+Downscaler::ClearRow(uint32_t aStartingAtCol)
-+{
-+ MOZ_ASSERT(int64_t(mOriginalSize.width) > int64_t(aStartingAtCol));
-+ uint32_t bytesToClear = (mOriginalSize.width - aStartingAtCol)
-+ * sizeof(uint32_t);
-+ memset(mRowBuffer.get() + (aStartingAtCol * sizeof(uint32_t)),
-+ 0, bytesToClear);
-+}
-+
-+void
-+Downscaler::CommitRow()
-+{
-+ MOZ_ASSERT(mOutputBuffer, "Should have a current frame");
-+ MOZ_ASSERT(mCurrentInLine < mOriginalSize.height, "Past end of input");
-+
-+ if (mCurrentOutLine < mTargetSize.height) {
-+ int32_t filterOffset = 0;
-+ int32_t filterLength = 0;
-+ GetFilterOffsetAndLength(mYFilter, mCurrentOutLine,
-+ &filterOffset, &filterLength);
-+
-+ int32_t inLineToRead = filterOffset + mLinesInBuffer;
-+ MOZ_ASSERT(mCurrentInLine <= inLineToRead, "Reading past end of input");
-+ if (mCurrentInLine == inLineToRead) {
-+ skia::ConvolveHorizontally(mRowBuffer.get(), *mXFilter,
-+ mWindow[mLinesInBuffer++], mHasAlpha,
-+ supports_sse2());
-+ }
-+
-+ MOZ_ASSERT(mCurrentOutLine < mTargetSize.height,
-+ "Writing past end of output");
-+
-+ while (mLinesInBuffer == filterLength) {
-+ DownscaleInputLine();
-+
-+ if (mCurrentOutLine == mTargetSize.height) {
-+ break; // We're done.
-+ }
-+
-+ GetFilterOffsetAndLength(mYFilter, mCurrentOutLine,
-+ &filterOffset, &filterLength);
-+ }
-+ }
-+
-+ mCurrentInLine += 1;
-+
-+ // If we're at the end of the part of the original image that has data, commit
-+ // rows to shift us to the end.
-+ if (mCurrentInLine == (mFrameRect.y + mFrameRect.height)) {
-+ SkipToRow(mOriginalSize.height - 1);
-+ }
-+}
-+
-+bool
-+Downscaler::HasInvalidation() const
-+{
-+ return mCurrentOutLine > mPrevInvalidatedLine;
-+}
-+
-+DownscalerInvalidRect
-+Downscaler::TakeInvalidRect()
-+{
-+ if (MOZ_UNLIKELY(!HasInvalidation())) {
-+ return DownscalerInvalidRect();
-+ }
-+
-+ DownscalerInvalidRect invalidRect;
-+
-+ // Compute the target size invalid rect.
-+ if (mFlipVertically) {
-+ // We need to flip it. This will implicitly flip the original size invalid
-+ // rect, since we compute it by scaling this rect.
-+ invalidRect.mTargetSizeRect =
-+ IntRect(0, mTargetSize.height - mCurrentOutLine,
-+ mTargetSize.width, mCurrentOutLine - mPrevInvalidatedLine);
-+ } else {
-+ invalidRect.mTargetSizeRect =
-+ IntRect(0, mPrevInvalidatedLine,
-+ mTargetSize.width, mCurrentOutLine - mPrevInvalidatedLine);
-+ }
-+
-+ mPrevInvalidatedLine = mCurrentOutLine;
-+
-+ // Compute the original size invalid rect.
-+ invalidRect.mOriginalSizeRect = invalidRect.mTargetSizeRect;
-+ invalidRect.mOriginalSizeRect.ScaleRoundOut(mScale.width, mScale.height);
-+
-+ return invalidRect;
-+}
-+
-+void
-+Downscaler::DownscaleInputLine()
-+{
-+ typedef skia::ConvolutionFilter1D::Fixed FilterValue;
-+
-+ MOZ_ASSERT(mOutputBuffer);
-+ MOZ_ASSERT(mCurrentOutLine < mTargetSize.height,
-+ "Writing past end of output");
-+
-+ int32_t filterOffset = 0;
-+ int32_t filterLength = 0;
-+ MOZ_ASSERT(mCurrentOutLine < mYFilter->num_values());
-+ auto filterValues =
-+ mYFilter->FilterForValue(mCurrentOutLine, &filterOffset, &filterLength);
-+
-+ int32_t currentOutLine = mFlipVertically
-+ ? mTargetSize.height - (mCurrentOutLine + 1)
-+ : mCurrentOutLine;
-+ MOZ_ASSERT(currentOutLine >= 0);
-+
-+ uint8_t* outputLine =
-+ &mOutputBuffer[currentOutLine * mTargetSize.width * sizeof(uint32_t)];
-+ skia::ConvolveVertically(static_cast<const FilterValue*>(filterValues),
-+ filterLength, mWindow.get(), mXFilter->num_values(),
-+ outputLine, mHasAlpha, supports_sse2());
-+
-+ mCurrentOutLine += 1;
-+
-+ if (mCurrentOutLine == mTargetSize.height) {
-+ // We're done.
-+ return;
-+ }
-+
-+ int32_t newFilterOffset = 0;
-+ int32_t newFilterLength = 0;
-+ GetFilterOffsetAndLength(mYFilter, mCurrentOutLine,
-+ &newFilterOffset, &newFilterLength);
-+
-+ int diff = newFilterOffset - filterOffset;
-+ MOZ_ASSERT(diff >= 0, "Moving backwards in the filter?");
-+
-+ // Shift the buffer. We're just moving pointers here, so this is cheap.
-+ mLinesInBuffer -= diff;
-+ mLinesInBuffer = max(mLinesInBuffer, 0);
-+ for (int32_t i = 0; i < mLinesInBuffer; ++i) {
-+ swap(mWindow[i], mWindow[filterLength - mLinesInBuffer + i]);
-+ }
-+}
-+
-+
-+
-+} // namespace image
-+} // namespace mozilla
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt02.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt02.patch
deleted file mode 100644
index e01b5eaf2f..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt02.patch
+++ /dev/null
@@ -1,58 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/9719b71d72dd
-
-# HG changeset patch
-# User Byron Campen [:bwc] <docfaraday@gmail.com>
-# Date 1454100887 21600
-# Node ID 9719b71d72dd2a3c5ee12ace156af2a63d9595ac
-# Parent b68673d974a10f65390f80b36d4307eb31e44669
-Bug 1234578 - Assert if PCM is destroyed improperly. r=rjesup, a=sylvestre
-
-diff --git a/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.cpp b/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.cpp
---- a/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.cpp
-+++ b/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.cpp
-@@ -712,16 +712,18 @@ PeerConnectionMedia::SelfDestruct_m()
- {
- CSFLogDebug(logTag, "%s: ", __FUNCTION__);
-
- ASSERT_ON_THREAD(mMainThread);
-
- mLocalSourceStreams.Clear();
- mRemoteSourceStreams.Clear();
-
-+ mMainThread = nullptr;
-+
- // Final self-destruct.
- this->Release();
- }
-
- void
- PeerConnectionMedia::ShutdownMediaTransport_s()
- {
- ASSERT_ON_THREAD(mSTSThread);
-diff --git a/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.h b/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.h
---- a/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.h
-+++ b/media/webrtc/signaling/src/peerconnection/PeerConnectionMedia.h
-@@ -210,17 +210,20 @@ class RemoteSourceStreamInfo : public So
- std::vector<std::string> mTrackIdMap;
-
- // True iff SetPullEnabled(true) has been called on the DOMMediaStream. This
- // happens when offer/answer concludes.
- bool mReceiving;
- };
-
- class PeerConnectionMedia : public sigslot::has_slots<> {
-- ~PeerConnectionMedia() {}
-+ ~PeerConnectionMedia()
-+ {
-+ MOZ_RELEASE_ASSERT(!mMainThread);
-+ }
-
- public:
- explicit PeerConnectionMedia(PeerConnectionImpl *parent);
-
- PeerConnectionImpl* GetPC() { return mParent; }
- nsresult Init(const std::vector<NrIceStunServer>& stun_servers,
- const std::vector<NrIceTurnServer>& turn_servers);
- // WARNING: This destroys the object!
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt03.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt03.patch
deleted file mode 100644
index 96b83c118c..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt03.patch
+++ /dev/null
@@ -1,60 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/2839062f84fb
-
-# HG changeset patch
-# User Jan de Mooij <jdemooij@mozilla.com>
-# Date 1455119320 -3600
-# Node ID 2839062f84fb6cba2781ea8d59150f13d4813ddc
-# Parent 185b233ea03f3811404e3979b65ec86b29d13555
-Bug 1242279 - r=bhackett1024 a=sylvestre
-
-diff --git a/js/src/vm/TypeInference.cpp b/js/src/vm/TypeInference.cpp
---- a/js/src/vm/TypeInference.cpp
-+++ b/js/src/vm/TypeInference.cpp
-@@ -3961,16 +3961,22 @@ JSScript::maybeSweepTypes(AutoClearTypeI
-
- unsigned num = TypeScript::NumTypeSets(this);
- StackTypeSet* typeArray = types_->typeArray();
-
- // Remove constraints and references to dead objects from stack type sets.
- for (unsigned i = 0; i < num; i++)
- typeArray[i].sweep(zone(), *oom);
-
-+ if (oom->hadOOM()) {
-+ // It's possible we OOM'd while copying freeze constraints, so they
-+ // need to be regenerated.
-+ hasFreezeConstraints_ = false;
-+ }
-+
- // Update the recompile indexes in any IonScripts still on the script.
- if (hasIonScript())
- ionScript()->recompileInfoRef().shouldSweep(types);
- }
-
- void
- TypeScript::destroy()
- {
-diff --git a/js/src/vm/TypeInference.h b/js/src/vm/TypeInference.h
---- a/js/src/vm/TypeInference.h
-+++ b/js/src/vm/TypeInference.h
-@@ -566,16 +566,19 @@ class AutoClearTypeInferenceStateOnOOM
- : zone(zone), oom(false)
- {}
-
- ~AutoClearTypeInferenceStateOnOOM();
-
- void setOOM() {
- oom = true;
- }
-+ bool hadOOM() const {
-+ return oom;
-+ }
- };
-
- /* Superclass common to stack and heap type sets. */
- class ConstraintTypeSet : public TypeSet
- {
- public:
- /* Chain of constraints which propagate changes out from this type set. */
- TypeConstraint* constraintList;
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt04.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt04.patch
deleted file mode 100644
index 4eeb2377b0..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt04.patch
+++ /dev/null
@@ -1,53 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/9dd60e798819
-
-# HG changeset patch
-# User Olli Pettay <bugs@pettay.fi>
-# Date 1455204078 -3600
-# Node ID 9dd60e798819fe2ebf1e5bd36aa9006ecd2f82c9
-# Parent c1d67bd4c993b9e344c68954e6f0392c82b81e38
-Bug 1244250 - r=mats, a=al
-
-diff --git a/layout/style/nsAnimationManager.cpp b/layout/style/nsAnimationManager.cpp
---- a/layout/style/nsAnimationManager.cpp
-+++ b/layout/style/nsAnimationManager.cpp
-@@ -715,16 +715,17 @@ nsAnimationManager::FlushAnimations(Flus
- }
-
- DispatchEvents(); // may destroy us
- }
-
- void
- nsAnimationManager::DoDispatchEvents()
- {
-+ nsRefPtr<nsAnimationManager> kungFuDeathGrip(this);
- EventArray events;
- mPendingEvents.SwapElements(events);
- for (uint32_t i = 0, i_end = events.Length(); i < i_end; ++i) {
- AnimationEventInfo &info = events[i];
- EventDispatcher::Dispatch(info.mElement, mPresContext, &info.mEvent);
-
- if (!mPresContext) {
- break;
-diff --git a/layout/style/nsTransitionManager.cpp b/layout/style/nsTransitionManager.cpp
---- a/layout/style/nsTransitionManager.cpp
-+++ b/layout/style/nsTransitionManager.cpp
-@@ -753,16 +753,17 @@ nsTransitionManager::FlushTransitions(Fl
- }
- }
- }
-
- if (didThrottle) {
- mPresContext->Document()->SetNeedStyleFlush();
- }
-
-+ nsRefPtr<nsTransitionManager> kungFuDeathGrip(this);
- for (uint32_t i = 0, i_end = events.Length(); i < i_end; ++i) {
- TransitionEventInfo &info = events[i];
- EventDispatcher::Dispatch(info.mElement, mPresContext, &info.mEvent);
-
- if (!mPresContext) {
- break;
- }
- }
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt05.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt05.patch
deleted file mode 100644
index d222feff2a..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt05.patch
+++ /dev/null
@@ -1,32 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/1dd0ca8e70bd
-
-# HG changeset patch
-# User Nicolas B. Pierron <nicolas.b.pierron@mozilla.com>
-# Date 1456161361 0
-# Node ID 1dd0ca8e70bd77b6fd93f36cc4e9c2cebfe8ba0a
-# Parent 95ff874886905ef46a7bbc760981d15ad0831096
-Bug 1221872 - ValueNumbering: Set the dominator index of fixup blocks when they are created. r=sunfish, a=ritu
-
-diff --git a/js/src/jit/ValueNumbering.cpp b/js/src/jit/ValueNumbering.cpp
---- a/js/src/jit/ValueNumbering.cpp
-+++ b/js/src/jit/ValueNumbering.cpp
-@@ -433,16 +433,17 @@ ValueNumberer::fixupOSROnlyLoop(MBasicBl
- MBasicBlock* fake = MBasicBlock::NewAsmJS(graph_, block->info(),
- nullptr, MBasicBlock::NORMAL);
- if (fake == nullptr)
- return false;
-
- graph_.insertBlockBefore(block, fake);
- fake->setImmediateDominator(fake);
- fake->addNumDominated(1);
-+ fake->setDomIndex(fake->id());
-
- // Create zero-input phis to use as inputs for any phis in |block|.
- // Again, this is a little odd, but it's the least-odd thing we can do
- // without significant complexity.
- for (MPhiIterator iter(block->phisBegin()), end(block->phisEnd()); iter != end; ++iter) {
- MPhi* phi = *iter;
- MPhi* fakePhi = MPhi::New(graph_.alloc(), phi->type());
- fake->addPhi(fakePhi);
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1952-pt06.patch b/gnu/packages/patches/icecat-CVE-2016-1952-pt06.patch
deleted file mode 100644
index 3de568493b..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1952-pt06.patch
+++ /dev/null
@@ -1,103 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/6f4d51302387
-
-# HG changeset patch
-# User Andrew McCreight <continuation@gmail.com>
-# Date 1456273423 28800
-# Node ID 6f4d5130238790fa5810c76ffeb9eccc65efa8c9
-# Parent 70f6c59d9d73a5edefd216b48ca74a931da12cf1
-Bug 1249685 - Use more nsCOMPtrs for stack variables in DOM code. r=smaug, a=ritu
-
-diff --git a/dom/base/nsRange.cpp b/dom/base/nsRange.cpp
---- a/dom/base/nsRange.cpp
-+++ b/dom/base/nsRange.cpp
-@@ -1985,17 +1985,17 @@ nsRange::CutContents(DocumentFragment**
- rv = closestAncestor ? PrependChild(closestAncestor, nodeToResult)
- : PrependChild(commonCloneAncestor, nodeToResult);
- NS_ENSURE_SUCCESS(rv, rv);
- NS_ENSURE_STATE(!guard.Mutated(parent ? 2 : 1) ||
- ValidateCurrentNode(this, iter));
- } else if (nodeToResult) {
- nsMutationGuard guard;
- nsCOMPtr<nsINode> node = nodeToResult;
-- nsINode* parent = node->GetParentNode();
-+ nsCOMPtr<nsINode> parent = node->GetParentNode();
- if (parent) {
- mozilla::ErrorResult error;
- parent->RemoveChild(*node, error);
- NS_ENSURE_FALSE(error.Failed(), error.ErrorCode());
- }
- NS_ENSURE_STATE(!guard.Mutated(1) ||
- ValidateCurrentNode(this, iter));
- }
-diff --git a/dom/base/nsTreeSanitizer.cpp b/dom/base/nsTreeSanitizer.cpp
---- a/dom/base/nsTreeSanitizer.cpp
-+++ b/dom/base/nsTreeSanitizer.cpp
-@@ -1423,18 +1423,18 @@ nsTreeSanitizer::SanitizeChildren(nsINod
- mAllowStyles,
- false);
- }
- node = node->GetNextNonChildNode(aRoot);
- continue;
- }
- if (MustFlatten(ns, localName)) {
- RemoveAllAttributes(node);
-- nsIContent* next = node->GetNextNode(aRoot);
-- nsIContent* parent = node->GetParent();
-+ nsCOMPtr<nsIContent> next = node->GetNextNode(aRoot);
-+ nsCOMPtr<nsIContent> parent = node->GetParent();
- nsCOMPtr<nsIContent> child; // Must keep the child alive during move
- ErrorResult rv;
- while ((child = node->GetFirstChild())) {
- parent->InsertBefore(*child, node, rv);
- if (rv.Failed()) {
- break;
- }
- }
-diff --git a/dom/html/HTMLSelectElement.cpp b/dom/html/HTMLSelectElement.cpp
---- a/dom/html/HTMLSelectElement.cpp
-+++ b/dom/html/HTMLSelectElement.cpp
-@@ -624,17 +624,17 @@ HTMLSelectElement::Add(nsGenericHTMLElem
- {
- if (!aBefore) {
- Element::AppendChild(aElement, aError);
- return;
- }
-
- // Just in case we're not the parent, get the parent of the reference
- // element
-- nsINode* parent = aBefore->Element::GetParentNode();
-+ nsCOMPtr<nsINode> parent = aBefore->Element::GetParentNode();
- if (!parent || !nsContentUtils::ContentIsDescendantOf(parent, this)) {
- // NOT_FOUND_ERR: Raised if before is not a descendant of the SELECT
- // element.
- aError.Throw(NS_ERROR_DOM_NOT_FOUND_ERR);
- return;
- }
-
- // If the before parameter is not null, we are equivalent to the
-diff --git a/dom/html/HTMLTableElement.cpp b/dom/html/HTMLTableElement.cpp
---- a/dom/html/HTMLTableElement.cpp
-+++ b/dom/html/HTMLTableElement.cpp
-@@ -516,18 +516,18 @@ HTMLTableElement::InsertRow(int32_t aInd
- if (rowCount > 0) {
- if (refIndex == rowCount || aIndex == -1) {
- // we set refIndex to the last row so we can get the last row's
- // parent we then do an AppendChild below if (rowCount<aIndex)
-
- refIndex = rowCount - 1;
- }
-
-- Element* refRow = rows->Item(refIndex);
-- nsINode* parent = refRow->GetParentNode();
-+ RefPtr<Element> refRow = rows->Item(refIndex);
-+ nsCOMPtr<nsINode> parent = refRow->GetParentNode();
-
- // create the row
- nsRefPtr<mozilla::dom::NodeInfo> nodeInfo;
- nsContentUtils::NameChanged(mNodeInfo, nsGkAtoms::tr,
- getter_AddRefs(nodeInfo));
-
- newRow = NS_NewHTMLTableRowElement(nodeInfo.forget());
-
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1954.patch b/gnu/packages/patches/icecat-CVE-2016-1954.patch
deleted file mode 100644
index bbb4b3217c..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1954.patch
+++ /dev/null
@@ -1,32 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/a5c4c18849b4
-
-# HG changeset patch
-# User Christoph Kerschbaumer <mozilla@christophkerschbaumer.com>
-# Date 1456157874 28800
-# Node ID a5c4c18849b486ef8693e20421b69239a2cbe574
-# Parent e93aeb25e2a44df8d22f5a065b4410620e2c8730
-Bug 1243178: CSP - Skip sending reports for non http schemes (r=dveditz) a=ritu
-
-diff --git a/dom/security/nsCSPContext.cpp b/dom/security/nsCSPContext.cpp
---- a/dom/security/nsCSPContext.cpp
-+++ b/dom/security/nsCSPContext.cpp
-@@ -798,16 +798,17 @@ nsCSPContext::SendReports(nsISupports* a
- (NS_SUCCEEDED(reportURI->SchemeIs("https", &isHttpScheme)) && isHttpScheme);
-
- if (!isHttpScheme) {
- const char16_t* params[] = { reportURIs[r].get() };
- CSP_LogLocalizedStr(NS_LITERAL_STRING("reportURInotHttpsOrHttp2").get(),
- params, ArrayLength(params),
- aSourceFile, aScriptSample, aLineNum, 0,
- nsIScriptError::errorFlag, "CSP", mInnerWindowID);
-+ continue;
- }
-
- // make sure this is an anonymous request (no cookies) so in case the
- // policy URI is injected, it can't be abused for CSRF.
- nsLoadFlags flags;
- rv = reportChannel->GetLoadFlags(&flags);
- NS_ENSURE_SUCCESS(rv, rv);
- flags |= nsIRequest::LOAD_ANONYMOUS;
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1960.patch b/gnu/packages/patches/icecat-CVE-2016-1960.patch
deleted file mode 100644
index 6c5c885e8b..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1960.patch
+++ /dev/null
@@ -1,55 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/185b233ea03f
-
-# HG changeset patch
-# User Henri Sivonen <hsivonen@hsivonen.fi>
-# Date 1455100746 -7200
-# Node ID 185b233ea03f3811404e3979b65ec86b29d13555
-# Parent 271e3a5a53d96871141e89271f611033b512e3e4
-Bug 1246014. r=wchen. a=sylvestre
-
-diff --git a/parser/html/javasrc/TreeBuilder.java b/parser/html/javasrc/TreeBuilder.java
---- a/parser/html/javasrc/TreeBuilder.java
-+++ b/parser/html/javasrc/TreeBuilder.java
-@@ -4437,17 +4437,17 @@ public abstract class TreeBuilder<T> imp
- return TreeBuilder.NOT_FOUND_ON_STACK;
- }
-
- private void clearStackBackTo(int eltPos) throws SAXException {
- int eltGroup = stack[eltPos].getGroup();
- while (currentPtr > eltPos) { // > not >= intentional
- if (stack[currentPtr].ns == "http://www.w3.org/1999/xhtml"
- && stack[currentPtr].getGroup() == TEMPLATE
-- && (eltGroup == TABLE || eltGroup == TBODY_OR_THEAD_OR_TFOOT|| eltGroup == TR || eltGroup == HTML)) {
-+ && (eltGroup == TABLE || eltGroup == TBODY_OR_THEAD_OR_TFOOT|| eltGroup == TR || eltPos == 0)) {
- return;
- }
- pop();
- }
- }
-
- private void resetTheInsertionMode() {
- StackNode<T> node;
-diff --git a/parser/html/nsHtml5TreeBuilder.cpp b/parser/html/nsHtml5TreeBuilder.cpp
---- a/parser/html/nsHtml5TreeBuilder.cpp
-+++ b/parser/html/nsHtml5TreeBuilder.cpp
-@@ -3301,17 +3301,17 @@ nsHtml5TreeBuilder::findLastInTableScope
- return NS_HTML5TREE_BUILDER_NOT_FOUND_ON_STACK;
- }
-
- void
- nsHtml5TreeBuilder::clearStackBackTo(int32_t eltPos)
- {
- int32_t eltGroup = stack[eltPos]->getGroup();
- while (currentPtr > eltPos) {
-- if (stack[currentPtr]->ns == kNameSpaceID_XHTML && stack[currentPtr]->getGroup() == NS_HTML5TREE_BUILDER_TEMPLATE && (eltGroup == NS_HTML5TREE_BUILDER_TABLE || eltGroup == NS_HTML5TREE_BUILDER_TBODY_OR_THEAD_OR_TFOOT || eltGroup == NS_HTML5TREE_BUILDER_TR || eltGroup == NS_HTML5TREE_BUILDER_HTML)) {
-+ if (stack[currentPtr]->ns == kNameSpaceID_XHTML && stack[currentPtr]->getGroup() == NS_HTML5TREE_BUILDER_TEMPLATE && (eltGroup == NS_HTML5TREE_BUILDER_TABLE || eltGroup == NS_HTML5TREE_BUILDER_TBODY_OR_THEAD_OR_TFOOT || eltGroup == NS_HTML5TREE_BUILDER_TR || !eltPos)) {
- return;
- }
- pop();
- }
- }
-
- void
- nsHtml5TreeBuilder::resetTheInsertionMode()
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1961.patch b/gnu/packages/patches/icecat-CVE-2016-1961.patch
deleted file mode 100644
index 10162be24b..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1961.patch
+++ /dev/null
@@ -1,33 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/e93aeb25e2a4
-
-# HG changeset patch
-# User Andrew McCreight <continuation@gmail.com>
-# Date 1455891967 28800
-# Node ID e93aeb25e2a44df8d22f5a065b4410620e2c8730
-# Parent 221de852fda32714a9e484774ceafafb450ea73c
-Bug 1249377 - Hold a strong reference to |root| in nsHTMLDocument::SetBody. r=bz, a=sylvestre
-
-diff --git a/dom/html/nsHTMLDocument.cpp b/dom/html/nsHTMLDocument.cpp
---- a/dom/html/nsHTMLDocument.cpp
-+++ b/dom/html/nsHTMLDocument.cpp
-@@ -1044,17 +1044,17 @@ nsHTMLDocument::SetBody(nsIDOMHTMLElemen
- ErrorResult rv;
- SetBody(static_cast<nsGenericHTMLElement*>(newBody.get()), rv);
- return rv.ErrorCode();
- }
-
- void
- nsHTMLDocument::SetBody(nsGenericHTMLElement* newBody, ErrorResult& rv)
- {
-- Element* root = GetRootElement();
-+ nsCOMPtr<Element> root = GetRootElement();
-
- // The body element must be either a body tag or a frameset tag. And we must
- // have a html root tag, otherwise GetBody will not return the newly set
- // body.
- if (!newBody || !(newBody->Tag() == nsGkAtoms::body ||
- newBody->Tag() == nsGkAtoms::frameset) ||
- !root || !root->IsHTML() ||
- root->Tag() != nsGkAtoms::html) {
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1962.patch b/gnu/packages/patches/icecat-CVE-2016-1962.patch
deleted file mode 100644
index 7eb4e072a1..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1962.patch
+++ /dev/null
@@ -1,107 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/221de852fda3
-
-# HG changeset patch
-# User Randell Jesup <rjesup@jesup.org>
-# Date 1455862087 18000
-# Node ID 221de852fda32714a9e484774ceafafb450ea73c
-# Parent b03db72e32f6e3acdc9f8705371cb222d7e6c456
-Bug 1240760: Update DataChannel::Close() r=mcmanus, a=ritu
-
-MozReview-Commit-ID: 7nN9h3M3O8w
-
-diff --git a/netwerk/sctp/datachannel/DataChannel.cpp b/netwerk/sctp/datachannel/DataChannel.cpp
---- a/netwerk/sctp/datachannel/DataChannel.cpp
-+++ b/netwerk/sctp/datachannel/DataChannel.cpp
-@@ -1771,17 +1771,17 @@ DataChannelConnection::HandleStreamReset
- }
- NS_DispatchToMainThread(new DataChannelOnMessageAvailable(
- DataChannelOnMessageAvailable::ON_CHANNEL_CLOSED, this,
- channel));
- mStreams[channel->mStream] = nullptr;
-
- LOG(("Disconnected DataChannel %p from connection %p",
- (void *) channel.get(), (void *) channel->mConnection.get()));
-- channel->Destroy();
-+ channel->DestroyLocked();
- // At this point when we leave here, the object is a zombie held alive only by the DOM object
- } else {
- LOG(("Can't find incoming channel %d",i));
- }
- }
- }
- }
-
-@@ -2498,17 +2498,17 @@ DataChannelConnection::CloseInt(DataChan
- mStreams[channel->mStream] = nullptr;
- } else {
- SendOutgoingStreamReset();
- }
- }
- aChannel->mState = CLOSING;
- if (mState == CLOSED) {
- // we're not going to hang around waiting
-- channel->Destroy();
-+ channel->DestroyLocked();
- }
- // At this point when we leave here, the object is a zombie held alive only by the DOM object
- }
-
- void DataChannelConnection::CloseAll()
- {
- LOG(("Closing all channels (connection %p)", (void*) this));
- // Don't need to lock here
-@@ -2552,23 +2552,25 @@ DataChannel::~DataChannel()
- // wrong, nothing bad happens. A worst it's a leak.
- NS_ASSERTION(mState == CLOSED || mState == CLOSING, "unexpected state in ~DataChannel");
- }
-
- void
- DataChannel::Close()
- {
- ENSURE_DATACONNECTION;
-+ RefPtr<DataChannelConnection> connection(mConnection);
- mConnection->Close(this);
- }
-
- // Used when disconnecting from the DataChannelConnection
- void
--DataChannel::Destroy()
-+DataChannel::DestroyLocked()
- {
-+ mConnection->mLock.AssertCurrentThreadOwns();
- ENSURE_DATACONNECTION;
-
- LOG(("Destroying Data channel %u", mStream));
- MOZ_ASSERT_IF(mStream != INVALID_STREAM,
- !mConnection->FindChannelByStream(mStream));
- mStream = INVALID_STREAM;
- mState = CLOSED;
- mConnection = nullptr;
-diff --git a/netwerk/sctp/datachannel/DataChannel.h b/netwerk/sctp/datachannel/DataChannel.h
---- a/netwerk/sctp/datachannel/DataChannel.h
-+++ b/netwerk/sctp/datachannel/DataChannel.h
-@@ -331,19 +331,20 @@ public:
- {
- NS_ASSERTION(mConnection,"NULL connection");
- }
-
- private:
- ~DataChannel();
-
- public:
-- void Destroy(); // when we disconnect from the connection after stream RESET
-+ NS_INLINE_DECL_THREADSAFE_REFCOUNTING(DataChannel)
-
-- NS_INLINE_DECL_THREADSAFE_REFCOUNTING(DataChannel)
-+ // when we disconnect from the connection after stream RESET
-+ void DestroyLocked();
-
- // Close this DataChannel. Can be called multiple times. MUST be called
- // before destroying the DataChannel (state must be CLOSED or CLOSING).
- void Close();
-
- // Set the listener (especially for channels created from the other side)
- void SetListener(DataChannelListener *aListener, nsISupports *aContext);
-
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1964.patch b/gnu/packages/patches/icecat-CVE-2016-1964.patch
deleted file mode 100644
index e53fc749b5..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1964.patch
+++ /dev/null
@@ -1,54 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/a653013e7b50
-
-# HG changeset patch
-# User Peter Van der Beken <peterv@propagandism.org>
-# Date 1454340035 -3600
-# Node ID a653013e7b503912a32621e8da64a37171316588
-# Parent 0d0d7e8292f7ecf5f1149d528c0524f04447c4ad
-Bug 1243335 - report bad QName. r=sicking, a=sylvestre
-
-diff --git a/dom/xslt/xslt/txInstructions.cpp b/dom/xslt/xslt/txInstructions.cpp
---- a/dom/xslt/xslt/txInstructions.cpp
-+++ b/dom/xslt/xslt/txInstructions.cpp
-@@ -93,16 +93,19 @@ txAttribute::txAttribute(nsAutoPtr<Expr>
- txNamespaceMap* aMappings)
- : mName(Move(aName)), mNamespace(Move(aNamespace)), mMappings(aMappings)
- {
- }
-
- nsresult
- txAttribute::execute(txExecutionState& aEs)
- {
-+ nsAutoPtr<txTextHandler> handler(
-+ static_cast<txTextHandler*>(aEs.popResultHandler()));
-+
- nsAutoString name;
- nsresult rv = mName->evaluateToString(aEs.getEvalContext(), name);
- NS_ENSURE_SUCCESS(rv, rv);
-
- const char16_t* colon;
- if (!XMLUtils::isValidQName(name, &colon) ||
- TX_StringEqualsAtom(name, nsGkAtoms::xmlns)) {
- return NS_OK;
-@@ -125,19 +128,16 @@ txAttribute::execute(txExecutionState& a
- if (!nspace.IsEmpty()) {
- nsId = txNamespaceManager::getNamespaceID(nspace);
- }
- }
- else if (colon) {
- nsId = mMappings->lookupNamespace(prefix);
- }
-
-- nsAutoPtr<txTextHandler> handler(
-- static_cast<txTextHandler*>(aEs.popResultHandler()));
--
- // add attribute if everything was ok
- return nsId != kNameSpaceID_Unknown ?
- aEs.mResultHandler->attribute(prefix, Substring(name, lnameStart),
- nsId, handler->mValue) :
- NS_OK;
- }
-
- txCallTemplate::txCallTemplate(const txExpandedName& aName)
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1965.patch b/gnu/packages/patches/icecat-CVE-2016-1965.patch
deleted file mode 100644
index 8a37d4975c..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1965.patch
+++ /dev/null
@@ -1,44 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/b4467681abd6
-
-# HG changeset patch
-# User Gijs Kruitbosch <gijskruitbosch@gmail.com>
-# Date 1455276061 0
-# Node ID b4467681abd676cd5575cbdf922927f8f54d2ad9
-# Parent 8c1d40e45a72c6432e879137a0afa519dc6c9841
-Bug 1245264 - r=bz, r=ritu
-
-MozReview-Commit-ID: I0sVdritpD3
-
-diff --git a/dom/base/nsLocation.cpp b/dom/base/nsLocation.cpp
---- a/dom/base/nsLocation.cpp
-+++ b/dom/base/nsLocation.cpp
-@@ -735,16 +735,27 @@ nsLocation::SetProtocol(const nsAString&
- return rv;
- }
-
- rv = uri->SetScheme(NS_ConvertUTF16toUTF8(aProtocol));
- if (NS_WARN_IF(NS_FAILED(rv))) {
- return rv;
- }
-
-+ nsAutoCString newSpec;
-+ rv = uri->GetSpec(newSpec);
-+ if (NS_FAILED(rv)) {
-+ return rv;
-+ }
-+ // We may want a new URI class for the new URI, so recreate it:
-+ rv = NS_NewURI(getter_AddRefs(uri), newSpec);
-+ if (NS_FAILED(rv)) {
-+ return rv;
-+ }
-+
- return SetURI(uri);
- }
-
- void
- nsLocation::GetUsername(nsAString& aUsername, ErrorResult& aError)
- {
- if (!CallerSubsumes()) {
- aError.Throw(NS_ERROR_DOM_SECURITY_ERR);
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1966.patch b/gnu/packages/patches/icecat-CVE-2016-1966.patch
deleted file mode 100644
index 6bf5f9f95e..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1966.patch
+++ /dev/null
@@ -1,36 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/291c2f31c48c
-
-# HG changeset patch
-# User Nicholas Nethercote <nnethercote@mozilla.com>
-# Date 1454650565 -39600
-# Node ID 291c2f31c48c7e96b1884b55273355970fa0fc30
-# Parent 11e6614756551cfd7291e73eefb90c52873a8480
-Bug 1246054 - Fix an erroneous nsNPObjWrapper assertion. r=froydnj. a=ritu
-
-diff --git a/dom/plugins/base/nsJSNPRuntime.cpp b/dom/plugins/base/nsJSNPRuntime.cpp
---- a/dom/plugins/base/nsJSNPRuntime.cpp
-+++ b/dom/plugins/base/nsJSNPRuntime.cpp
-@@ -1915,18 +1915,19 @@ nsNPObjWrapper::GetNewOrUsed(NPP npp, JS
- // No existing JSObject, create one.
-
- JS::Rooted<JSObject*> obj(cx, ::JS_NewObject(cx, js::Jsvalify(&sNPObjectJSWrapperClass)));
-
- if (generation != sNPObjWrappers.Generation()) {
- // Reload entry if the JS_NewObject call caused a GC and reallocated
- // the table (see bug 445229). This is guaranteed to succeed.
-
-- NS_ASSERTION(PL_DHashTableSearch(&sNPObjWrappers, npobj),
-- "Hashtable didn't find what we just added?");
-+ entry = static_cast<NPObjWrapperHashEntry*>
-+ (PL_DHashTableSearch(&sNPObjWrappers, npobj));
-+ NS_ASSERTION(entry, "Hashtable didn't find what we just added?");
- }
-
- if (!obj) {
- // OOM? Remove the stale entry from the hash.
-
- PL_DHashTableRawRemove(&sNPObjWrappers, entry);
-
- return nullptr;
-
diff --git a/gnu/packages/patches/icecat-CVE-2016-1974.patch b/gnu/packages/patches/icecat-CVE-2016-1974.patch
deleted file mode 100644
index 70fc23b8f3..0000000000
--- a/gnu/packages/patches/icecat-CVE-2016-1974.patch
+++ /dev/null
@@ -1,530 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/271e3a5a53d9
-
-# HG changeset patch
-# User Henri Sivonen <hsivonen@hsivonen.fi>
-# Date 1455014759 -7200
-# Node ID 271e3a5a53d96871141e89271f611033b512e3e4
-# Parent 9719b71d72dd2a3c5ee12ace156af2a63d9595ac
-Bug 1228103. r=smaug. a=sylvestre
-
-diff --git a/parser/htmlparser/nsExpatDriver.cpp b/parser/htmlparser/nsExpatDriver.cpp
---- a/parser/htmlparser/nsExpatDriver.cpp
-+++ b/parser/htmlparser/nsExpatDriver.cpp
-@@ -1127,22 +1127,28 @@ nsExpatDriver::ConsumeToken(nsScanner& a
- XML_Size lastLineLength = XML_GetCurrentColumnNumber(mExpatParser);
-
- if (lastLineLength <= consumed) {
- // The length of the last line was less than what expat consumed, so
- // there was at least one line break in the consumed data. Store the
- // last line until the point where we stopped parsing.
- nsScannerIterator startLastLine = currentExpatPosition;
- startLastLine.advance(-((ptrdiff_t)lastLineLength));
-- CopyUnicodeTo(startLastLine, currentExpatPosition, mLastLine);
-+ if (!CopyUnicodeTo(startLastLine, currentExpatPosition, mLastLine)) {
-+ return (mInternalState = NS_ERROR_OUT_OF_MEMORY);
-+ }
- }
- else {
- // There was no line break in the consumed data, append the consumed
- // data.
-- AppendUnicodeTo(oldExpatPosition, currentExpatPosition, mLastLine);
-+ if (!AppendUnicodeTo(oldExpatPosition,
-+ currentExpatPosition,
-+ mLastLine)) {
-+ return (mInternalState = NS_ERROR_OUT_OF_MEMORY);
-+ }
- }
- }
-
- mExpatBuffered += length - consumed;
-
- if (BlockedOrInterrupted()) {
- PR_LOG(GetExpatDriverLog(), PR_LOG_DEBUG,
- ("Blocked or interrupted parser (probably for loading linked "
-diff --git a/parser/htmlparser/nsParser.cpp b/parser/htmlparser/nsParser.cpp
---- a/parser/htmlparser/nsParser.cpp
-+++ b/parser/htmlparser/nsParser.cpp
-@@ -1508,17 +1508,19 @@ nsParser::ResumeParse(bool allowIteratio
- DidBuildModel(mStreamStatus);
- return NS_OK;
- }
- } else {
- CParserContext* theContext = PopContext();
- if (theContext) {
- theIterationIsOk = allowIteration && theContextIsStringBased;
- if (theContext->mCopyUnused) {
-- theContext->mScanner->CopyUnusedData(mUnusedInput);
-+ if (!theContext->mScanner->CopyUnusedData(mUnusedInput)) {
-+ mInternalState = NS_ERROR_OUT_OF_MEMORY;
-+ }
- }
-
- delete theContext;
- }
-
- result = mInternalState;
- aIsFinalChunk = mParserContext &&
- mParserContext->mStreamListenerState == eOnStop;
-diff --git a/parser/htmlparser/nsScanner.cpp b/parser/htmlparser/nsScanner.cpp
---- a/parser/htmlparser/nsScanner.cpp
-+++ b/parser/htmlparser/nsScanner.cpp
-@@ -379,17 +379,19 @@ nsresult nsScanner::Peek(nsAString& aStr
- if (mCountRemaining < uint32_t(aNumChars + aOffset)) {
- end = mEndPosition;
- }
- else {
- end = start;
- end.advance(aNumChars);
- }
-
-- CopyUnicodeTo(start, end, aStr);
-+ if (!CopyUnicodeTo(start, end, aStr)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
-
- return NS_OK;
- }
-
-
- /**
- * Skip whitespace on scanner input stream
- *
-@@ -542,17 +544,19 @@ nsresult nsScanner::ReadTagIdentifier(ns
-
- if (!found) {
- ++current;
- }
- }
-
- // Don't bother appending nothing.
- if (current != mCurrentPosition) {
-- AppendUnicodeTo(mCurrentPosition, current, aString);
-+ if (!AppendUnicodeTo(mCurrentPosition, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- }
-
- SetPosition(current);
- if (current == end) {
- result = kEOF;
- }
-
- //DoErrTest(aString);
-@@ -597,26 +601,30 @@ nsresult nsScanner::ReadEntityIdentifier
- default:
- found = ('a'<=theChar && theChar<='z') ||
- ('A'<=theChar && theChar<='Z') ||
- ('0'<=theChar && theChar<='9');
- break;
- }
-
- if(!found) {
-- AppendUnicodeTo(mCurrentPosition, current, aString);
-+ if (!AppendUnicodeTo(mCurrentPosition, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- break;
- }
- }
- ++current;
- }
-
- SetPosition(current);
- if (current == end) {
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- return kEOF;
- }
-
- //DoErrTest(aString);
-
- return result;
- }
-
-@@ -646,26 +654,30 @@ nsresult nsScanner::ReadNumber(nsString&
- while(current != end) {
- theChar=*current;
- if(theChar) {
- done = (theChar < '0' || theChar > '9') &&
- ((aBase == 16)? (theChar < 'A' || theChar > 'F') &&
- (theChar < 'a' || theChar > 'f')
- :true);
- if(done) {
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- break;
- }
- }
- ++current;
- }
-
- SetPosition(current);
- if (current == end) {
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- return kEOF;
- }
-
- //DoErrTest(aString);
-
- return result;
- }
-
-@@ -712,37 +724,43 @@ nsresult nsScanner::ReadWhitespace(nsSca
- char16_t thePrevChar = theChar;
- theChar = (++current != end) ? *current : '\0';
- if ((thePrevChar == '\r' && theChar == '\n') ||
- (thePrevChar == '\n' && theChar == '\r')) {
- theChar = (++current != end) ? *current : '\0'; // CRLF == LFCR => LF
- haveCR = true;
- } else if (thePrevChar == '\r') {
- // Lone CR becomes CRLF; callers should know to remove extra CRs
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- aString.writable().Append(char16_t('\n'));
- origin = current;
- haveCR = true;
- }
- }
- break;
- case ' ' :
- case '\t':
- theChar = (++current != end) ? *current : '\0';
- break;
- default:
- done = true;
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- break;
- }
- }
-
- SetPosition(current);
- if (current == end) {
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- result = kEOF;
- }
-
- aHaveCR = haveCR;
- return result;
- }
-
- //XXXbz callers of this have to manage their lone '\r' themselves if they want
-@@ -846,34 +864,38 @@ nsresult nsScanner::ReadUntil(nsAString&
- if(!(theChar & aEndCondition.mFilter)) {
- // They were. Do a thorough check.
-
- setcurrent = setstart;
- while (*setcurrent) {
- if (*setcurrent == theChar) {
- if(addTerminal)
- ++current;
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- SetPosition(current);
-
- //DoErrTest(aString);
-
- return NS_OK;
- }
- ++setcurrent;
- }
- }
-
- ++current;
- }
-
- // If we are here, we didn't find any terminator in the string and
- // current = mEndPosition
- SetPosition(current);
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- return kEOF;
- }
-
- nsresult nsScanner::ReadUntil(nsScannerSharedSubstring& aString,
- const nsReadEndCondition& aEndCondition,
- bool addTerminal)
- {
- if (!mSlidingBuffer) {
-@@ -906,34 +928,38 @@ nsresult nsScanner::ReadUntil(nsScannerS
- if(!(theChar & aEndCondition.mFilter)) {
- // They were. Do a thorough check.
-
- setcurrent = setstart;
- while (*setcurrent) {
- if (*setcurrent == theChar) {
- if(addTerminal)
- ++current;
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- SetPosition(current);
-
- //DoErrTest(aString);
-
- return NS_OK;
- }
- ++setcurrent;
- }
- }
-
- ++current;
- }
-
- // If we are here, we didn't find any terminator in the string and
- // current = mEndPosition
- SetPosition(current);
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- return kEOF;
- }
-
- nsresult nsScanner::ReadUntil(nsScannerIterator& aStart,
- nsScannerIterator& aEnd,
- const nsReadEndCondition &aEndCondition,
- bool addTerminal)
- {
-@@ -1025,26 +1051,30 @@ nsresult nsScanner::ReadUntil(nsAString&
- if (theChar == '\0') {
- ReplaceCharacter(current, sInvalid);
- theChar = sInvalid;
- }
-
- if (aTerminalChar == theChar) {
- if(addTerminal)
- ++current;
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- SetPosition(current);
- return NS_OK;
- }
- ++current;
- }
-
- // If we are here, we didn't find any terminator in the string and
- // current = mEndPosition
-- AppendUnicodeTo(origin, current, aString);
-+ if (!AppendUnicodeTo(origin, current, aString)) {
-+ return NS_ERROR_OUT_OF_MEMORY;
-+ }
- SetPosition(current);
- return kEOF;
-
- }
-
- void nsScanner::BindSubstring(nsScannerSubstring& aSubstring, const nsScannerIterator& aStart, const nsScannerIterator& aEnd)
- {
- aSubstring.Rebind(*mSlidingBuffer, aStart, aEnd);
-@@ -1142,29 +1172,29 @@ bool nsScanner::AppendToBuffer(nsScanner
- }
-
- /**
- * call this to copy bytes out of the scanner that have not yet been consumed
- * by the tokenization process.
- *
- * @update gess 5/12/98
- * @param aCopyBuffer is where the scanner buffer will be copied to
-- * @return nada
-+ * @return true if OK or false on OOM
- */
--void nsScanner::CopyUnusedData(nsString& aCopyBuffer) {
-+bool nsScanner::CopyUnusedData(nsString& aCopyBuffer) {
- if (!mSlidingBuffer) {
- aCopyBuffer.Truncate();
-- return;
-+ return true;
- }
-
- nsScannerIterator start, end;
- start = mCurrentPosition;
- end = mEndPosition;
-
-- CopyUnicodeTo(start, end, aCopyBuffer);
-+ return CopyUnicodeTo(start, end, aCopyBuffer);
- }
-
- /**
- * Retrieve the name of the file that the scanner is reading from.
- * In some cases, it's just a given name, because the scanner isn't
- * really reading from a file.
- *
- * @update gess 5/12/98
-diff --git a/parser/htmlparser/nsScanner.h b/parser/htmlparser/nsScanner.h
---- a/parser/htmlparser/nsScanner.h
-+++ b/parser/htmlparser/nsScanner.h
-@@ -204,19 +204,19 @@ class nsScanner {
- nsIRequest *aRequest);
-
- /**
- * Call this to copy bytes out of the scanner that have not yet been consumed
- * by the tokenization process.
- *
- * @update gess 5/12/98
- * @param aCopyBuffer is where the scanner buffer will be copied to
-- * @return nada
-+ * @return true if OK or false on OOM
- */
-- void CopyUnusedData(nsString& aCopyBuffer);
-+ bool CopyUnusedData(nsString& aCopyBuffer);
-
- /**
- * Retrieve the name of the file that the scanner is reading from.
- * In some cases, it's just a given name, because the scanner isn't
- * really reading from a file.
- *
- * @update gess 5/12/98
- * @return
-diff --git a/parser/htmlparser/nsScannerString.cpp b/parser/htmlparser/nsScannerString.cpp
---- a/parser/htmlparser/nsScannerString.cpp
-+++ b/parser/htmlparser/nsScannerString.cpp
-@@ -461,61 +461,63 @@ copy_multifragment_string( nsScannerIter
- sink_traits::write(result, source_traits::read(first), distance);
- NS_ASSERTION(distance > 0, "|copy_multifragment_string| will never terminate");
- source_traits::advance(first, distance);
- }
-
- return result;
- }
-
--void
-+bool
- CopyUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsAString& aDest )
- {
- nsAString::iterator writer;
- if (!aDest.SetLength(Distance(aSrcStart, aSrcEnd), mozilla::fallible)) {
- aDest.Truncate();
-- return; // out of memory
-+ return false; // out of memory
- }
- aDest.BeginWriting(writer);
- nsScannerIterator fromBegin(aSrcStart);
-
- copy_multifragment_string(fromBegin, aSrcEnd, writer);
-+ return true;
- }
-
--void
-+bool
- AppendUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsScannerSharedSubstring& aDest )
- {
- // Check whether we can just create a dependent string.
- if (aDest.str().IsEmpty()) {
- // We can just make |aDest| point to the buffer.
- // This will take care of copying if the buffer spans fragments.
- aDest.Rebind(aSrcStart, aSrcEnd);
-- } else {
-- // The dest string is not empty, so it can't be a dependent substring.
-- AppendUnicodeTo(aSrcStart, aSrcEnd, aDest.writable());
-+ return true;
- }
-+ // The dest string is not empty, so it can't be a dependent substring.
-+ return AppendUnicodeTo(aSrcStart, aSrcEnd, aDest.writable());
- }
-
--void
-+bool
- AppendUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsAString& aDest )
- {
- nsAString::iterator writer;
- uint32_t oldLength = aDest.Length();
- if (!aDest.SetLength(oldLength + Distance(aSrcStart, aSrcEnd), mozilla::fallible))
-- return; // out of memory
-+ return false; // out of memory
- aDest.BeginWriting(writer).advance(oldLength);
- nsScannerIterator fromBegin(aSrcStart);
-
- copy_multifragment_string(fromBegin, aSrcEnd, writer);
-+ return true;
- }
-
- bool
- FindCharInReadable( char16_t aChar,
- nsScannerIterator& aSearchStart,
- const nsScannerIterator& aSearchEnd )
- {
- while ( aSearchStart != aSearchEnd )
-diff --git a/parser/htmlparser/nsScannerString.h b/parser/htmlparser/nsScannerString.h
---- a/parser/htmlparser/nsScannerString.h
-+++ b/parser/htmlparser/nsScannerString.h
-@@ -539,43 +539,43 @@ nsScannerBufferList::Position::operator=
- inline
- size_t
- Distance( const nsScannerIterator& aStart, const nsScannerIterator& aEnd )
- {
- typedef nsScannerBufferList::Position Position;
- return Position::Distance(Position(aStart), Position(aEnd));
- }
-
--void
-+bool
- CopyUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsAString& aDest );
-
- inline
--void
-+bool
- CopyUnicodeTo( const nsScannerSubstring& aSrc, nsAString& aDest )
- {
- nsScannerIterator begin, end;
-- CopyUnicodeTo(aSrc.BeginReading(begin), aSrc.EndReading(end), aDest);
-+ return CopyUnicodeTo(aSrc.BeginReading(begin), aSrc.EndReading(end), aDest);
- }
-
--void
-+bool
- AppendUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsAString& aDest );
-
- inline
--void
-+bool
- AppendUnicodeTo( const nsScannerSubstring& aSrc, nsAString& aDest )
- {
- nsScannerIterator begin, end;
-- AppendUnicodeTo(aSrc.BeginReading(begin), aSrc.EndReading(end), aDest);
-+ return AppendUnicodeTo(aSrc.BeginReading(begin), aSrc.EndReading(end), aDest);
- }
-
--void
-+bool
- AppendUnicodeTo( const nsScannerIterator& aSrcStart,
- const nsScannerIterator& aSrcEnd,
- nsScannerSharedSubstring& aDest );
-
- bool
- FindCharInReadable( char16_t aChar,
- nsScannerIterator& aStart,
- const nsScannerIterator& aEnd );
-
diff --git a/gnu/packages/patches/icecat-bug-1248851.patch b/gnu/packages/patches/icecat-bug-1248851.patch
deleted file mode 100644
index ea4d6831b5..0000000000
--- a/gnu/packages/patches/icecat-bug-1248851.patch
+++ /dev/null
@@ -1,37 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/8c1d40e45a72
-
-# HG changeset patch
-# User Xidorn Quan <quanxunzhen@gmail.com>
-# Date 1456199544 -28800
-# Node ID 8c1d40e45a72c6432e879137a0afa519dc6c9841
-# Parent 1dd0ca8e70bd77b6fd93f36cc4e9c2cebfe8ba0a
-Bug 1248851 - r=sicking, a=ritu
-
-diff --git a/dom/indexedDB/ActorsParent.cpp b/dom/indexedDB/ActorsParent.cpp
---- a/dom/indexedDB/ActorsParent.cpp
-+++ b/dom/indexedDB/ActorsParent.cpp
-@@ -14823,22 +14823,19 @@ ObjectStoreAddOrPutRequestOp::DoDatabase
- }
-
- snappy::RawCompress(uncompressed, uncompressedLength, compressed,
- &compressedLength);
-
- uint8_t* dataBuffer = reinterpret_cast<uint8_t*>(compressed);
- size_t dataBufferLength = compressedLength;
-
-- // If this call succeeds, | compressed | is now owned by the statement, and
-- // we are no longer responsible for it.
- rv = stmt->BindAdoptedBlobByName(NS_LITERAL_CSTRING("data"), dataBuffer,
- dataBufferLength);
- if (NS_WARN_IF(NS_FAILED(rv))) {
-- moz_free(compressed);
- return rv;
- }
- }
-
- nsCOMPtr<nsIFile> fileDirectory;
- nsCOMPtr<nsIFile> journalDirectory;
-
- if (mFileManager) {
-
diff --git a/gnu/packages/patches/icecat-update-graphite2-pt2.patch b/gnu/packages/patches/icecat-update-graphite2-pt2.patch
deleted file mode 100644
index 8acde75d6c..0000000000
--- a/gnu/packages/patches/icecat-update-graphite2-pt2.patch
+++ /dev/null
@@ -1,861 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/ec9cff7bb543
-
-# HG changeset patch
-# User Jonathan Kew <jkew@mozilla.com>
-# Date 1456760339 0
-# Node ID ec9cff7bb5439b2b4c1249ff9376d07a80172c27
-# Parent 6f4d5130238790fa5810c76ffeb9eccc65efa8c9
-Bug 1248876 - Update graphite2 to upstream release 1.3.6. r=jrmuizel a=sledru
-
-diff --git a/gfx/graphite2/README.mozilla b/gfx/graphite2/README.mozilla
---- a/gfx/graphite2/README.mozilla
-+++ b/gfx/graphite2/README.mozilla
-@@ -1,7 +1,3 @@
--This directory contains the Graphite2 library release 1.3.5 from
--https://github.com/silnrsi/graphite/releases/download/1.3.5/graphite2-minimal-1.3.5.tgz
-+This directory contains the Graphite2 library release 1.3.6 from
-+https://github.com/silnrsi/graphite/releases/download/1.3.6/graphite-minimal-1.3.6.tgz
- See gfx/graphite2/moz-gr-update.sh for update procedure.
--
--Also includes two post-1.3.5 fixes:
--a8b3ac2aed0eb132cd80efe7de88f8153e73c829
--e569e28d83491fedb31b9220493f3c07f6ec6d80
-diff --git a/gfx/graphite2/include/graphite2/Font.h b/gfx/graphite2/include/graphite2/Font.h
---- a/gfx/graphite2/include/graphite2/Font.h
-+++ b/gfx/graphite2/include/graphite2/Font.h
-@@ -25,17 +25,17 @@
- either version 2 of the License or (at your option) any later version.
- */
- #pragma once
-
- #include "graphite2/Types.h"
-
- #define GR2_VERSION_MAJOR 1
- #define GR2_VERSION_MINOR 3
--#define GR2_VERSION_BUGFIX 5
-+#define GR2_VERSION_BUGFIX 6
-
- #ifdef __cplusplus
- extern "C"
- {
- #endif
-
- typedef struct gr_face gr_face;
- typedef struct gr_font gr_font;
-diff --git a/gfx/graphite2/moz-gr-update.sh b/gfx/graphite2/moz-gr-update.sh
---- a/gfx/graphite2/moz-gr-update.sh
-+++ b/gfx/graphite2/moz-gr-update.sh
-@@ -14,17 +14,17 @@
- RELEASE=$1
-
- if [ "x$RELEASE" == "x" ]
- then
- echo "Must provide the version number to be used."
- exit 1
- fi
-
--TARBALL="https://github.com/silnrsi/graphite/releases/download/$RELEASE/graphite2-minimal-$RELEASE.tgz"
-+TARBALL="https://github.com/silnrsi/graphite/releases/download/$RELEASE/graphite-minimal-$RELEASE.tgz"
-
- foo=`basename $0`
- TMPFILE=`mktemp -t ${foo}` || exit 1
-
- curl -L "$TARBALL" -o "$TMPFILE"
- tar -x -z -C gfx/graphite2/ --strip-components 1 -f "$TMPFILE" || exit 1
- rm "$TMPFILE"
-
-diff --git a/gfx/graphite2/src/CmapCache.cpp b/gfx/graphite2/src/CmapCache.cpp
---- a/gfx/graphite2/src/CmapCache.cpp
-+++ b/gfx/graphite2/src/CmapCache.cpp
-@@ -33,43 +33,43 @@ of the License or (at your option) any l
-
-
- using namespace graphite2;
-
- const void * bmp_subtable(const Face::Table & cmap)
- {
- const void * stbl;
- if (!cmap.size()) return 0;
-- if (TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 1, cmap.size()), cmap.size())
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 3, cmap.size()), cmap.size())
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 2, cmap.size()), cmap.size())
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 1, cmap.size()), cmap.size())
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 0, cmap.size()), cmap.size()))
-+ if (TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 1, cmap.size()), cmap + cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 3, cmap.size()), cmap + cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 2, cmap.size()), cmap + cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 1, cmap.size()), cmap + cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 0, cmap.size()), cmap + cmap.size()))
- return stbl;
- return 0;
- }
-
- const void * smp_subtable(const Face::Table & cmap)
- {
- const void * stbl;
- if (!cmap.size()) return 0;
-- if (TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 10, cmap.size()), cmap.size())
-- || TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 4, cmap.size()), cmap.size()))
-+ if (TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 10, cmap.size()), cmap + cmap.size())
-+ || TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 4, cmap.size()), cmap + cmap.size()))
- return stbl;
- return 0;
- }
-
- template <unsigned int (*NextCodePoint)(const void *, unsigned int, int *),
- uint16 (*LookupCodePoint)(const void *, unsigned int, int)>
- bool cache_subtable(uint16 * blocks[], const void * cst, const unsigned int limit)
- {
- int rangeKey = 0;
- uint32 codePoint = NextCodePoint(cst, 0, &rangeKey),
- prevCodePoint = 0;
-- while (codePoint != limit)
-+ while (codePoint < limit)
- {
- unsigned int block = codePoint >> 8;
- if (!blocks[block])
- {
- blocks[block] = grzeroalloc<uint16>(0x100);
- if (!blocks[block])
- return false;
- }
-diff --git a/gfx/graphite2/src/Code.cpp b/gfx/graphite2/src/Code.cpp
---- a/gfx/graphite2/src/Code.cpp
-+++ b/gfx/graphite2/src/Code.cpp
-@@ -79,18 +79,19 @@ struct context
-
-
- class Machine::Code::decoder
- {
- public:
- struct limits;
- struct analysis
- {
-+ static const int NUMCONTEXTS = 256;
- uint8 slotref;
-- context contexts[256];
-+ context contexts[NUMCONTEXTS];
- byte max_ref;
-
- analysis() : slotref(0), max_ref(0) {};
- void set_ref(int index, bool incinsert=false) throw();
- void set_noref(int index) throw();
- void set_changed(int index) throw();
-
- };
-@@ -363,29 +364,33 @@ opcode Machine::Code::decoder::fetch_opc
- break;
- case ATTR_SET :
- case ATTR_ADD :
- case ATTR_SUB :
- case ATTR_SET_SLOT :
- if (--_stack_depth < 0)
- failure(underfull_stack);
- valid_upto(gr_slatMax, bc[0]);
-+ if (attrCode(bc[0]) == gr_slatUserDefn) // use IATTR for user attributes
-+ failure(out_of_range_data);
- test_context();
- break;
- case IATTR_SET_SLOT :
- if (--_stack_depth < 0)
- failure(underfull_stack);
- if (valid_upto(gr_slatMax, bc[0]))
- valid_upto(_max.attrid[bc[0]], bc[1]);
- test_context();
- break;
- case PUSH_SLOT_ATTR :
- ++_stack_depth;
- valid_upto(gr_slatMax, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
-+ if (attrCode(bc[0]) == gr_slatUserDefn) // use IATTR for user attributes
-+ failure(out_of_range_data);
- break;
- case PUSH_GLYPH_ATTR_OBS :
- ++_stack_depth;
- valid_upto(_max.glyf_attrs, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- break;
- case PUSH_GLYPH_METRIC :
- ++_stack_depth;
-@@ -656,24 +661,24 @@ bool Machine::Code::decoder::validate_op
- return false;
- }
- return true;
- }
-
-
- bool Machine::Code::decoder::valid_upto(const uint16 limit, const uint16 x) const throw()
- {
-- const bool t = x < limit;
-+ const bool t = (limit != 0) && (x < limit);
- if (!t) failure(out_of_range_data);
- return t;
- }
-
- bool Machine::Code::decoder::test_context() const throw()
- {
-- if (_pre_context >= _rule_length)
-+ if (_pre_context >= _rule_length || _analysis.slotref >= analysis::NUMCONTEXTS - 1)
- {
- failure(out_of_range_data);
- return false;
- }
- return true;
- }
-
- inline
-@@ -681,34 +686,34 @@ void Machine::Code::failure(const status
- release_buffers();
- _status = s;
- }
-
-
- inline
- void Machine::Code::decoder::analysis::set_ref(int index, bool incinsert) throw() {
- if (incinsert && contexts[slotref].flags.inserted) --index;
-- if (index + slotref < 0) return;
-+ if (index + slotref < 0 || index + slotref >= NUMCONTEXTS) return;
- contexts[index + slotref].flags.referenced = true;
- if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
- }
-
-
- inline
- void Machine::Code::decoder::analysis::set_noref(int index) throw() {
- if (contexts[slotref].flags.inserted) --index;
-- if (index + slotref < 0) return;
-+ if (index + slotref < 0 || index + slotref >= NUMCONTEXTS) return;
- if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
- }
-
-
- inline
- void Machine::Code::decoder::analysis::set_changed(int index) throw() {
- if (contexts[slotref].flags.inserted) --index;
-- if (index + slotref < 0) return;
-+ if (index + slotref < 0 || index + slotref >= NUMCONTEXTS) return;
- contexts[index + slotref].flags.changed = true;
- if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
- }
-
-
- void Machine::Code::release_buffers() throw()
- {
- if (_own)
-diff --git a/gfx/graphite2/src/GlyphCache.cpp b/gfx/graphite2/src/GlyphCache.cpp
---- a/gfx/graphite2/src/GlyphCache.cpp
-+++ b/gfx/graphite2/src/GlyphCache.cpp
-@@ -260,17 +260,17 @@ GlyphCache::Loader::Loader(const Face &
- _head = Face::Table();
- return;
- }
-
- if (!dumb_font)
- {
- if ((m_pGlat = Face::Table(face, Tag::Glat, 0x00030000)) == NULL
- || (m_pGloc = Face::Table(face, Tag::Gloc)) == NULL
-- || m_pGloc.size() < 6)
-+ || m_pGloc.size() < 8)
- {
- _head = Face::Table();
- return;
- }
- const byte * p = m_pGloc;
- int version = be::read<uint32>(p);
- const uint16 flags = be::read<uint16>(p);
- _num_attrs = be::read<uint16>(p);
-diff --git a/gfx/graphite2/src/Pass.cpp b/gfx/graphite2/src/Pass.cpp
---- a/gfx/graphite2/src/Pass.cpp
-+++ b/gfx/graphite2/src/Pass.cpp
-@@ -233,17 +233,17 @@ bool Pass::readRules(const byte * rule_m
- m_codes = new Code [m_numRules*2];
- const size_t prog_pool_sz = vm::Machine::Code::estimateCodeDataOut(ac_end - ac_data + rc_end - rc_data);
- m_progs = gralloc<byte>(prog_pool_sz);
- byte * prog_pool_free = m_progs,
- * prog_pool_end = m_progs + prog_pool_sz;
- if (e.test(!(m_rules && m_codes && m_progs), E_OUTOFMEM)) return face.error(e);
-
- Rule * r = m_rules + m_numRules - 1;
-- for (size_t n = m_numRules; n; --n, --r, ac_end = ac_begin, rc_end = rc_begin)
-+ for (size_t n = m_numRules; r >= m_rules; --n, --r, ac_end = ac_begin, rc_end = rc_begin)
- {
- face.error_context((face.error_context() & 0xFFFF00) + EC_ARULE + ((n - 1) << 24));
- r->preContext = *--precontext;
- r->sort = be::peek<uint16>(--sort_key);
- #ifndef NDEBUG
- r->rule_idx = n - 1;
- #endif
- if (r->sort > 63 || r->preContext >= r->sort || r->preContext > m_maxPreCtxt || r->preContext < m_minPreCtxt)
-@@ -405,16 +405,17 @@ bool Pass::runGraphite(vm::Machine & m,
- json::closer rules_array_closer(fsm.dbgout);
- #endif
-
- m.slotMap().highwater(currHigh);
- int lc = m_iMaxLoop;
- do
- {
- findNDoRule(s, m, fsm);
-+ if (m.status() != Machine::finished) return false;
- if (s && (s == m.slotMap().highwater() || m.slotMap().highpassed() || --lc == 0)) {
- if (!lc)
- s = m.slotMap().highwater();
- lc = m_iMaxLoop;
- if (s)
- m.slotMap().highwater(s->next());
- }
- } while (s);
-@@ -495,17 +496,22 @@ void Pass::findNDoRule(Slot * & slot, Ma
- {
- assert(slot);
-
- if (runFSM(fsm, slot))
- {
- // Search for the first rule which passes the constraint
- const RuleEntry * r = fsm.rules.begin(),
- * const re = fsm.rules.end();
-- while (r != re && !testConstraint(*r->rule, m)) ++r;
-+ while (r != re && !testConstraint(*r->rule, m))
-+ {
-+ ++r;
-+ if (m.status() != Machine::finished)
-+ return;
-+ }
-
- #if !defined GRAPHITE2_NTRACING
- if (fsm.dbgout)
- {
- if (fsm.rules.size() != 0)
- {
- *fsm.dbgout << json::item << json::object;
- dumpRuleEventConsidered(fsm, *r);
-@@ -530,16 +536,17 @@ void Pass::findNDoRule(Slot * & slot, Ma
- }
- }
- else
- #endif
- {
- if (r != re)
- {
- const int adv = doAction(r->rule->action, slot, m);
-+ if (m.status() != Machine::finished) return;
- if (r->rule->action->deletes()) fsm.slots.collectGarbage(slot);
- adjustSlot(adv, slot, fsm.slots);
- return;
- }
- }
- }
-
- slot = slot->next();
-diff --git a/gfx/graphite2/src/Segment.cpp b/gfx/graphite2/src/Segment.cpp
---- a/gfx/graphite2/src/Segment.cpp
-+++ b/gfx/graphite2/src/Segment.cpp
-@@ -205,18 +205,23 @@ Slot *Segment::newSlot()
- void Segment::freeSlot(Slot *aSlot)
- {
- if (m_last == aSlot) m_last = aSlot->prev();
- if (m_first == aSlot) m_first = aSlot->next();
- if (aSlot->attachedTo())
- aSlot->attachedTo()->removeChild(aSlot);
- while (aSlot->firstChild())
- {
-- aSlot->firstChild()->attachTo(NULL);
-- aSlot->removeChild(aSlot->firstChild());
-+ if (aSlot->firstChild()->attachedTo() == aSlot)
-+ {
-+ aSlot->firstChild()->attachTo(NULL);
-+ aSlot->removeChild(aSlot->firstChild());
-+ }
-+ else
-+ aSlot->firstChild(NULL);
- }
- // reset the slot incase it is reused
- ::new (aSlot) Slot(aSlot->userAttrs());
- memset(aSlot->userAttrs(), 0, m_silf->numUser() * sizeof(int16));
- // Update generation counter for debug
- #if !defined GRAPHITE2_NTRACING
- if (m_face->logger())
- ++aSlot->userAttrs()[m_silf->numUser()];
-diff --git a/gfx/graphite2/src/Slot.cpp b/gfx/graphite2/src/Slot.cpp
---- a/gfx/graphite2/src/Slot.cpp
-+++ b/gfx/graphite2/src/Slot.cpp
-@@ -192,16 +192,18 @@ int32 Slot::clusterMetric(const Segment
- #define SLOTGETCOLATTR(x) { SlotCollision *c = seg->collisionInfo(this); return c ? int(c-> x) : 0; }
-
- int Slot::getAttr(const Segment *seg, attrCode ind, uint8 subindex) const
- {
- if (ind == gr_slatUserDefnV1)
- {
- ind = gr_slatUserDefn;
- subindex = 0;
-+ if (seg->numAttrs() == 0)
-+ return 0;
- }
- else if (ind >= gr_slatJStretch && ind < gr_slatJStretch + 20 && ind != gr_slatJWidth)
- {
- int indx = ind - gr_slatJStretch;
- return getJustify(seg, indx / 5, indx % 5);
- }
-
- switch (ind)
-@@ -269,16 +271,18 @@ int Slot::getAttr(const Segment *seg, at
- break; }
-
- void Slot::setAttr(Segment *seg, attrCode ind, uint8 subindex, int16 value, const SlotMap & map)
- {
- if (ind == gr_slatUserDefnV1)
- {
- ind = gr_slatUserDefn;
- subindex = 0;
-+ if (seg->numAttrs() == 0)
-+ return;
- }
- else if (ind >= gr_slatJStretch && ind < gr_slatJStretch + 20 && ind != gr_slatJWidth)
- {
- int indx = ind - gr_slatJStretch;
- return setJustify(seg, indx / 5, indx % 5, value);
- }
-
- switch (ind)
-@@ -416,32 +420,32 @@ bool Slot::sibling(Slot *ap)
- }
-
- bool Slot::removeChild(Slot *ap)
- {
- if (this == ap || !m_child) return false;
- else if (ap == m_child)
- {
- Slot *nSibling = m_child->nextSibling();
-- m_child->sibling(NULL);
-+ m_child->removeSibling(nSibling);
- m_child = nSibling;
- return true;
- }
- else
- return m_child->removeSibling(ap);
- return true;
- }
-
- bool Slot::removeSibling(Slot *ap)
- {
- if (this == ap || !m_sibling) return false;
- else if (ap == m_sibling)
- {
- m_sibling = m_sibling->nextSibling();
-- ap->sibling(NULL);
-+ if (m_sibling) ap->removeSibling(m_sibling);
- return true;
- }
- else
- return m_sibling->removeSibling(ap);
- return true;
- }
-
- void Slot::setGlyph(Segment *seg, uint16 glyphid, const GlyphFace * theGlyph)
-diff --git a/gfx/graphite2/src/TtfUtil.cpp b/gfx/graphite2/src/TtfUtil.cpp
---- a/gfx/graphite2/src/TtfUtil.cpp
-+++ b/gfx/graphite2/src/TtfUtil.cpp
-@@ -884,18 +884,19 @@ const void * FindCmapSubtable(const void
- }
-
- return 0;
- }
-
- /*----------------------------------------------------------------------------------------------
- Check the Microsoft Unicode subtable for expected values
- ----------------------------------------------------------------------------------------------*/
--bool CheckCmapSubtable4(const void * pCmapSubtable4, size_t table_len /*, unsigned int maxgid*/)
-+bool CheckCmapSubtable4(const void * pCmapSubtable4, const void * pCmapEnd /*, unsigned int maxgid*/)
- {
-+ size_t table_len = (const byte *)pCmapEnd - (const byte *)pCmapSubtable4;
- if (!pCmapSubtable4) return false;
- const Sfnt::CmapSubTable * pTable = reinterpret_cast<const Sfnt::CmapSubTable *>(pCmapSubtable4);
- // Bob H say some freeware TT fonts have version 1 (eg, CALIGULA.TTF)
- // so don't check subtable version. 21 Mar 2002 spec changes version to language.
- if (be::swap(pTable->format) != 4) return false;
- const Sfnt::CmapSubTableFormat4 * pTable4 = reinterpret_cast<const Sfnt::CmapSubTableFormat4 *>(pCmapSubtable4);
- uint16 length = be::swap(pTable4->length);
- if (length > table_len)
-@@ -1044,17 +1045,17 @@ unsigned int CmapSubtable4NextCodepoint(
- *pRangeKey = nRange - 1;
- return 0xFFFF;
- }
-
- int iRange = (pRangeKey) ? *pRangeKey : 0;
- // Just in case we have a bad key:
- while (iRange > 0 && be::peek<uint16>(pStartCode + iRange) > nUnicodePrev)
- iRange--;
-- while (be::peek<uint16>(pTable->end_code + iRange) < nUnicodePrev)
-+ while (iRange < nRange - 1 && be::peek<uint16>(pTable->end_code + iRange) < nUnicodePrev)
- iRange++;
-
- // Now iRange is the range containing nUnicodePrev.
- unsigned int nStartCode = be::peek<uint16>(pStartCode + iRange);
- unsigned int nEndCode = be::peek<uint16>(pTable->end_code + iRange);
-
- if (nStartCode > nUnicodePrev)
- // Oops, nUnicodePrev is not in the cmap! Adjust so we get a reasonable
-@@ -1069,36 +1070,37 @@ unsigned int CmapSubtable4NextCodepoint(
- return nUnicodePrev + 1;
- }
-
- // Otherwise the next codepoint is the first one in the next range.
- // There is guaranteed to be a next range because there must be one that
- // ends with 0xFFFF.
- if (pRangeKey)
- *pRangeKey = iRange + 1;
-- return be::peek<uint16>(pStartCode + iRange + 1);
-+ return (iRange + 1 >= nRange) ? 0xFFFF : be::peek<uint16>(pStartCode + iRange + 1);
- }
-
- /*----------------------------------------------------------------------------------------------
- Check the Microsoft UCS-4 subtable for expected values.
- ----------------------------------------------------------------------------------------------*/
--bool CheckCmapSubtable12(const void *pCmapSubtable12, size_t table_len /*, unsigned int maxgid*/)
-+bool CheckCmapSubtable12(const void *pCmapSubtable12, const void *pCmapEnd /*, unsigned int maxgid*/)
- {
-+ size_t table_len = (const byte *)pCmapEnd - (const byte *)pCmapSubtable12;
- if (!pCmapSubtable12) return false;
- const Sfnt::CmapSubTable * pTable = reinterpret_cast<const Sfnt::CmapSubTable *>(pCmapSubtable12);
- if (be::swap(pTable->format) != 12)
- return false;
- const Sfnt::CmapSubTableFormat12 * pTable12 = reinterpret_cast<const Sfnt::CmapSubTableFormat12 *>(pCmapSubtable12);
- uint32 length = be::swap(pTable12->length);
- if (length > table_len)
- return false;
- if (length < sizeof(Sfnt::CmapSubTableFormat12))
- return false;
- uint32 num_groups = be::swap(pTable12->num_groups);
-- if (length != (sizeof(Sfnt::CmapSubTableFormat12) + (num_groups - 1) * sizeof(uint32) * 3))
-+ if (num_groups > 0x10000000 || length != (sizeof(Sfnt::CmapSubTableFormat12) + (num_groups - 1) * sizeof(uint32) * 3))
- return false;
- #if 0
- for (unsigned int i = 0; i < num_groups; ++i)
- {
- if (be::swap(pTable12->group[i].end_char_code) - be::swap(pTable12->group[i].start_char_code) + be::swap(pTable12->group[i].start_glyph_id) > maxgid)
- return false;
- if (i > 0 && be::swap(pTable12->group[i].start_char_code) <= be::swap(pTable12->group[i-1].end_char_code))
- return false;
-@@ -1161,17 +1163,17 @@ unsigned int CmapSubtable12NextCodepoint
- *pRangeKey = nRange;
- return 0x10FFFF;
- }
-
- int iRange = (pRangeKey) ? *pRangeKey : 0;
- // Just in case we have a bad key:
- while (iRange > 0 && be::swap(pTable->group[iRange].start_char_code) > nUnicodePrev)
- iRange--;
-- while (be::swap(pTable->group[iRange].end_char_code) < nUnicodePrev)
-+ while (iRange < nRange - 1 && be::swap(pTable->group[iRange].end_char_code) < nUnicodePrev)
- iRange++;
-
- // Now iRange is the range containing nUnicodePrev.
-
- unsigned int nStartCode = be::swap(pTable->group[iRange].start_char_code);
- unsigned int nEndCode = be::swap(pTable->group[iRange].end_char_code);
-
- if (nStartCode > nUnicodePrev)
-diff --git a/gfx/graphite2/src/call_machine.cpp b/gfx/graphite2/src/call_machine.cpp
---- a/gfx/graphite2/src/call_machine.cpp
-+++ b/gfx/graphite2/src/call_machine.cpp
-@@ -67,32 +67,34 @@ using namespace vm;
- struct regbank {
- slotref is;
- slotref * map;
- SlotMap & smap;
- slotref * const map_base;
- const instr * & ip;
- uint8 direction;
- int8 flags;
-+ Machine::status_t & status;
- };
-
- typedef bool (* ip_t)(registers);
-
- // Pull in the opcode definitions
- // We pull these into a private namespace so these otherwise common names dont
- // pollute the toplevel namespace.
- namespace {
- #define smap reg.smap
- #define seg smap.segment
- #define is reg.is
- #define ip reg.ip
- #define map reg.map
- #define mapb reg.map_base
- #define flags reg.flags
- #define dir reg.direction
-+#define status reg.status
-
- #include "inc/opcodes.h"
-
- #undef smap
- #undef seg
- #undef is
- #undef ip
- #undef map
-@@ -108,17 +110,17 @@ Machine::stack_t Machine::run(const ins
- {
- assert(program != 0);
-
- // Declare virtual machine registers
- const instr * ip = program-1;
- const byte * dp = data;
- stack_t * sp = _stack + Machine::STACK_GUARD,
- * const sb = sp;
-- regbank reg = {*map, map, _map, _map.begin()+_map.context(), ip, _map.dir(), 0};
-+ regbank reg = {*map, map, _map, _map.begin()+_map.context(), ip, _map.dir(), 0, _status};
-
- // Run the program
- while ((reinterpret_cast<ip_t>(*++ip))(dp, sp, sb, reg)) {}
- const stack_t ret = sp == _stack+STACK_GUARD+1 ? *sp-- : 0;
-
- check_final_stack(sp);
- map = reg.map;
- *map = reg.is;
-diff --git a/gfx/graphite2/src/direct_machine.cpp b/gfx/graphite2/src/direct_machine.cpp
---- a/gfx/graphite2/src/direct_machine.cpp
-+++ b/gfx/graphite2/src/direct_machine.cpp
-@@ -57,36 +57,37 @@ using namespace vm;
- namespace {
-
- const void * direct_run(const bool get_table_mode,
- const instr * program,
- const byte * data,
- Machine::stack_t * stack,
- slotref * & __map,
- uint8 _dir,
-+ Machine::status_t & status,
- SlotMap * __smap=0)
- {
- // We need to define and return to opcode table from within this function
- // other inorder to take the addresses of the instruction bodies.
- #include "inc/opcode_table.h"
- if (get_table_mode)
- return opcode_table;
-
- // Declare virtual machine registers
-- const instr * ip = program;
-- const byte * dp = data;
-- Machine::stack_t * sp = stack + Machine::STACK_GUARD,
-- * const sb = sp;
-- SlotMap & smap = *__smap;
-- Segment & seg = smap.segment;
-- slotref is = *__map,
-- * map = __map,
-- * const mapb = smap.begin()+smap.context();
-- uint8 dir = _dir;
-- int8 flags = 0;
-+ const instr * ip = program;
-+ const byte * dp = data;
-+ Machine::stack_t * sp = stack + Machine::STACK_GUARD,
-+ * const sb = sp;
-+ SlotMap & smap = *__smap;
-+ Segment & seg = smap.segment;
-+ slotref is = *__map,
-+ * map = __map,
-+ * const mapb = smap.begin()+smap.context();
-+ uint8 dir = _dir;
-+ int8 flags = 0;
-
- // start the program
- goto **ip;
-
- // Pull in the opcode definitions
- #include "inc/opcodes.h"
-
- end:
-@@ -95,25 +96,26 @@ const void * direct_run(const bool
- return sp;
- }
-
- }
-
- const opcode_t * Machine::getOpcodeTable() throw()
- {
- slotref * dummy;
-- return static_cast<const opcode_t *>(direct_run(true, 0, 0, 0, dummy, 0));
-+ Machine::status_t dumstat = Machine::finished;
-+ return static_cast<const opcode_t *>(direct_run(true, 0, 0, 0, dummy, 0, dumstat));
- }
-
-
- Machine::stack_t Machine::run(const instr * program,
- const byte * data,
- slotref * & is)
- {
- assert(program != 0);
-
- const stack_t *sp = static_cast<const stack_t *>(
-- direct_run(false, program, data, _stack, is, _map.dir(), &_map));
-+ direct_run(false, program, data, _stack, is, _map.dir(), _status, &_map));
- const stack_t ret = sp == _stack+STACK_GUARD+1 ? *sp-- : 0;
- check_final_stack(sp);
- return ret;
- }
-
-diff --git a/gfx/graphite2/src/inc/Code.h b/gfx/graphite2/src/inc/Code.h
---- a/gfx/graphite2/src/inc/Code.h
-+++ b/gfx/graphite2/src/inc/Code.h
-@@ -109,17 +109,17 @@ public:
- int32 run(Machine &m, slotref * & map) const;
-
- CLASS_NEW_DELETE;
- };
-
- inline
- size_t Machine::Code::estimateCodeDataOut(size_t n_bc)
- {
-- return n_bc * (sizeof(instr)+sizeof(byte));
-+ return (n_bc + 1) * (sizeof(instr)+sizeof(byte));
- }
-
-
- inline Machine::Code::Code() throw()
- : _code(0), _data(0), _data_size(0), _instr_count(0), _max_ref(0),
- _status(loaded), _constraint(false), _modify(false), _delete(false),
- _own(false)
- {
-diff --git a/gfx/graphite2/src/inc/Machine.h b/gfx/graphite2/src/inc/Machine.h
---- a/gfx/graphite2/src/inc/Machine.h
-+++ b/gfx/graphite2/src/inc/Machine.h
-@@ -135,17 +135,18 @@ public:
-
- class Code;
-
- enum status_t {
- finished = 0,
- stack_underflow,
- stack_not_empty,
- stack_overflow,
-- slot_offset_out_bounds
-+ slot_offset_out_bounds,
-+ died_early
- };
-
- Machine(SlotMap &) throw();
- static const opcode_t * getOpcodeTable() throw();
-
- CLASS_NEW_DELETE;
-
- SlotMap & slotMap() const throw();
-diff --git a/gfx/graphite2/src/inc/TtfUtil.h b/gfx/graphite2/src/inc/TtfUtil.h
---- a/gfx/graphite2/src/inc/TtfUtil.h
-+++ b/gfx/graphite2/src/inc/TtfUtil.h
-@@ -132,21 +132,21 @@ public:
- int GetLangsForNames(const void * pName, int nPlatformId, int nEncodingId,
- int *nameIdList, int cNameIds, short *langIdList);
- void SwapWString(void * pWStr, size_t nSize = 0); // throw (std::invalid_argument);
- #endif
-
- ////////////////////////////////// cmap lookup tools
- const void * FindCmapSubtable(const void * pCmap, int nPlatformId = 3,
- int nEncodingId = 1, size_t length = 0);
-- bool CheckCmapSubtable4(const void * pCmap31, size_t table_len /*, unsigned int maxgid*/);
-+ bool CheckCmapSubtable4(const void * pCmap31, const void * pCmapEnd /*, unsigned int maxgid*/);
- gid16 CmapSubtable4Lookup(const void * pCmapSubtabel4, unsigned int nUnicodeId, int rangeKey = 0);
- unsigned int CmapSubtable4NextCodepoint(const void *pCmap31, unsigned int nUnicodeId,
- int * pRangeKey = 0);
-- bool CheckCmapSubtable12(const void *pCmap310, size_t table_len /*, unsigned int maxgid*/);
-+ bool CheckCmapSubtable12(const void *pCmap310, const void * pCmapEnd /*, unsigned int maxgid*/);
- gid16 CmapSubtable12Lookup(const void * pCmap310, unsigned int uUnicodeId, int rangeKey = 0);
- unsigned int CmapSubtable12NextCodepoint(const void *pCmap310, unsigned int nUnicodeId,
- int * pRangeKey = 0);
-
- ///////////////////////////////// horizontal metric data for a glyph
- bool HorMetrics(gid16 nGlyphId, const void * pHmtx, size_t lHmtxSize,
- const void * pHhea, int & nLsb, unsigned int & nAdvWid);
-
-diff --git a/gfx/graphite2/src/inc/opcodes.h b/gfx/graphite2/src/inc/opcodes.h
---- a/gfx/graphite2/src/inc/opcodes.h
-+++ b/gfx/graphite2/src/inc/opcodes.h
-@@ -71,17 +71,17 @@ of the License or (at your option) any l
- #define use_params(n) dp += n
-
- #define declare_params(n) const byte * param = dp; \
- use_params(n);
-
- #define push(n) { *++sp = n; }
- #define pop() (*sp--)
- #define slotat(x) (map[(x)])
--#define DIE { is=seg.last(); EXIT(1); }
-+#define DIE { is=seg.last(); status = Machine::died_early; EXIT(1); }
- #define POSITIONED 1
-
- STARTOP(nop)
- do {} while (0);
- ENDOP
-
- STARTOP(push_byte)
- declare_params(1);
-@@ -387,30 +387,30 @@ STARTOP(attr_set)
- ENDOP
-
- STARTOP(attr_add)
- declare_params(1);
- const attrCode slat = attrCode(uint8(*param));
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), seg.currdir());
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, 0);
- is->setAttr(&seg, slat, 0, val + res, smap);
- ENDOP
-
- STARTOP(attr_sub)
- declare_params(1);
- const attrCode slat = attrCode(uint8(*param));
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), seg.currdir());
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, 0);
- is->setAttr(&seg, slat, 0, res - val, smap);
- ENDOP
-
- STARTOP(attr_set_slot)
- declare_params(1);
-@@ -429,17 +429,17 @@ STARTOP(iattr_set_slot)
- ENDOP
-
- STARTOP(push_slot_attr)
- declare_params(2);
- const attrCode slat = attrCode(uint8(param[0]));
- const int slot_ref = int8(param[1]);
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), seg.currdir());
- flags |= POSITIONED;
- }
- slotref slot = slotat(slot_ref);
- if (slot)
- {
- int res = slot->getAttr(&seg, slat, 0);
- push(res);
- }
-@@ -505,17 +505,17 @@ ENDOP
-
- STARTOP(push_islot_attr)
- declare_params(3);
- const attrCode slat = attrCode(uint8(param[0]));
- const int slot_ref = int8(param[1]),
- idx = uint8(param[2]);
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), seg.currdir());
- flags |= POSITIONED;
- }
- slotref slot = slotat(slot_ref);
- if (slot)
- {
- int res = slot->getAttr(&seg, slat, idx);
- push(res);
- }
-
diff --git a/gnu/packages/patches/icecat-update-graphite2.patch b/gnu/packages/patches/icecat-update-graphite2.patch
deleted file mode 100644
index af2c47bef7..0000000000
--- a/gnu/packages/patches/icecat-update-graphite2.patch
+++ /dev/null
@@ -1,9988 +0,0 @@
-Copied from upstream:
-https://hg.mozilla.org/releases/mozilla-esr38/raw-rev/ed4d2ce6046b
-
-# HG changeset patch
-# User Jonathan Kew <jkew@mozilla.com>
-# Date 1455126706 0
-# Node ID ed4d2ce6046b20287fd13c548dd3982fe1a24875
-# Parent 78d3632feb7b6f6046025352630bd4f5365f3106
-Bug 1246093 - Update graphite2 library to latest release on esr38. r=me,a=sledru+lizzard
-
-diff --git a/gfx/graphite2/COPYING b/gfx/graphite2/COPYING
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/COPYING
-@@ -0,0 +1,26 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+ Alternatively, you may use this library under the terms of the Mozilla
-+ Public License (http://mozilla.org/MPL) or under the GNU General Public
-+ License, as published by the Free Sofware Foundation; either version
-+ 2 of the license or (at your option) any later version.
-+*/
-diff --git a/gfx/graphite2/LICENSE b/gfx/graphite2/LICENSE
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/LICENSE
-@@ -0,0 +1,510 @@
-+
-+ GNU LESSER GENERAL PUBLIC LICENSE
-+ Version 2.1, February 1999
-+
-+ Copyright (C) 1991, 1999 Free Software Foundation, Inc.
-+ 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
-+ Everyone is permitted to copy and distribute verbatim copies
-+ of this license document, but changing it is not allowed.
-+
-+[This is the first released version of the Lesser GPL. It also counts
-+ as the successor of the GNU Library Public License, version 2, hence
-+ the version number 2.1.]
-+
-+ Preamble
-+
-+ The licenses for most software are designed to take away your
-+freedom to share and change it. By contrast, the GNU General Public
-+Licenses are intended to guarantee your freedom to share and change
-+free software--to make sure the software is free for all its users.
-+
-+ This license, the Lesser General Public License, applies to some
-+specially designated software packages--typically libraries--of the
-+Free Software Foundation and other authors who decide to use it. You
-+can use it too, but we suggest you first think carefully about whether
-+this license or the ordinary General Public License is the better
-+strategy to use in any particular case, based on the explanations
-+below.
-+
-+ When we speak of free software, we are referring to freedom of use,
-+not price. Our General Public Licenses are designed to make sure that
-+you have the freedom to distribute copies of free software (and charge
-+for this service if you wish); that you receive source code or can get
-+it if you want it; that you can change the software and use pieces of
-+it in new free programs; and that you are informed that you can do
-+these things.
-+
-+ To protect your rights, we need to make restrictions that forbid
-+distributors to deny you these rights or to ask you to surrender these
-+rights. These restrictions translate to certain responsibilities for
-+you if you distribute copies of the library or if you modify it.
-+
-+ For example, if you distribute copies of the library, whether gratis
-+or for a fee, you must give the recipients all the rights that we gave
-+you. You must make sure that they, too, receive or can get the source
-+code. If you link other code with the library, you must provide
-+complete object files to the recipients, so that they can relink them
-+with the library after making changes to the library and recompiling
-+it. And you must show them these terms so they know their rights.
-+
-+ We protect your rights with a two-step method: (1) we copyright the
-+library, and (2) we offer you this license, which gives you legal
-+permission to copy, distribute and/or modify the library.
-+
-+ To protect each distributor, we want to make it very clear that
-+there is no warranty for the free library. Also, if the library is
-+modified by someone else and passed on, the recipients should know
-+that what they have is not the original version, so that the original
-+author's reputation will not be affected by problems that might be
-+introduced by others.
-+
-+ Finally, software patents pose a constant threat to the existence of
-+any free program. We wish to make sure that a company cannot
-+effectively restrict the users of a free program by obtaining a
-+restrictive license from a patent holder. Therefore, we insist that
-+any patent license obtained for a version of the library must be
-+consistent with the full freedom of use specified in this license.
-+
-+ Most GNU software, including some libraries, is covered by the
-+ordinary GNU General Public License. This license, the GNU Lesser
-+General Public License, applies to certain designated libraries, and
-+is quite different from the ordinary General Public License. We use
-+this license for certain libraries in order to permit linking those
-+libraries into non-free programs.
-+
-+ When a program is linked with a library, whether statically or using
-+a shared library, the combination of the two is legally speaking a
-+combined work, a derivative of the original library. The ordinary
-+General Public License therefore permits such linking only if the
-+entire combination fits its criteria of freedom. The Lesser General
-+Public License permits more lax criteria for linking other code with
-+the library.
-+
-+ We call this license the "Lesser" General Public License because it
-+does Less to protect the user's freedom than the ordinary General
-+Public License. It also provides other free software developers Less
-+of an advantage over competing non-free programs. These disadvantages
-+are the reason we use the ordinary General Public License for many
-+libraries. However, the Lesser license provides advantages in certain
-+special circumstances.
-+
-+ For example, on rare occasions, there may be a special need to
-+encourage the widest possible use of a certain library, so that it
-+becomes a de-facto standard. To achieve this, non-free programs must
-+be allowed to use the library. A more frequent case is that a free
-+library does the same job as widely used non-free libraries. In this
-+case, there is little to gain by limiting the free library to free
-+software only, so we use the Lesser General Public License.
-+
-+ In other cases, permission to use a particular library in non-free
-+programs enables a greater number of people to use a large body of
-+free software. For example, permission to use the GNU C Library in
-+non-free programs enables many more people to use the whole GNU
-+operating system, as well as its variant, the GNU/Linux operating
-+system.
-+
-+ Although the Lesser General Public License is Less protective of the
-+users' freedom, it does ensure that the user of a program that is
-+linked with the Library has the freedom and the wherewithal to run
-+that program using a modified version of the Library.
-+
-+ The precise terms and conditions for copying, distribution and
-+modification follow. Pay close attention to the difference between a
-+"work based on the library" and a "work that uses the library". The
-+former contains code derived from the library, whereas the latter must
-+be combined with the library in order to run.
-+
-+ GNU LESSER GENERAL PUBLIC LICENSE
-+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
-+
-+ 0. This License Agreement applies to any software library or other
-+program which contains a notice placed by the copyright holder or
-+other authorized party saying it may be distributed under the terms of
-+this Lesser General Public License (also called "this License").
-+Each licensee is addressed as "you".
-+
-+ A "library" means a collection of software functions and/or data
-+prepared so as to be conveniently linked with application programs
-+(which use some of those functions and data) to form executables.
-+
-+ The "Library", below, refers to any such software library or work
-+which has been distributed under these terms. A "work based on the
-+Library" means either the Library or any derivative work under
-+copyright law: that is to say, a work containing the Library or a
-+portion of it, either verbatim or with modifications and/or translated
-+straightforwardly into another language. (Hereinafter, translation is
-+included without limitation in the term "modification".)
-+
-+ "Source code" for a work means the preferred form of the work for
-+making modifications to it. For a library, complete source code means
-+all the source code for all modules it contains, plus any associated
-+interface definition files, plus the scripts used to control
-+compilation and installation of the library.
-+
-+ Activities other than copying, distribution and modification are not
-+covered by this License; they are outside its scope. The act of
-+running a program using the Library is not restricted, and output from
-+such a program is covered only if its contents constitute a work based
-+on the Library (independent of the use of the Library in a tool for
-+writing it). Whether that is true depends on what the Library does
-+and what the program that uses the Library does.
-+
-+ 1. You may copy and distribute verbatim copies of the Library's
-+complete source code as you receive it, in any medium, provided that
-+you conspicuously and appropriately publish on each copy an
-+appropriate copyright notice and disclaimer of warranty; keep intact
-+all the notices that refer to this License and to the absence of any
-+warranty; and distribute a copy of this License along with the
-+Library.
-+
-+ You may charge a fee for the physical act of transferring a copy,
-+and you may at your option offer warranty protection in exchange for a
-+fee.
-+
-+ 2. You may modify your copy or copies of the Library or any portion
-+of it, thus forming a work based on the Library, and copy and
-+distribute such modifications or work under the terms of Section 1
-+above, provided that you also meet all of these conditions:
-+
-+ a) The modified work must itself be a software library.
-+
-+ b) You must cause the files modified to carry prominent notices
-+ stating that you changed the files and the date of any change.
-+
-+ c) You must cause the whole of the work to be licensed at no
-+ charge to all third parties under the terms of this License.
-+
-+ d) If a facility in the modified Library refers to a function or a
-+ table of data to be supplied by an application program that uses
-+ the facility, other than as an argument passed when the facility
-+ is invoked, then you must make a good faith effort to ensure that,
-+ in the event an application does not supply such function or
-+ table, the facility still operates, and performs whatever part of
-+ its purpose remains meaningful.
-+
-+ (For example, a function in a library to compute square roots has
-+ a purpose that is entirely well-defined independent of the
-+ application. Therefore, Subsection 2d requires that any
-+ application-supplied function or table used by this function must
-+ be optional: if the application does not supply it, the square
-+ root function must still compute square roots.)
-+
-+These requirements apply to the modified work as a whole. If
-+identifiable sections of that work are not derived from the Library,
-+and can be reasonably considered independent and separate works in
-+themselves, then this License, and its terms, do not apply to those
-+sections when you distribute them as separate works. But when you
-+distribute the same sections as part of a whole which is a work based
-+on the Library, the distribution of the whole must be on the terms of
-+this License, whose permissions for other licensees extend to the
-+entire whole, and thus to each and every part regardless of who wrote
-+it.
-+
-+Thus, it is not the intent of this section to claim rights or contest
-+your rights to work written entirely by you; rather, the intent is to
-+exercise the right to control the distribution of derivative or
-+collective works based on the Library.
-+
-+In addition, mere aggregation of another work not based on the Library
-+with the Library (or with a work based on the Library) on a volume of
-+a storage or distribution medium does not bring the other work under
-+the scope of this License.
-+
-+ 3. You may opt to apply the terms of the ordinary GNU General Public
-+License instead of this License to a given copy of the Library. To do
-+this, you must alter all the notices that refer to this License, so
-+that they refer to the ordinary GNU General Public License, version 2,
-+instead of to this License. (If a newer version than version 2 of the
-+ordinary GNU General Public License has appeared, then you can specify
-+that version instead if you wish.) Do not make any other change in
-+these notices.
-+
-+ Once this change is made in a given copy, it is irreversible for
-+that copy, so the ordinary GNU General Public License applies to all
-+subsequent copies and derivative works made from that copy.
-+
-+ This option is useful when you wish to copy part of the code of
-+the Library into a program that is not a library.
-+
-+ 4. You may copy and distribute the Library (or a portion or
-+derivative of it, under Section 2) in object code or executable form
-+under the terms of Sections 1 and 2 above provided that you accompany
-+it with the complete corresponding machine-readable source code, which
-+must be distributed under the terms of Sections 1 and 2 above on a
-+medium customarily used for software interchange.
-+
-+ If distribution of object code is made by offering access to copy
-+from a designated place, then offering equivalent access to copy the
-+source code from the same place satisfies the requirement to
-+distribute the source code, even though third parties are not
-+compelled to copy the source along with the object code.
-+
-+ 5. A program that contains no derivative of any portion of the
-+Library, but is designed to work with the Library by being compiled or
-+linked with it, is called a "work that uses the Library". Such a
-+work, in isolation, is not a derivative work of the Library, and
-+therefore falls outside the scope of this License.
-+
-+ However, linking a "work that uses the Library" with the Library
-+creates an executable that is a derivative of the Library (because it
-+contains portions of the Library), rather than a "work that uses the
-+library". The executable is therefore covered by this License.
-+Section 6 states terms for distribution of such executables.
-+
-+ When a "work that uses the Library" uses material from a header file
-+that is part of the Library, the object code for the work may be a
-+derivative work of the Library even though the source code is not.
-+Whether this is true is especially significant if the work can be
-+linked without the Library, or if the work is itself a library. The
-+threshold for this to be true is not precisely defined by law.
-+
-+ If such an object file uses only numerical parameters, data
-+structure layouts and accessors, and small macros and small inline
-+functions (ten lines or less in length), then the use of the object
-+file is unrestricted, regardless of whether it is legally a derivative
-+work. (Executables containing this object code plus portions of the
-+Library will still fall under Section 6.)
-+
-+ Otherwise, if the work is a derivative of the Library, you may
-+distribute the object code for the work under the terms of Section 6.
-+Any executables containing that work also fall under Section 6,
-+whether or not they are linked directly with the Library itself.
-+
-+ 6. As an exception to the Sections above, you may also combine or
-+link a "work that uses the Library" with the Library to produce a
-+work containing portions of the Library, and distribute that work
-+under terms of your choice, provided that the terms permit
-+modification of the work for the customer's own use and reverse
-+engineering for debugging such modifications.
-+
-+ You must give prominent notice with each copy of the work that the
-+Library is used in it and that the Library and its use are covered by
-+this License. You must supply a copy of this License. If the work
-+during execution displays copyright notices, you must include the
-+copyright notice for the Library among them, as well as a reference
-+directing the user to the copy of this License. Also, you must do one
-+of these things:
-+
-+ a) Accompany the work with the complete corresponding
-+ machine-readable source code for the Library including whatever
-+ changes were used in the work (which must be distributed under
-+ Sections 1 and 2 above); and, if the work is an executable linked
-+ with the Library, with the complete machine-readable "work that
-+ uses the Library", as object code and/or source code, so that the
-+ user can modify the Library and then relink to produce a modified
-+ executable containing the modified Library. (It is understood
-+ that the user who changes the contents of definitions files in the
-+ Library will not necessarily be able to recompile the application
-+ to use the modified definitions.)
-+
-+ b) Use a suitable shared library mechanism for linking with the
-+ Library. A suitable mechanism is one that (1) uses at run time a
-+ copy of the library already present on the user's computer system,
-+ rather than copying library functions into the executable, and (2)
-+ will operate properly with a modified version of the library, if
-+ the user installs one, as long as the modified version is
-+ interface-compatible with the version that the work was made with.
-+
-+ c) Accompany the work with a written offer, valid for at least
-+ three years, to give the same user the materials specified in
-+ Subsection 6a, above, for a charge no more than the cost of
-+ performing this distribution.
-+
-+ d) If distribution of the work is made by offering access to copy
-+ from a designated place, offer equivalent access to copy the above
-+ specified materials from the same place.
-+
-+ e) Verify that the user has already received a copy of these
-+ materials or that you have already sent this user a copy.
-+
-+ For an executable, the required form of the "work that uses the
-+Library" must include any data and utility programs needed for
-+reproducing the executable from it. However, as a special exception,
-+the materials to be distributed need not include anything that is
-+normally distributed (in either source or binary form) with the major
-+components (compiler, kernel, and so on) of the operating system on
-+which the executable runs, unless that component itself accompanies
-+the executable.
-+
-+ It may happen that this requirement contradicts the license
-+restrictions of other proprietary libraries that do not normally
-+accompany the operating system. Such a contradiction means you cannot
-+use both them and the Library together in an executable that you
-+distribute.
-+
-+ 7. You may place library facilities that are a work based on the
-+Library side-by-side in a single library together with other library
-+facilities not covered by this License, and distribute such a combined
-+library, provided that the separate distribution of the work based on
-+the Library and of the other library facilities is otherwise
-+permitted, and provided that you do these two things:
-+
-+ a) Accompany the combined library with a copy of the same work
-+ based on the Library, uncombined with any other library
-+ facilities. This must be distributed under the terms of the
-+ Sections above.
-+
-+ b) Give prominent notice with the combined library of the fact
-+ that part of it is a work based on the Library, and explaining
-+ where to find the accompanying uncombined form of the same work.
-+
-+ 8. You may not copy, modify, sublicense, link with, or distribute
-+the Library except as expressly provided under this License. Any
-+attempt otherwise to copy, modify, sublicense, link with, or
-+distribute the Library is void, and will automatically terminate your
-+rights under this License. However, parties who have received copies,
-+or rights, from you under this License will not have their licenses
-+terminated so long as such parties remain in full compliance.
-+
-+ 9. You are not required to accept this License, since you have not
-+signed it. However, nothing else grants you permission to modify or
-+distribute the Library or its derivative works. These actions are
-+prohibited by law if you do not accept this License. Therefore, by
-+modifying or distributing the Library (or any work based on the
-+Library), you indicate your acceptance of this License to do so, and
-+all its terms and conditions for copying, distributing or modifying
-+the Library or works based on it.
-+
-+ 10. Each time you redistribute the Library (or any work based on the
-+Library), the recipient automatically receives a license from the
-+original licensor to copy, distribute, link with or modify the Library
-+subject to these terms and conditions. You may not impose any further
-+restrictions on the recipients' exercise of the rights granted herein.
-+You are not responsible for enforcing compliance by third parties with
-+this License.
-+
-+ 11. If, as a consequence of a court judgment or allegation of patent
-+infringement or for any other reason (not limited to patent issues),
-+conditions are imposed on you (whether by court order, agreement or
-+otherwise) that contradict the conditions of this License, they do not
-+excuse you from the conditions of this License. If you cannot
-+distribute so as to satisfy simultaneously your obligations under this
-+License and any other pertinent obligations, then as a consequence you
-+may not distribute the Library at all. For example, if a patent
-+license would not permit royalty-free redistribution of the Library by
-+all those who receive copies directly or indirectly through you, then
-+the only way you could satisfy both it and this License would be to
-+refrain entirely from distribution of the Library.
-+
-+If any portion of this section is held invalid or unenforceable under
-+any particular circumstance, the balance of the section is intended to
-+apply, and the section as a whole is intended to apply in other
-+circumstances.
-+
-+It is not the purpose of this section to induce you to infringe any
-+patents or other property right claims or to contest validity of any
-+such claims; this section has the sole purpose of protecting the
-+integrity of the free software distribution system which is
-+implemented by public license practices. Many people have made
-+generous contributions to the wide range of software distributed
-+through that system in reliance on consistent application of that
-+system; it is up to the author/donor to decide if he or she is willing
-+to distribute software through any other system and a licensee cannot
-+impose that choice.
-+
-+This section is intended to make thoroughly clear what is believed to
-+be a consequence of the rest of this License.
-+
-+ 12. If the distribution and/or use of the Library is restricted in
-+certain countries either by patents or by copyrighted interfaces, the
-+original copyright holder who places the Library under this License
-+may add an explicit geographical distribution limitation excluding those
-+countries, so that distribution is permitted only in or among
-+countries not thus excluded. In such case, this License incorporates
-+the limitation as if written in the body of this License.
-+
-+ 13. The Free Software Foundation may publish revised and/or new
-+versions of the Lesser General Public License from time to time.
-+Such new versions will be similar in spirit to the present version,
-+but may differ in detail to address new problems or concerns.
-+
-+Each version is given a distinguishing version number. If the Library
-+specifies a version number of this License which applies to it and
-+"any later version", you have the option of following the terms and
-+conditions either of that version or of any later version published by
-+the Free Software Foundation. If the Library does not specify a
-+license version number, you may choose any version ever published by
-+the Free Software Foundation.
-+
-+ 14. If you wish to incorporate parts of the Library into other free
-+programs whose distribution conditions are incompatible with these,
-+write to the author to ask for permission. For software which is
-+copyrighted by the Free Software Foundation, write to the Free
-+Software Foundation; we sometimes make exceptions for this. Our
-+decision will be guided by the two goals of preserving the free status
-+of all derivatives of our free software and of promoting the sharing
-+and reuse of software generally.
-+
-+ NO WARRANTY
-+
-+ 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO
-+WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
-+EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
-+OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY
-+KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE
-+IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
-+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE
-+LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME
-+THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
-+
-+ 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
-+WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
-+AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU
-+FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR
-+CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE
-+LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING
-+RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A
-+FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF
-+SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
-+DAMAGES.
-+
-+ END OF TERMS AND CONDITIONS
-+
-+ How to Apply These Terms to Your New Libraries
-+
-+ If you develop a new library, and you want it to be of the greatest
-+possible use to the public, we recommend making it free software that
-+everyone can redistribute and change. You can do so by permitting
-+redistribution under these terms (or, alternatively, under the terms
-+of the ordinary General Public License).
-+
-+ To apply these terms, attach the following notices to the library.
-+It is safest to attach them to the start of each source file to most
-+effectively convey the exclusion of warranty; and each file should
-+have at least the "copyright" line and a pointer to where the full
-+notice is found.
-+
-+
-+ <one line to give the library's name and a brief idea of what it does.>
-+ Copyright (C) <year> <name of author>
-+
-+ This library is free software; you can redistribute it and/or
-+ modify it under the terms of the GNU Lesser General Public
-+ License as published by the Free Software Foundation; either
-+ version 2.1 of the License, or (at your option) any later version.
-+
-+ This library is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should have received a copy of the GNU Lesser General Public
-+ License along with this library; if not, write to the Free Software
-+ Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
-+
-+Also add information on how to contact you by electronic and paper mail.
-+
-+You should also get your employer (if you work as a programmer) or
-+your school, if any, to sign a "copyright disclaimer" for the library,
-+if necessary. Here is a sample; alter the names:
-+
-+ Yoyodyne, Inc., hereby disclaims all copyright interest in the
-+ library `Frob' (a library for tweaking knobs) written by James
-+ Random Hacker.
-+
-+ <signature of Ty Coon>, 1 April 1990
-+ Ty Coon, President of Vice
-+
-+That's all there is to it!
-+
-+
-diff --git a/gfx/graphite2/README.md b/gfx/graphite2/README.md
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/README.md
-@@ -0,0 +1,32 @@
-+# Graphite engine
-+
-+## What is Graphite?
-+
-+Graphite is a system that can be used to create “smart fonts” capable of displaying writing systems with various complex behaviors. A smart font contains not only letter shapes but also additional instructions indicating how to combine and position the letters in complex ways.
-+
-+Graphite was primarily developed to provide the flexibility needed for minority languages which often need to be written according to slightly different rules than well-known languages that use the same script.
-+
-+Examples of complex script behaviors Graphite can handle include:
-+
-+* contextual shaping
-+* ligatures
-+* reordering
-+* split glyphs
-+* bidirectionality
-+* stacking diacritics
-+* complex positioning
-+* shape aware kerning
-+* automatic diacritic collision avoidance
-+
-+See [examples of scripts with complex rendering](http://scripts.sil.org/CmplxRndExamples).
-+
-+## Graphite system overview
-+The Graphite system consists of:
-+
-+* A rule-based programming language [Graphite Description Language](http://scripts.sil.org/cms/scripts/page.php?site_id=projects&item_id=graphite_devFont#gdl) (GDL) that can be used to describe the behavior of a writing system
-+* A compiler for that language
-+* A rendering engine that can serve as the layout component of a text-processing application
-+
-+Graphite renders TrueType fonts that have been extended by means of compiling a GDL program.
-+
-+Further technical information is available on the [Graphite technical overview](http://scripts.sil.org/cms/scripts/page.php?site_id=projects&item_id=graphite_techAbout) page.
-diff --git a/gfx/graphite2/README.mozilla b/gfx/graphite2/README.mozilla
---- a/gfx/graphite2/README.mozilla
-+++ b/gfx/graphite2/README.mozilla
-@@ -1,6 +1,7 @@
--This directory contains the Graphite2 library from http://hg.palaso.org/graphitedev
--
--Current version derived from upstream changeset 1efd96aeade9
--
-+This directory contains the Graphite2 library release 1.3.5 from
-+https://github.com/silnrsi/graphite/releases/download/1.3.5/graphite2-minimal-1.3.5.tgz
- See gfx/graphite2/moz-gr-update.sh for update procedure.
-
-+Also includes two post-1.3.5 fixes:
-+a8b3ac2aed0eb132cd80efe7de88f8153e73c829
-+e569e28d83491fedb31b9220493f3c07f6ec6d80
-diff --git a/gfx/graphite2/include/graphite2/Font.h b/gfx/graphite2/include/graphite2/Font.h
---- a/gfx/graphite2/include/graphite2/Font.h
-+++ b/gfx/graphite2/include/graphite2/Font.h
-@@ -24,18 +24,18 @@
- General Public License, as published by the Free Software Foundation,
- either version 2 of the License or (at your option) any later version.
- */
- #pragma once
-
- #include "graphite2/Types.h"
-
- #define GR2_VERSION_MAJOR 1
--#define GR2_VERSION_MINOR 2
--#define GR2_VERSION_BUGFIX 4
-+#define GR2_VERSION_MINOR 3
-+#define GR2_VERSION_BUGFIX 5
-
- #ifdef __cplusplus
- extern "C"
- {
- #endif
-
- typedef struct gr_face gr_face;
- typedef struct gr_font gr_font;
-diff --git a/gfx/graphite2/include/graphite2/Segment.h b/gfx/graphite2/include/graphite2/Segment.h
---- a/gfx/graphite2/include/graphite2/Segment.h
-+++ b/gfx/graphite2/include/graphite2/Segment.h
-@@ -115,35 +115,68 @@ enum gr_attrCode {
- gr_slatJStretch,
- /// Amount this slot can shrink (not implemented)
- gr_slatJShrink,
- /// Granularity by which this slot can stretch or shrink (not implemented)
- gr_slatJStep,
- /// Justification weight for this glyph (not implemented)
- gr_slatJWeight,
- /// Amount this slot mush shrink or stretch in design units
-- gr_slatJWidth,
-+ gr_slatJWidth = 29,
- /// SubSegment split point
- gr_slatSegSplit = gr_slatJStretch + 29,
- /// User defined attribute, see subattr for user attr number
- gr_slatUserDefn,
- /// Bidi level
-- gr_slatBidiLevel,
-+ gr_slatBidiLevel = 56,
-+ /// Collision flags
-+ gr_slatColFlags,
-+ /// Collision constraint rectangle left (bl.x)
-+ gr_slatColLimitblx,
-+ /// Collision constraint rectangle lower (bl.y)
-+ gr_slatColLimitbly,
-+ /// Collision constraint rectangle right (tr.x)
-+ gr_slatColLimittrx,
-+ /// Collision constraint rectangle upper (tr.y)
-+ gr_slatColLimittry,
-+ /// Collision shift x
-+ gr_slatColShiftx,
-+ /// Collision shift y
-+ gr_slatColShifty,
-+ /// Collision margin
-+ gr_slatColMargin,
-+ /// Margin cost weight
-+ gr_slatColMarginWt,
-+ // Additional glyph that excludes movement near this one:
-+ gr_slatColExclGlyph,
-+ gr_slatColExclOffx,
-+ gr_slatColExclOffy,
-+ // Collision sequence enforcing attributes:
-+ gr_slatSeqClass,
-+ gr_slatSeqProxClass,
-+ gr_slatSeqOrder,
-+ gr_slatSeqAboveXoff,
-+ gr_slatSeqAboveWt,
-+ gr_slatSeqBelowXlim,
-+ gr_slatSeqBelowWt,
-+ gr_slatSeqValignHt,
-+ gr_slatSeqValignWt,
-
- /// not implemented
- gr_slatMax,
- /// not implemented
- gr_slatNoEffect = gr_slatMax + 1
- };
-
- enum gr_bidirtl {
- /// Underlying paragraph direction is RTL
- gr_rtl = 1,
- /// Set this to not run the bidi pass internally, even if the font asks for it.
-- /// This presumes that the segment is in a single direction.
-+ /// This presumes that the segment is in a single direction. Most of the time
-+ /// this bit should be set unless you know you are passing full paragraphs of text.
- gr_nobidi = 2,
- /// Disable auto mirroring for rtl text
- gr_nomirror = 4
- };
-
- typedef struct gr_char_info gr_char_info;
- typedef struct gr_segment gr_segment;
- typedef struct gr_slot gr_slot;
-diff --git a/gfx/graphite2/include/graphite2/Types.h b/gfx/graphite2/include/graphite2/Types.h
---- a/gfx/graphite2/include/graphite2/Types.h
-+++ b/gfx/graphite2/include/graphite2/Types.h
-@@ -53,17 +53,20 @@ enum gr_encform {
- #else
- #if defined __GNUC__
- #define GR2_API __attribute__((dllimport))
- #else
- #define GR2_API __declspec(dllimport)
- #endif
- #endif
- #define GR2_LOCAL
-+#elif __GNUC__ >= 4
-+ #if defined GRAPHITE2_STATIC
-+ #define GR2_API __attribute__ ((visibility("hidden")))
-+ #else
-+ #define GR2_API __attribute__ ((visibility("default")))
-+ #endif
-+ #define GR2_LOCAL __attribute__ ((visibility("hidden")))
- #else
-- #if __GNUC__ >= 4
-- #define GR2_API __attribute__ ((visibility("default")))
-- #define GR2_LOCAL __attribute__ ((visibility("hidden")))
-- #else
-- #define GR2_API
-- #define GR2_LOCAL
-- #endif
-+ #define GR2_API
-+ #define GR2_LOCAL
- #endif
-+
-diff --git a/gfx/graphite2/moz-gr-update.sh b/gfx/graphite2/moz-gr-update.sh
---- a/gfx/graphite2/moz-gr-update.sh
-+++ b/gfx/graphite2/moz-gr-update.sh
-@@ -1,35 +1,49 @@
- #!/bin/bash
-
- # Script used to update the Graphite2 library in the mozilla source tree
-
- # This script lives in gfx/graphite2, along with the library source,
- # but must be run from the top level of the mozilla-central tree.
-
--# It expects to find a checkout of the graphite2 tree in a directory "graphitedev"
--# alongside the current mozilla tree that is to be updated.
--# Expect error messages from the copy commands if this is not found!
-+# Run as
-+#
-+# ./gfx/graphite2/moz-gr-update.sh RELEASE
-+#
-+# where RELEASE is the graphite2 release to be used, e.g. "1.3.4".
-
--# copy the source and headers
--cp -R ../graphitedev/src/* gfx/graphite2/src
--cp ../graphitedev/include/graphite2/* gfx/graphite2/include/graphite2
-+RELEASE=$1
-
--# record the upstream changeset that was used
--CHANGESET=$(cd ../graphitedev/ && hg log | head -n 1 | cut -d : -f 1,3 | sed -e 's/:/ /')
--echo "This directory contains the Graphite2 library from http://hg.palaso.org/graphitedev\n" > gfx/graphite2/README.mozilla
--echo "Current version derived from upstream" $CHANGESET >> gfx/graphite2/README.mozilla
--echo "\nSee" $0 "for update procedure.\n" >> gfx/graphite2/README.mozilla
-+if [ "x$RELEASE" == "x" ]
-+then
-+ echo "Must provide the version number to be used."
-+ exit 1
-+fi
-+
-+TARBALL="https://github.com/silnrsi/graphite/releases/download/$RELEASE/graphite2-minimal-$RELEASE.tgz"
-+
-+foo=`basename $0`
-+TMPFILE=`mktemp -t ${foo}` || exit 1
-+
-+curl -L "$TARBALL" -o "$TMPFILE"
-+tar -x -z -C gfx/graphite2/ --strip-components 1 -f "$TMPFILE" || exit 1
-+rm "$TMPFILE"
-+
-+echo "This directory contains the Graphite2 library release $RELEASE from" > gfx/graphite2/README.mozilla
-+echo "$TARBALL" >> gfx/graphite2/README.mozilla
-+echo ""
-+echo "See" $0 "for update procedure." >> gfx/graphite2/README.mozilla
-
- # fix up includes because of bug 721839 (cstdio) and bug 803066 (Windows.h)
--find gfx/graphite2/ -name "*.cpp" -exec perl -p -i -e "s/<cstdio>/<stdio.h>/;s/Windows.h/windows.h/;" {} \;
--find gfx/graphite2/ -name "*.h" -exec perl -p -i -e "s/<cstdio>/<stdio.h>/;s/Windows.h/windows.h/;" {} \;
-+#find gfx/graphite2/ -name "*.cpp" -exec perl -p -i -e "s/<cstdio>/<stdio.h>/;s/Windows.h/windows.h/;" {} \;
-+#find gfx/graphite2/ -name "*.h" -exec perl -p -i -e "s/<cstdio>/<stdio.h>/;s/Windows.h/windows.h/;" {} \;
-
- # summarize what's been touched
--echo Updated to $CHANGESET.
-+echo Updated to $RELEASE.
- echo Here is what changed in the gfx/graphite2 directory:
- echo
-
- hg stat gfx/graphite2
-
- echo
- echo If gfx/graphite2/src/files.mk has changed, please make corresponding
- echo changes to gfx/graphite2/src/moz.build
-diff --git a/gfx/graphite2/src/CMakeLists.txt b/gfx/graphite2/src/CMakeLists.txt
---- a/gfx/graphite2/src/CMakeLists.txt
-+++ b/gfx/graphite2/src/CMakeLists.txt
-@@ -69,29 +69,31 @@ add_library(graphite2 SHARED
- ${GRAPHITE2_VM_TYPE}_machine.cpp
- gr_char_info.cpp
- gr_features.cpp
- gr_face.cpp
- gr_font.cpp
- gr_logging.cpp
- gr_segment.cpp
- gr_slot.cpp
-- Bidi.cpp
- CachedFace.cpp
- CmapCache.cpp
- Code.cpp
-+ Collider.cpp
-+ Decompressor.cpp
- Face.cpp
- FeatureMap.cpp
- Font.cpp
- GlyphFace.cpp
- GlyphCache.cpp
-+ Intervals.cpp
- Justifier.cpp
- NameTable.cpp
- Pass.cpp
-- Rule.cpp
-+ Position.cpp
- Segment.cpp
- Silf.cpp
- Slot.cpp
- Sparse.cpp
- TtfUtil.cpp
- UtfCodec.cpp
- ${FILEFACE}
- ${SEGCACHE}
-@@ -99,27 +101,28 @@ add_library(graphite2 SHARED
-
- set_target_properties(graphite2 PROPERTIES PUBLIC_HEADER "${GRAPHITE_HEADERS}"
- SOVERSION ${GRAPHITE_SO_VERSION}
- VERSION ${GRAPHITE_VERSION}
- LT_VERSION_CURRENT ${GRAPHITE_API_CURRENT}
- LT_VERSION_REVISION ${GRAPHITE_API_REVISION}
- LT_VERSION_AGE ${GRAPHITE_API_AGE})
-
--if (${CMAKE_BUILD_TYPE} STREQUAL "ClangASN")
-- set(GRAPHITE_LINK_FLAGS "-fsanitize=address")
--else (${CMAKE_BUILD_TYPE} STREQUAL "ClangASN")
-- set(GRAPHITE_LINK_FLAGS "")
--endif (${CMAKE_BUILD_TYPE} STREQUAL "ClangASN")
--
- if (${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
- set_target_properties(graphite2 PROPERTIES
- COMPILE_FLAGS "-Wall -Wextra -Wno-unknown-pragmas -Wendif-labels -Wshadow -Wctor-dtor-privacy -Wnon-virtual-dtor -fno-rtti -fno-exceptions -fvisibility=hidden -fvisibility-inlines-hidden -fno-stack-protector"
- LINK_FLAGS "-nodefaultlibs ${GRAPHITE_LINK_FLAGS}"
- LINKER_LANGUAGE C)
-+ if (CMAKE_COMPILER_IS_GNUCXX)
-+ add_definitions(-Wdouble-promotion)
-+ endif (CMAKE_COMPILER_IS_GNUCXX)
-+ message(STATUS "Compiler ID is: ${CMAKE_CXX_COMPILER_ID}")
-+ if (${CMAKE_CXX_COMPILER_ID} STREQUAL "Clang")
-+ add_definitions(-Wimplicit-fallthrough)
-+ endif (${CMAKE_CXX_COMPILER_ID} STREQUAL "Clang")
- if (${CMAKE_CXX_COMPILER} MATCHES ".*mingw.*")
- target_link_libraries(graphite2 kernel32 msvcr90 mingw32 gcc user32)
- else (${CMAKE_CXX_COMPILER} MATCHES ".*mingw.*")
- if (GRAPHITE2_ASAN)
- target_link_libraries(graphite2 c gcc_s)
- else (GRAPHITE2_ASAN)
- target_link_libraries(graphite2 c gcc)
- endif (GRAPHITE2_ASAN)
-@@ -127,17 +130,17 @@ if (${CMAKE_SYSTEM_NAME} STREQUAL "Linu
- nolib_test(stdc++ $<TARGET_SONAME_FILE:graphite2>)
- endif (${CMAKE_CXX_COMPILER} MATCHES ".*mingw.*")
- set(CMAKE_CXX_IMPLICIT_LINK_LIBRARIES "")
- CREATE_LIBTOOL_FILE(graphite2 "/lib${LIB_SUFFIX}")
- endif (${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
-
- if (${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
- set_target_properties(graphite2 PROPERTIES
-- COMPILE_FLAGS "-Wall -Wextra -Wno-unknown-pragmas -Wendif-labels -Wshadow -Wno-ctor-dtor-privacy -Wno-non-virtual-dtor -fno-rtti -fno-exceptions -fvisibility=hidden -fvisibility-inlines-hidden -fno-stack-protector"
-+ COMPILE_FLAGS "-Wall -Wextra -Wno-unknown-pragmas -Wimplicit-fallthrough -Wendif-labels -Wshadow -Wno-ctor-dtor-privacy -Wno-non-virtual-dtor -fno-rtti -fno-exceptions -fvisibility=hidden -fvisibility-inlines-hidden -fno-stack-protector"
- LINK_FLAGS "-nodefaultlibs"
- LINKER_LANGUAGE C)
- target_link_libraries(graphite2 c)
- include(Graphite)
- nolib_test(stdc++ $<TARGET_SONAME_FILE:graphite2>)
- set(CMAKE_CXX_IMPLICIT_LINK_LIBRARIES "")
- CREATE_LIBTOOL_FILE(graphite2 "/lib${LIB_SUFFIX}")
- endif (${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
-diff --git a/gfx/graphite2/src/CmapCache.cpp b/gfx/graphite2/src/CmapCache.cpp
---- a/gfx/graphite2/src/CmapCache.cpp
-+++ b/gfx/graphite2/src/CmapCache.cpp
-@@ -33,31 +33,31 @@ of the License or (at your option) any l
-
-
- using namespace graphite2;
-
- const void * bmp_subtable(const Face::Table & cmap)
- {
- const void * stbl;
- if (!cmap.size()) return 0;
-- if (TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 1, cmap.size()))
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 3, cmap.size()))
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 2, cmap.size()))
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 1, cmap.size()))
-- || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 0, cmap.size())))
-+ if (TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 1, cmap.size()), cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 3, cmap.size()), cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 2, cmap.size()), cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 1, cmap.size()), cmap.size())
-+ || TtfUtil::CheckCmapSubtable4(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 0, cmap.size()), cmap.size()))
- return stbl;
- return 0;
- }
-
- const void * smp_subtable(const Face::Table & cmap)
- {
- const void * stbl;
- if (!cmap.size()) return 0;
-- if (TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 10, cmap.size()))
-- || TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 4, cmap.size())))
-+ if (TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 3, 10, cmap.size()), cmap.size())
-+ || TtfUtil::CheckCmapSubtable12(stbl = TtfUtil::FindCmapSubtable(cmap, 0, 4, cmap.size()), cmap.size()))
- return stbl;
- return 0;
- }
-
- template <unsigned int (*NextCodePoint)(const void *, unsigned int, int *),
- uint16 (*LookupCodePoint)(const void *, unsigned int, int)>
- bool cache_subtable(uint16 * blocks[], const void * cst, const unsigned int limit)
- {
-diff --git a/gfx/graphite2/src/Code.cpp b/gfx/graphite2/src/Code.cpp
---- a/gfx/graphite2/src/Code.cpp
-+++ b/gfx/graphite2/src/Code.cpp
-@@ -37,17 +37,17 @@ of the License or (at your option) any l
- #include "inc/Code.h"
- #include "inc/Face.h"
- #include "inc/GlyphFace.h"
- #include "inc/GlyphCache.h"
- #include "inc/Machine.h"
- #include "inc/Rule.h"
- #include "inc/Silf.h"
-
--#include <stdio.h>
-+#include <cstdio>
-
- #ifdef NDEBUG
- #ifdef __GNUC__
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- #endif
-
-
-@@ -84,112 +84,119 @@ public:
- struct limits;
- struct analysis
- {
- uint8 slotref;
- context contexts[256];
- byte max_ref;
-
- analysis() : slotref(0), max_ref(0) {};
-- void set_ref(int index) throw();
-+ void set_ref(int index, bool incinsert=false) throw();
-+ void set_noref(int index) throw();
- void set_changed(int index) throw();
-
- };
-
-- decoder(const limits & lims, Code &code) throw();
-+ decoder(limits & lims, Code &code, enum passtype pt) throw();
-
- bool load(const byte * bc_begin, const byte * bc_end);
- void apply_analysis(instr * const code, instr * code_end);
- byte max_ref() { return _analysis.max_ref; }
- int pre_context() const { return _pre_context; }
-
- private:
- opcode fetch_opcode(const byte * bc);
- void analyse_opcode(const opcode, const int8 * const dp) throw();
- bool emit_opcode(opcode opc, const byte * & bc);
- bool validate_opcode(const opcode opc, const byte * const bc);
- bool valid_upto(const uint16 limit, const uint16 x) const throw();
-+ bool test_context() const throw();
- void failure(const status_t s) const throw() { _code.failure(s); }
-
- Code & _code;
- int _pre_context;
- uint16 _rule_length;
- instr * _instr;
- byte * _data;
-- const limits & _max;
-+ limits & _max;
- analysis _analysis;
-+ enum passtype _passtype;
-+ int _stack_depth;
-+ bool _in_ctxt_item;
- };
-
-
- struct Machine::Code::decoder::limits
- {
-- const byte * const bytecode;
-+ const byte * bytecode;
- const uint8 pre_context;
- const uint16 rule_length,
- classes,
- glyf_attrs,
- features;
- const byte attrid[gr_slatMax];
- };
-
--inline Machine::Code::decoder::decoder(const limits & lims, Code &code) throw()
-+inline Machine::Code::decoder::decoder(limits & lims, Code &code, enum passtype pt) throw()
- : _code(code),
- _pre_context(code._constraint ? 0 : lims.pre_context),
- _rule_length(code._constraint ? 1 : lims.rule_length),
-- _instr(code._code), _data(code._data), _max(lims)
-+ _instr(code._code), _data(code._data), _max(lims), _passtype(pt),
-+ _stack_depth(0),
-+ _in_ctxt_item(false)
- { }
-
-
-
- Machine::Code::Code(bool is_constraint, const byte * bytecode_begin, const byte * const bytecode_end,
-- uint8 pre_context, uint16 rule_length, const Silf & silf, const Face & face)
-+ uint8 pre_context, uint16 rule_length, const Silf & silf, const Face & face,
-+ enum passtype pt, byte * * const _out)
- : _code(0), _data(0), _data_size(0), _instr_count(0), _max_ref(0), _status(loaded),
-- _constraint(is_constraint), _modify(false), _delete(false), _own(true)
-+ _constraint(is_constraint), _modify(false), _delete(false), _own(_out==0)
- {
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::category _code_cat(face.tele.code);
- #endif
- assert(bytecode_begin != 0);
- if (bytecode_begin == bytecode_end)
- {
-- ::new (this) Code();
-+ // ::new (this) Code();
- return;
- }
- assert(bytecode_end > bytecode_begin);
- const opcode_t * op_to_fn = Machine::getOpcodeTable();
-
-- // Allocate code and dat target buffers, these sizes are a worst case
-+ // Allocate code and data target buffers, these sizes are a worst case
- // estimate. Once we know their real sizes the we'll shrink them.
-- _code = static_cast<instr *>(malloc((bytecode_end - bytecode_begin)
-- * sizeof(instr)));
-- _data = static_cast<byte *>(malloc((bytecode_end - bytecode_begin)
-- * sizeof(byte)));
-+ if (_out) _code = reinterpret_cast<instr *>(*_out);
-+ else _code = static_cast<instr *>(malloc(estimateCodeDataOut(bytecode_end-bytecode_begin)));
-+ _data = reinterpret_cast<byte *>(_code + (bytecode_end - bytecode_begin));
-
- if (!_code || !_data) {
- failure(alloc_failed);
- return;
- }
-
-- const decoder::limits lims = {
-+ decoder::limits lims = {
- bytecode_end,
- pre_context,
- rule_length,
- silf.numClasses(),
- face.glyphs().numAttrs(),
- face.numFeatures(),
- {1,1,1,1,1,1,1,1,
- 1,1,1,1,1,1,1,255,
- 1,1,1,1,1,1,1,1,
- 1,1,1,1,1,1,0,0,
- 0,0,0,0,0,0,0,0,
- 0,0,0,0,0,0,0,0,
- 0,0,0,0,0,0,0, silf.numUser()}
- };
-
-- decoder dec(lims, *this);
-+ decoder dec(lims, *this, pt);
- if(!dec.load(bytecode_begin, bytecode_end))
- return;
-
- // Is this an empty program?
- if (_instr_count == 0)
- {
- release_buffers();
- ::new (this) Code();
-@@ -204,20 +211,25 @@ Machine::Code::Code(bool is_constraint,
-
- assert((_constraint && immutable()) || !_constraint);
- dec.apply_analysis(_code, _code + _instr_count);
- _max_ref = dec.max_ref();
-
- // Now we know exactly how much code and data the program really needs
- // realloc the buffers to exactly the right size so we don't waste any
- // memory.
-- assert((bytecode_end - bytecode_begin) >= std::ptrdiff_t(_instr_count));
-- assert((bytecode_end - bytecode_begin) >= std::ptrdiff_t(_data_size));
-- _code = static_cast<instr *>(realloc(_code, (_instr_count+1)*sizeof(instr)));
-- _data = static_cast<byte *>(realloc(_data, _data_size*sizeof(byte)));
-+ assert((bytecode_end - bytecode_begin) >= ptrdiff_t(_instr_count));
-+ assert((bytecode_end - bytecode_begin) >= ptrdiff_t(_data_size));
-+ memmove(_code + (_instr_count+1), _data, _data_size*sizeof(byte));
-+ size_t const total_sz = ((_instr_count+1) + (_data_size + sizeof(instr)-1)/sizeof(instr))*sizeof(instr);
-+ if (_out)
-+ *_out += total_sz;
-+ else
-+ _code = static_cast<instr *>(realloc(_code, total_sz));
-+ _data = reinterpret_cast<byte *>(_code + (_instr_count+1));
-
- if (!_code)
- {
- failure(alloc_failed);
- return;
- }
-
- // Make this RET_ZERO, we should never reach this but just in case ...
-@@ -232,16 +244,17 @@ Machine::Code::~Code() throw ()
- {
- if (_own)
- release_buffers();
- }
-
-
- bool Machine::Code::decoder::load(const byte * bc, const byte * bc_end)
- {
-+ _max.bytecode = bc_end;
- while (bc < bc_end)
- {
- const opcode opc = fetch_opcode(bc++);
- if (opc == vm::MAX_OPCODE)
- return false;
-
- analyse_opcode(opc, reinterpret_cast<const int8 *>(bc));
-
-@@ -261,141 +274,194 @@ opcode Machine::Code::decoder::fetch_opc
-
- // Do some basic sanity checks based on what we know about the opcode
- if (!validate_opcode(opc, bc)) return MAX_OPCODE;
-
- // And check it's arguments as far as possible
- switch (opc)
- {
- case NOP :
-+ break;
- case PUSH_BYTE :
- case PUSH_BYTEU :
- case PUSH_SHORT :
- case PUSH_SHORTU :
- case PUSH_LONG :
-+ ++_stack_depth;
-+ break;
- case ADD :
- case SUB :
- case MUL :
- case DIV :
- case MIN_ :
- case MAX_ :
-- case NEG :
-- case TRUNC8 :
-- case TRUNC16 :
-- case COND :
- case AND :
- case OR :
-- case NOT :
- case EQUAL :
- case NOT_EQ :
- case LESS :
- case GTR :
- case LESS_EQ :
- case GTR_EQ :
-+ case BITOR :
-+ case BITAND :
-+ if (--_stack_depth <= 0)
-+ failure(underfull_stack);
-+ break;
-+ case NEG :
-+ case TRUNC8 :
-+ case TRUNC16 :
-+ case NOT :
-+ case BITNOT :
-+ case BITSET :
-+ if (_stack_depth <= 0)
-+ failure(underfull_stack);
-+ break;
-+ case COND :
-+ _stack_depth -= 2;
-+ if (_stack_depth <= 0)
-+ failure(underfull_stack);
- break;
- case NEXT :
- case NEXT_N : // runtime checked
- case COPY_NEXT :
-+ test_context();
- ++_pre_context;
- break;
- case PUT_GLYPH_8BIT_OBS :
- valid_upto(_max.classes, bc[0]);
-+ test_context();
- break;
- case PUT_SUBS_8BIT_OBS :
- valid_upto(_rule_length, _pre_context + int8(bc[0]));
- valid_upto(_max.classes, bc[1]);
- valid_upto(_max.classes, bc[2]);
-+ test_context();
- break;
- case PUT_COPY :
- valid_upto(_rule_length, _pre_context + int8(bc[0]));
-+ test_context();
- break;
- case INSERT :
-- --_pre_context;
-+ if (_passtype >= PASS_TYPE_POSITIONING)
-+ failure(invalid_opcode);
-+ else
-+ --_pre_context;
- break;
- case DELETE :
-+ if (_passtype >= PASS_TYPE_POSITIONING)
-+ failure(invalid_opcode);
-+ test_context();
- break;
- case ASSOC :
- for (uint8 num = bc[0]; num; --num)
- valid_upto(_rule_length, _pre_context + int8(bc[num]));
-+ test_context();
- break;
- case CNTXT_ITEM :
- valid_upto(_max.rule_length, _max.pre_context + int8(bc[0]));
-- if (bc + 2 + bc[1] >= _max.bytecode) failure(jump_past_end);
-- if (_pre_context != 0) failure(nested_context_item);
-+ if (bc + 2 + bc[1] >= _max.bytecode) failure(jump_past_end);
-+ if (_in_ctxt_item) failure(nested_context_item);
- break;
- case ATTR_SET :
- case ATTR_ADD :
- case ATTR_SUB :
- case ATTR_SET_SLOT :
-+ if (--_stack_depth < 0)
-+ failure(underfull_stack);
- valid_upto(gr_slatMax, bc[0]);
-+ test_context();
- break;
- case IATTR_SET_SLOT :
-+ if (--_stack_depth < 0)
-+ failure(underfull_stack);
- if (valid_upto(gr_slatMax, bc[0]))
- valid_upto(_max.attrid[bc[0]], bc[1]);
-+ test_context();
- break;
- case PUSH_SLOT_ATTR :
-+ ++_stack_depth;
- valid_upto(gr_slatMax, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- break;
- case PUSH_GLYPH_ATTR_OBS :
-+ ++_stack_depth;
- valid_upto(_max.glyf_attrs, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- break;
- case PUSH_GLYPH_METRIC :
-+ ++_stack_depth;
- valid_upto(kgmetDescent, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- // level: dp[2] no check necessary
- break;
- case PUSH_FEAT :
-+ ++_stack_depth;
- valid_upto(_max.features, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- break;
- case PUSH_ATT_TO_GATTR_OBS :
-+ ++_stack_depth;
- valid_upto(_max.glyf_attrs, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- break;
- case PUSH_ATT_TO_GLYPH_METRIC :
-+ ++_stack_depth;
- valid_upto(kgmetDescent, bc[0]);
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- // level: dp[2] no check necessary
- break;
- case PUSH_ISLOT_ATTR :
-+ ++_stack_depth;
- if (valid_upto(gr_slatMax, bc[0]))
- {
- valid_upto(_rule_length, _pre_context + int8(bc[1]));
- valid_upto(_max.attrid[bc[0]], bc[2]);
- }
- break;
- case PUSH_IGLYPH_ATTR :// not implemented
-+ ++_stack_depth;
-+ break;
- case POP_RET :
-+ if (--_stack_depth < 0)
-+ failure(underfull_stack);
-+ GR_FALLTHROUGH;
-+ // no break
- case RET_ZERO :
- case RET_TRUE :
- break;
- case IATTR_SET :
- case IATTR_ADD :
- case IATTR_SUB :
-+ if (--_stack_depth < 0)
-+ failure(underfull_stack);
- if (valid_upto(gr_slatMax, bc[0]))
- valid_upto(_max.attrid[bc[0]], bc[1]);
-+ test_context();
- break;
- case PUSH_PROC_STATE : // dummy: dp[0] no check necessary
- case PUSH_VERSION :
-+ ++_stack_depth;
- break;
- case PUT_SUBS :
- valid_upto(_rule_length, _pre_context + int8(bc[0]));
- valid_upto(_max.classes, uint16(bc[1]<< 8) | bc[2]);
- valid_upto(_max.classes, uint16(bc[3]<< 8) | bc[4]);
-+ test_context();
- break;
- case PUT_SUBS2 : // not implemented
- case PUT_SUBS3 : // not implemented
- break;
- case PUT_GLYPH :
- valid_upto(_max.classes, uint16(bc[0]<< 8) | bc[1]);
-+ test_context();
- break;
- case PUSH_GLYPH_ATTR :
- case PUSH_ATT_TO_GLYPH_ATTR :
-+ ++_stack_depth;
- valid_upto(_max.glyf_attrs, uint16(bc[0]<< 8) | bc[1]);
- valid_upto(_rule_length, _pre_context + int8(bc[2]));
- break;
- default:
- failure(invalid_opcode);
- break;
- }
-
-@@ -410,62 +476,77 @@ void Machine::Code::decoder::analyse_opc
- switch (opc)
- {
- case DELETE :
- _code._delete = true;
- break;
- case PUT_GLYPH_8BIT_OBS :
- case PUT_GLYPH :
- _code._modify = true;
-- _analysis.set_changed(_analysis.slotref);
-+ _analysis.set_changed(0);
-+ break;
-+ case ATTR_SET :
-+ case ATTR_ADD :
-+ case ATTR_SET_SLOT :
-+ case IATTR_SET_SLOT :
-+ case IATTR_SET :
-+ case IATTR_ADD :
-+ case IATTR_SUB :
-+ _analysis.set_noref(0);
- break;
- case NEXT :
- case COPY_NEXT :
- if (!_analysis.contexts[_analysis.slotref].flags.inserted)
- ++_analysis.slotref;
- _analysis.contexts[_analysis.slotref] = context(_code._instr_count+1);
-- if (_analysis.slotref > _analysis.max_ref) _analysis.max_ref = _analysis.slotref;
-+ // if (_analysis.slotref > _analysis.max_ref) _analysis.max_ref = _analysis.slotref;
- break;
- case INSERT :
- _analysis.contexts[_analysis.slotref].flags.inserted = true;
- _code._modify = true;
- break;
- case PUT_SUBS_8BIT_OBS : // slotref on 1st parameter
- case PUT_SUBS :
- _code._modify = true;
-- _analysis.set_changed(_analysis.slotref);
-+ _analysis.set_changed(0);
-+ GR_FALLTHROUGH;
- // no break
- case PUT_COPY :
- {
-- if (arg[0] != 0) { _analysis.set_changed(_analysis.slotref); _code._modify = true; }
-+ if (arg[0] != 0) { _analysis.set_changed(0); _code._modify = true; }
- if (arg[0] <= 0 && -arg[0] <= _analysis.slotref - _analysis.contexts[_analysis.slotref].flags.inserted)
-- _analysis.set_ref(_analysis.slotref + arg[0] - _analysis.contexts[_analysis.slotref].flags.inserted);
-- else if (_analysis.slotref + arg[0] > _analysis.max_ref) _analysis.max_ref = _analysis.slotref + arg[0];
-+ _analysis.set_ref(arg[0], true);
-+ else if (arg[0] > 0)
-+ _analysis.set_ref(arg[0], true);
- break;
- }
- case PUSH_ATT_TO_GATTR_OBS : // slotref on 2nd parameter
- if (_code._constraint) return;
-+ GR_FALLTHROUGH;
- // no break
- case PUSH_GLYPH_ATTR_OBS :
- case PUSH_SLOT_ATTR :
- case PUSH_GLYPH_METRIC :
- case PUSH_ATT_TO_GLYPH_METRIC :
- case PUSH_ISLOT_ATTR :
- case PUSH_FEAT :
- if (arg[1] <= 0 && -arg[1] <= _analysis.slotref - _analysis.contexts[_analysis.slotref].flags.inserted)
-- _analysis.set_ref(_analysis.slotref + arg[1] - _analysis.contexts[_analysis.slotref].flags.inserted);
-- else if (_analysis.slotref + arg[1] > _analysis.max_ref) _analysis.max_ref = _analysis.slotref + arg[1];
-+ _analysis.set_ref(arg[1], true);
-+ else if (arg[1] > 0)
-+ _analysis.set_ref(arg[1], true);
- break;
- case PUSH_ATT_TO_GLYPH_ATTR :
- if (_code._constraint) return;
-+ GR_FALLTHROUGH;
- // no break
- case PUSH_GLYPH_ATTR :
- if (arg[2] <= 0 && -arg[2] <= _analysis.slotref - _analysis.contexts[_analysis.slotref].flags.inserted)
-- _analysis.set_ref(_analysis.slotref + arg[2] - _analysis.contexts[_analysis.slotref].flags.inserted);
-- else if (_analysis.slotref + arg[2] > _analysis.max_ref) _analysis.max_ref = _analysis.slotref + arg[2];
-+ _analysis.set_ref(arg[2], true);
-+ else if (arg[2] > 0)
-+ _analysis.set_ref(arg[2], true);
- break;
- case ASSOC : // slotrefs in varargs
- break;
- default:
- break;
- }
- }
-
-@@ -494,32 +575,41 @@ bool Machine::Code::decoder::emit_opcode
- _code._data_size += param_sz;
- }
-
- // recursively decode a context item so we can split the skip into
- // instruction and data portions.
- if (opc == CNTXT_ITEM)
- {
- assert(_pre_context == 0);
-+ _in_ctxt_item = true;
- _pre_context = _max.pre_context + int8(_data[-2]);
- _rule_length = _max.rule_length;
-
- const size_t ctxt_start = _code._instr_count;
- byte & instr_skip = _data[-1];
- byte & data_skip = *_data++;
- ++_code._data_size;
-+ const byte *curr_end = _max.bytecode;
-
- if (load(bc, bc + instr_skip))
- {
- bc += instr_skip;
- data_skip = instr_skip - (_code._instr_count - ctxt_start);
- instr_skip = _code._instr_count - ctxt_start;
-+ _max.bytecode = curr_end;
-
- _rule_length = 1;
- _pre_context = 0;
-+ _in_ctxt_item = false;
-+ }
-+ else
-+ {
-+ _pre_context = 0;
-+ return false;
- }
- }
-
- return bool(_code);
- }
-
-
- void Machine::Code::decoder::apply_analysis(instr * const code, instr * code_end)
-@@ -533,87 +623,115 @@ void Machine::Code::decoder::apply_analy
- {
- if (!c->flags.referenced || !c->flags.changed) continue;
-
- instr * const tip = code + c->codeRef + tempcount;
- memmove(tip+1, tip, (code_end - tip) * sizeof(instr));
- *tip = temp_copy;
- ++code_end;
- ++tempcount;
-+ _code._delete = true;
- }
-
- _code._instr_count = code_end - code;
- }
-
-
- inline
- bool Machine::Code::decoder::validate_opcode(const opcode opc, const byte * const bc)
- {
- if (opc >= MAX_OPCODE)
- {
- failure(invalid_opcode);
- return false;
- }
- const opcode_t & op = Machine::getOpcodeTable()[opc];
-+ if (op.param_sz == VARARGS && bc >= _max.bytecode)
-+ {
-+ failure(arguments_exhausted);
-+ return false;
-+ }
- const size_t param_sz = op.param_sz == VARARGS ? bc[0] + 1 : op.param_sz;
-- if (bc + param_sz > _max.bytecode)
-+ if (bc - 1 + param_sz >= _max.bytecode)
- {
- failure(arguments_exhausted);
- return false;
- }
- return true;
- }
-
-
- bool Machine::Code::decoder::valid_upto(const uint16 limit, const uint16 x) const throw()
- {
- const bool t = x < limit;
- if (!t) failure(out_of_range_data);
- return t;
- }
-
-+bool Machine::Code::decoder::test_context() const throw()
-+{
-+ if (_pre_context >= _rule_length)
-+ {
-+ failure(out_of_range_data);
-+ return false;
-+ }
-+ return true;
-+}
-
- inline
- void Machine::Code::failure(const status_t s) throw() {
- release_buffers();
- _status = s;
- }
-
-
- inline
--void Machine::Code::decoder::analysis::set_ref(const int index) throw() {
-- contexts[index].flags.referenced = true;
-- if (index > max_ref) max_ref = index;
-+void Machine::Code::decoder::analysis::set_ref(int index, bool incinsert) throw() {
-+ if (incinsert && contexts[slotref].flags.inserted) --index;
-+ if (index + slotref < 0) return;
-+ contexts[index + slotref].flags.referenced = true;
-+ if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
- }
-
-
- inline
--void Machine::Code::decoder::analysis::set_changed(const int index) throw() {
-- contexts[index].flags.changed = true;
-- if (index > max_ref) max_ref = index;
-+void Machine::Code::decoder::analysis::set_noref(int index) throw() {
-+ if (contexts[slotref].flags.inserted) --index;
-+ if (index + slotref < 0) return;
-+ if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
-+}
-+
-+
-+inline
-+void Machine::Code::decoder::analysis::set_changed(int index) throw() {
-+ if (contexts[slotref].flags.inserted) --index;
-+ if (index + slotref < 0) return;
-+ contexts[index + slotref].flags.changed = true;
-+ if ((index > 0 || !contexts[index + slotref].flags.inserted) && index + slotref > max_ref) max_ref = index + slotref;
- }
-
-
- void Machine::Code::release_buffers() throw()
- {
-- free(_code);
-- free(_data);
-+ if (_own)
-+ free(_code);
- _code = 0;
- _data = 0;
- _own = false;
- }
-
-
- int32 Machine::Code::run(Machine & m, slotref * & map) const
- {
-- assert(_own);
-+// assert(_own);
- assert(*this); // Check we are actually runnable
-
-- if (m.slotMap().size() <= size_t(_max_ref + m.slotMap().context()))
-+ if (m.slotMap().size() <= size_t(_max_ref + m.slotMap().context())
-+ || m.slotMap()[_max_ref + m.slotMap().context()] == 0)
- {
- m._status = Machine::slot_offset_out_bounds;
--// return (m.slotMap().end() - map);
- return 1;
-+// return m.run(_code, _data, map);
- }
-
- return m.run(_code, _data, map);
- }
-
-diff --git a/gfx/graphite2/src/Collider.cpp b/gfx/graphite2/src/Collider.cpp
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/Collider.cpp
-@@ -0,0 +1,1088 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#include <algorithm>
-+#include <limits>
-+#include <math.h>
-+#include <string>
-+#include <functional>
-+#include "inc/Collider.h"
-+#include "inc/Segment.h"
-+#include "inc/Slot.h"
-+#include "inc/GlyphCache.h"
-+#include "inc/Sparse.h"
-+
-+#define ISQRT2 0.707106781f
-+
-+// Possible rounding error for subbox boundaries: 0.016 = 1/64 = 1/256 * 4
-+// (values in font range from 0..256)
-+// #define SUBBOX_RND_ERR 0.016
-+
-+using namespace graphite2;
-+
-+//// SHIFT-COLLIDER ////
-+
-+// Initialize the Collider to hold the basic movement limits for the
-+// target slot, the one we are focusing on fixing.
-+bool ShiftCollider::initSlot(Segment *seg, Slot *aSlot, const Rect &limit, float margin, float marginWeight,
-+ const Position &currShift, const Position &currOffset, int dir, GR_MAYBE_UNUSED json * const dbgout)
-+{
-+ int i;
-+ float mx, mn;
-+ float a, shift;
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+ unsigned short gid = aSlot->gid();
-+ if (!gc.check(gid))
-+ return false;
-+ const BBox &bb = gc.getBoundingBBox(gid);
-+ const SlantBox &sb = gc.getBoundingSlantBox(gid);
-+ //float sx = aSlot->origin().x + currShift.x;
-+ //float sy = aSlot->origin().y + currShift.y;
-+ if (currOffset.x != 0.f || currOffset.y != 0.f)
-+ _limit = Rect(limit.bl - currOffset, limit.tr - currOffset);
-+ else
-+ _limit = limit;
-+ // For a ShiftCollider, these indices indicate which vector we are moving by:
-+ // each _ranges represents absolute space with respect to the origin of the slot. Thus take into account true origins but subtract the vmin for the slot
-+ for (i = 0; i < 4; ++i)
-+ {
-+ switch (i) {
-+ case 0 : // x direction
-+ mn = _limit.bl.x + currOffset.x;
-+ mx = _limit.tr.x + currOffset.x;
-+ _len[i] = bb.xa - bb.xi;
-+ a = currOffset.y + currShift.y;
-+ _ranges[i].initialise<XY>(mn, mx, margin, marginWeight, a);
-+ break;
-+ case 1 : // y direction
-+ mn = _limit.bl.y + currOffset.y;
-+ mx = _limit.tr.y + currOffset.y;
-+ _len[i] = bb.ya - bb.yi;
-+ a = currOffset.x + currShift.x;
-+ _ranges[i].initialise<XY>(mn, mx, margin, marginWeight, a);
-+ break;
-+ case 2 : // sum (negatively sloped diagonal boundaries)
-+ // pick closest x,y limit boundaries in s direction
-+ shift = currOffset.x + currOffset.y + currShift.x + currShift.y;
-+ mn = -2 * min(currShift.x - _limit.bl.x, currShift.y - _limit.bl.y) + shift;
-+ mx = 2 * min(_limit.tr.x - currShift.x, _limit.tr.y - currShift.y) + shift;
-+ _len[i] = sb.sa - sb.si;
-+ a = currOffset.x - currOffset.y + currShift.x - currShift.y;
-+ _ranges[i].initialise<SD>(mn, mx, margin / ISQRT2, marginWeight, a);
-+ break;
-+ case 3 : // diff (positively sloped diagonal boundaries)
-+ // pick closest x,y limit boundaries in d direction
-+ shift = currOffset.x - currOffset.y + currShift.x - currShift.y;
-+ mn = -2 * min(currShift.x - _limit.bl.x, _limit.tr.y - currShift.y) + shift;
-+ mx = 2 * min(_limit.tr.x - currShift.x, currShift.y - _limit.bl.y) + shift;
-+ _len[i] = sb.da - sb.di;
-+ a = currOffset.x + currOffset.y + currShift.x + currShift.y;
-+ _ranges[i].initialise<SD>(mn, mx, margin / ISQRT2, marginWeight, a);
-+ break;
-+ }
-+ }
-+
-+ _target = aSlot;
-+ if ((dir & 1) == 0)
-+ {
-+ // For LTR, switch and negate x limits.
-+ _limit.bl.x = -1 * limit.tr.x;
-+ //_limit.tr.x = -1 * limit.bl.x;
-+ }
-+ _currOffset = currOffset;
-+ _currShift = currShift;
-+ _origin = aSlot->origin() - currOffset; // the original anchor position of the glyph
-+
-+ _margin = margin;
-+ _marginWt = marginWeight;
-+
-+ SlotCollision *c = seg->collisionInfo(aSlot);
-+ _seqClass = c->seqClass();
-+ _seqProxClass = c->seqProxClass();
-+ _seqOrder = c->seqOrder();
-+ return true;
-+}
-+
-+template <class O>
-+float sdm(float vi, float va, float mx, float my, O op)
-+{
-+ float res = 2 * mx - vi;
-+ if (op(res, vi + 2 * my))
-+ {
-+ res = va + 2 * my;
-+ if (op(res, 2 * mx - va))
-+ res = mx + my;
-+ }
-+ return res;
-+}
-+
-+// Mark an area with a cost that can vary along the x or y axis. The region is expressed in terms of the centre of the target glyph in each axis
-+void ShiftCollider::addBox_slope(bool isx, const Rect &box, const BBox &bb, const SlantBox &sb, const Position &org, float weight, float m, bool minright, int axis)
-+{
-+ float a, c;
-+ switch (axis) {
-+ case 0 :
-+ if (box.bl.y < org.y + bb.ya && box.tr.y > org.y + bb.yi && box.width() > 0)
-+ {
-+ a = org.y + 0.5f * (bb.yi + bb.ya);
-+ c = 0.5f * (bb.xi + bb.xa);
-+ if (isx)
-+ _ranges[axis].weighted<XY>(box.bl.x - c, box.tr.x - c, weight, a, m,
-+ (minright ? box.tr.x : box.bl.x) - c, a, 0, false);
-+ else
-+ _ranges[axis].weighted<XY>(box.bl.x - c, box.tr.x - c, weight, a, 0, 0, org.y,
-+ m * (a * a + sqr((minright ? box.tr.y : box.bl.y) - 0.5f * (bb.yi + bb.ya))), false);
-+ }
-+ break;
-+ case 1 :
-+ if (box.bl.x < org.x + bb.xa && box.tr.x > org.x + bb.xi && box.height() > 0)
-+ {
-+ a = org.x + 0.5f * (bb.xi + bb.xa);
-+ c = 0.5f * (bb.yi + bb.ya);
-+ if (isx)
-+ _ranges[axis].weighted<XY>(box.bl.y - c, box.tr.y - c, weight, a, 0, 0, org.x,
-+ m * (a * a + sqr((minright ? box.tr.x : box.bl.x) - 0.5f * (bb.xi + bb.xa))), false);
-+ else
-+ _ranges[axis].weighted<XY>(box.bl.y - c, box.tr.y - c, weight, a, m,
-+ (minright ? box.tr.y : box.bl.y) - c, a, 0, false);
-+ }
-+ break;
-+ case 2 :
-+ if (box.bl.x - box.tr.y < org.x - org.y + sb.da && box.tr.x - box.bl.y > org.x - org.y + sb.di)
-+ {
-+ float d = org.x - org.y + 0.5f * (sb.di + sb.da);
-+ c = 0.5f * (sb.si + sb.sa);
-+ float smax = min(2 * box.tr.x - d, 2 * box.tr.y + d);
-+ float smin = max(2 * box.bl.x - d, 2 * box.bl.y + d);
-+ if (smin > smax) return;
-+ float si;
-+ a = d;
-+ if (isx)
-+ si = 2 * (minright ? box.tr.x : box.bl.x) - a;
-+ else
-+ si = 2 * (minright ? box.tr.y : box.bl.y) + a;
-+ _ranges[axis].weighted<SD>(smin - c, smax - c, weight / 2, a, m / 2, si, 0, 0, isx);
-+ }
-+ break;
-+ case 3 :
-+ if (box.bl.x + box.bl.y < org.x + org.y + sb.sa && box.tr.x + box.tr.y > org.x + org.y + sb.si)
-+ {
-+ float s = org.x + org.y + 0.5f * (sb.si + sb.sa);
-+ c = 0.5f * (sb.di + sb.da);
-+ float dmax = min(2 * box.tr.x - s, s - 2 * box.bl.y);
-+ float dmin = max(2 * box.bl.x - s, s - 2 * box.tr.y);
-+ if (dmin > dmax) return;
-+ float di;
-+ a = s;
-+ if (isx)
-+ di = 2 * (minright ? box.tr.x : box.bl.x) - a;
-+ else
-+ di = 2 * (minright ? box.tr.y : box.bl.y) + a;
-+ _ranges[axis].weighted<SD>(dmin - c, dmax - c, weight / 2, a, m / 2, di, 0, 0, !isx);
-+ }
-+ break;
-+ default :
-+ break;
-+ }
-+ return;
-+}
-+
-+// Mark an area with an absolute cost, making it completely inaccessible.
-+inline void ShiftCollider::removeBox(const Rect &box, const BBox &bb, const SlantBox &sb, const Position &org, int axis)
-+{
-+ float c;
-+ switch (axis) {
-+ case 0 :
-+ if (box.bl.y < org.y + bb.ya && box.tr.y > org.y + bb.yi && box.width() > 0)
-+ {
-+ c = 0.5f * (bb.xi + bb.xa);
-+ _ranges[axis].exclude(box.bl.x - c, box.tr.x - c);
-+ }
-+ break;
-+ case 1 :
-+ if (box.bl.x < org.x + bb.xa && box.tr.x > org.x + bb.xi && box.height() > 0)
-+ {
-+ c = 0.5f * (bb.yi + bb.ya);
-+ _ranges[axis].exclude(box.bl.y - c, box.tr.y - c);
-+ }
-+ break;
-+ case 2 :
-+ if (box.bl.x - box.tr.y < org.x - org.y + sb.da && box.tr.x - box.bl.y > org.x - org.y + sb.di
-+ && box.width() > 0 && box.height() > 0)
-+ {
-+ float di = org.x - org.y + sb.di;
-+ float da = org.x - org.y + sb.da;
-+ float smax = sdm(di, da, box.tr.x, box.tr.y, std::greater<float>());
-+ float smin = sdm(da, di, box.bl.x, box.bl.y, std::less<float>());
-+ c = 0.5f * (sb.si + sb.sa);
-+ _ranges[axis].exclude(smin - c, smax - c);
-+ }
-+ break;
-+ case 3 :
-+ if (box.bl.x + box.bl.y < org.x + org.y + sb.sa && box.tr.x + box.tr.y > org.x + org.y + sb.si
-+ && box.width() > 0 && box.height() > 0)
-+ {
-+ float si = org.x + org.y + sb.si;
-+ float sa = org.x + org.y + sb.sa;
-+ float dmax = sdm(si, sa, box.tr.x, -box.bl.y, std::greater<float>());
-+ float dmin = sdm(sa, si, box.bl.x, -box.tr.y, std::less<float>());
-+ c = 0.5f * (sb.di + sb.da);
-+ _ranges[axis].exclude(dmin - c, dmax - c);
-+ }
-+ break;
-+ default :
-+ break;
-+ }
-+ return;
-+}
-+
-+// Adjust the movement limits for the target to avoid having it collide
-+// with the given neighbor slot. Also determine if there is in fact a collision
-+// between the target and the given slot.
-+bool ShiftCollider::mergeSlot(Segment *seg, Slot *slot, const Position &currShift,
-+ bool isAfter, // slot is logically after _target
-+ bool sameCluster, bool &hasCol, bool isExclusion,
-+ GR_MAYBE_UNUSED json * const dbgout )
-+{
-+ bool isCol = false;
-+ const float sx = slot->origin().x - _origin.x + currShift.x;
-+ const float sy = slot->origin().y - _origin.y + currShift.y;
-+ const float sd = sx - sy;
-+ const float ss = sx + sy;
-+ float vmin, vmax;
-+ float omin, omax, otmin, otmax;
-+ float cmin, cmax; // target limits
-+ float torg;
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+ const unsigned short gid = slot->gid();
-+ if (!gc.check(gid))
-+ return false;
-+ const BBox &bb = gc.getBoundingBBox(gid);
-+
-+ SlotCollision * cslot = seg->collisionInfo(slot);
-+ int orderFlags = 0;
-+ bool sameClass = _seqProxClass == 0 && cslot->seqClass() == _seqClass;
-+ if (sameCluster && _seqClass
-+ && (sameClass || (_seqProxClass != 0 && cslot->seqClass() == _seqProxClass)))
-+ // Force the target glyph to be in the specified direction from the slot we're testing.
-+ orderFlags = _seqOrder;
-+
-+ // short circuit if only interested in direct collision and we are out of range
-+ if (orderFlags || (sx + bb.xa + _margin >= _limit.bl.x && sx + bb.xi - _margin <= _limit.tr.x)
-+ || (sy + bb.ya + _margin >= _limit.bl.y && sy + bb.yi - _margin <= _limit.tr.y))
-+
-+ {
-+ const float tx = _currOffset.x + _currShift.x;
-+ const float ty = _currOffset.y + _currShift.y;
-+ const float td = tx - ty;
-+ const float ts = tx + ty;
-+ const SlantBox &sb = gc.getBoundingSlantBox(gid);
-+ const unsigned short tgid = _target->gid();
-+ const BBox &tbb = gc.getBoundingBBox(tgid);
-+ const SlantBox &tsb = gc.getBoundingSlantBox(tgid);
-+ float seq_above_wt = cslot->seqAboveWt();
-+ float seq_below_wt = cslot->seqBelowWt();
-+ float seq_valign_wt = cslot->seqValignWt();
-+ // if isAfter, invert orderFlags for diagonal orders.
-+ if (isAfter)
-+ {
-+ // invert appropriate bits
-+ orderFlags ^= (sameClass ? 0x3F : 0x3);
-+ // consider 2 bits at a time, non overlapping. If both bits set, clear them
-+ orderFlags = orderFlags ^ ((((orderFlags >> 1) & orderFlags) & 0x15) * 3);
-+ }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ dbgout->setenv(0, slot);
-+#endif
-+
-+ // Process main bounding octabox.
-+ for (int i = 0; i < 4; ++i)
-+ {
-+ switch (i) {
-+ case 0 : // x direction
-+ vmin = max(max(bb.xi - tbb.xa + sx, sb.di - tsb.da + ty + sd), sb.si - tsb.sa - ty + ss);
-+ vmax = min(min(bb.xa - tbb.xi + sx, sb.da - tsb.di + ty + sd), sb.sa - tsb.si - ty + ss);
-+ otmin = tbb.yi + ty;
-+ otmax = tbb.ya + ty;
-+ omin = bb.yi + sy;
-+ omax = bb.ya + sy;
-+ torg = _currOffset.x;
-+ cmin = _limit.bl.x + torg;
-+ cmax = _limit.tr.x - tbb.xi + tbb.xa + torg;
-+ break;
-+ case 1 : // y direction
-+ vmin = max(max(bb.yi - tbb.ya + sy, tsb.di - sb.da + tx - sd), sb.si - tsb.sa - tx + ss);
-+ vmax = min(min(bb.ya - tbb.yi + sy, tsb.da - sb.di + tx - sd), sb.sa - tsb.si - tx + ss);
-+ otmin = tbb.xi + tx;
-+ otmax = tbb.xa + tx;
-+ omin = bb.xi + sx;
-+ omax = bb.xa + sx;
-+ torg = _currOffset.y;
-+ cmin = _limit.bl.y + torg;
-+ cmax = _limit.tr.y - tbb.yi + tbb.ya + torg;
-+ break;
-+ case 2 : // sum - moving along the positively-sloped vector, so the boundaries are the
-+ // negatively-sloped boundaries.
-+ vmin = max(max(sb.si - tsb.sa + ss, 2 * (bb.yi - tbb.ya + sy) + td), 2 * (bb.xi - tbb.xa + sx) - td);
-+ vmax = min(min(sb.sa - tsb.si + ss, 2 * (bb.ya - tbb.yi + sy) + td), 2 * (bb.xa - tbb.xi + sx) - td);
-+ otmin = tsb.di + td;
-+ otmax = tsb.da + td;
-+ omin = sb.di + sd;
-+ omax = sb.da + sd;
-+ torg = _currOffset.x + _currOffset.y;
-+ cmin = _limit.bl.x + _limit.bl.y + torg;
-+ cmax = _limit.tr.x + _limit.tr.y - tsb.si + tsb.sa + torg;
-+ break;
-+ case 3 : // diff - moving along the negatively-sloped vector, so the boundaries are the
-+ // positively-sloped boundaries.
-+ vmin = max(max(sb.di - tsb.da + sd, 2 * (bb.xi - tbb.xa + sx) - ts), -2 * (bb.ya - tbb.yi + sy) + ts);
-+ vmax = min(min(sb.da - tsb.di + sd, 2 * (bb.xa - tbb.xi + sx) - ts), -2 * (bb.yi - tbb.ya + sy) + ts);
-+ otmin = tsb.si + ts;
-+ otmax = tsb.sa + ts;
-+ omin = sb.si + ss;
-+ omax = sb.sa + ss;
-+ torg = _currOffset.x - _currOffset.y;
-+ cmin = _limit.bl.x - _limit.tr.y + torg;
-+ cmax = _limit.tr.x - _limit.bl.y - tsb.di + tsb.da + torg;
-+ break;
-+ default :
-+ continue;
-+ }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ dbgout->setenv(1, reinterpret_cast<void *>(-1));
-+#define DBGTAG(x) if (dbgout) dbgout->setenv(1, reinterpret_cast<void *>(-x));
-+#else
-+#define DBGTAG(x)
-+#endif
-+
-+ if (orderFlags)
-+ {
-+ Position org(tx, ty);
-+ float xminf = _limit.bl.x + _currOffset.x + tbb.xi;
-+ float xpinf = _limit.tr.x + _currOffset.x + tbb.xa;
-+ float ypinf = _limit.tr.y + _currOffset.y + tbb.ya;
-+ float yminf = _limit.bl.y + _currOffset.y + tbb.yi;
-+ switch (orderFlags) {
-+ case SlotCollision::SEQ_ORDER_RIGHTUP :
-+ {
-+ float r1Xedge = cslot->seqAboveXoff() + 0.5f * (bb.xi + bb.xa) + sx;
-+ float r3Xedge = cslot->seqBelowXlim() + bb.xa + sx + 0.5f * (tbb.xa - tbb.xi);
-+ float r2Yedge = 0.5f * (bb.yi + bb.ya) + sy;
-+
-+ // DBGTAG(1x) means the regions are up and right
-+ // region 1
-+ DBGTAG(11)
-+ addBox_slope(true, Rect(Position(xminf, r2Yedge), Position(r1Xedge, ypinf)),
-+ tbb, tsb, org, 0, seq_above_wt, true, i);
-+ // region 2
-+ DBGTAG(12)
-+ removeBox(Rect(Position(xminf, yminf), Position(r3Xedge, r2Yedge)), tbb, tsb, org, i);
-+ // region 3, which end is zero is irrelevant since m weight is 0
-+ DBGTAG(13)
-+ addBox_slope(true, Rect(Position(r3Xedge, yminf), Position(xpinf, r2Yedge - cslot->seqValignHt())),
-+ tbb, tsb, org, seq_below_wt, 0, true, i);
-+ // region 4
-+ DBGTAG(14)
-+ addBox_slope(false, Rect(Position(sx + bb.xi, r2Yedge), Position(xpinf, r2Yedge + cslot->seqValignHt())),
-+ tbb, tsb, org, 0, seq_valign_wt, true, i);
-+ // region 5
-+ DBGTAG(15)
-+ addBox_slope(false, Rect(Position(sx + bb.xi, r2Yedge - cslot->seqValignHt()), Position(xpinf, r2Yedge)),
-+ tbb, tsb, org, seq_below_wt, seq_valign_wt, false, i);
-+ break;
-+ }
-+ case SlotCollision::SEQ_ORDER_LEFTDOWN :
-+ {
-+ float r1Xedge = 0.5f * (bb.xi + bb.xa) + cslot->seqAboveXoff() + sx;
-+ float r3Xedge = bb.xi - cslot->seqBelowXlim() + sx - 0.5f * (tbb.xa - tbb.xi);
-+ float r2Yedge = 0.5f * (bb.yi + bb.ya) + sy;
-+ // DBGTAG(2x) means the regions are up and right
-+ // region 1
-+ DBGTAG(21)
-+ addBox_slope(true, Rect(Position(r1Xedge, yminf), Position(xpinf, r2Yedge)),
-+ tbb, tsb, org, 0, seq_above_wt, false, i);
-+ // region 2
-+ DBGTAG(22)
-+ removeBox(Rect(Position(r3Xedge, r2Yedge), Position(xpinf, ypinf)), tbb, tsb, org, i);
-+ // region 3
-+ DBGTAG(23)
-+ addBox_slope(true, Rect(Position(xminf, r2Yedge - cslot->seqValignHt()), Position(r3Xedge, ypinf)),
-+ tbb, tsb, org, seq_below_wt, 0, false, i);
-+ // region 4
-+ DBGTAG(24)
-+ addBox_slope(false, Rect(Position(xminf, r2Yedge), Position(sx + bb.xa, r2Yedge + cslot->seqValignHt())),
-+ tbb, tsb, org, 0, seq_valign_wt, true, i);
-+ // region 5
-+ DBGTAG(25)
-+ addBox_slope(false, Rect(Position(xminf, r2Yedge - cslot->seqValignHt()),
-+ Position(sx + bb.xa, r2Yedge)), tbb, tsb, org, seq_below_wt, seq_valign_wt, false, i);
-+ break;
-+ }
-+ case SlotCollision::SEQ_ORDER_NOABOVE : // enforce neighboring glyph being above
-+ DBGTAG(31);
-+ removeBox(Rect(Position(bb.xi - tbb.xa + sx, sy + bb.ya),
-+ Position(bb.xa - tbb.xi + sx, ypinf)), tbb, tsb, org, i);
-+ break;
-+ case SlotCollision::SEQ_ORDER_NOBELOW : // enforce neighboring glyph being below
-+ DBGTAG(32);
-+ removeBox(Rect(Position(bb.xi - tbb.xa + sx, yminf),
-+ Position(bb.xa - tbb.xi + sx, sy + bb.yi)), tbb, tsb, org, i);
-+ break;
-+ case SlotCollision::SEQ_ORDER_NOLEFT : // enforce neighboring glyph being to the left
-+ DBGTAG(33)
-+ removeBox(Rect(Position(xminf, bb.yi - tbb.ya + sy),
-+ Position(bb.xi - tbb.xa + sx, bb.ya - tbb.yi + sy)), tbb, tsb, org, i);
-+ break;
-+ case SlotCollision::SEQ_ORDER_NORIGHT : // enforce neighboring glyph being to the right
-+ DBGTAG(34)
-+ removeBox(Rect(Position(bb.xa - tbb.xi + sx, bb.yi - tbb.ya + sy),
-+ Position(xpinf, bb.ya - tbb.yi + sy)), tbb, tsb, org, i);
-+ break;
-+ default :
-+ break;
-+ }
-+ }
-+
-+ if (vmax < cmin - _margin || vmin > cmax + _margin || omax < otmin - _margin || omin > otmax + _margin)
-+ continue;
-+
-+ // Process sub-boxes that are defined for this glyph.
-+ // We only need to do this if there was in fact a collision with the main octabox.
-+ uint8 numsub = gc.numSubBounds(gid);
-+ if (numsub > 0)
-+ {
-+ bool anyhits = false;
-+ for (int j = 0; j < numsub; ++j)
-+ {
-+ const BBox &sbb = gc.getSubBoundingBBox(gid, j);
-+ const SlantBox &ssb = gc.getSubBoundingSlantBox(gid, j);
-+ switch (i) {
-+ case 0 : // x
-+ vmin = max(max(sbb.xi-tbb.xa+sx, ssb.di-tsb.da+sd+ty), ssb.si-tsb.sa+ss-ty);
-+ vmax = min(min(sbb.xa-tbb.xi+sx, ssb.da-tsb.di+sd+ty), ssb.sa-tsb.si+ss-ty);
-+ omin = sbb.yi + sy;
-+ omax = sbb.ya + sy;
-+ break;
-+ case 1 : // y
-+ vmin = max(max(sbb.yi-tbb.ya+sy, tsb.di-ssb.da-sd+tx), ssb.si-tsb.sa+ss-tx);
-+ vmax = min(min(sbb.ya-tbb.yi+sy, tsb.da-ssb.di-sd+tx), ssb.sa-tsb.si+ss-tx);
-+ omin = sbb.xi + sx;
-+ omax = sbb.xa + sx;
-+ break;
-+ case 2 : // sum
-+ vmin = max(max(ssb.si-tsb.sa+ss, 2*(sbb.yi-tbb.ya+sy)+td), 2*(sbb.xi-tbb.xa+sx)-td);
-+ vmax = min(min(ssb.sa-tsb.si+ss, 2*(sbb.ya-tbb.yi+sy)+td), 2*(sbb.xa-tbb.xi+sx)-td);
-+ omin = ssb.di + sd;
-+ omax = ssb.da + sd;
-+ break;
-+ case 3 : // diff
-+ vmin = max(max(ssb.di-tsb.da+sd, 2*(sbb.xi-tbb.xa+sx)-ts), -2*(sbb.ya-tbb.yi+sy)+ts);
-+ vmax = min(min(ssb.da-tsb.di+sd, 2*(sbb.xa-tbb.xi+sx)-ts), -2*(sbb.yi-tbb.ya+sy)+ts);
-+ omin = ssb.si + ss;
-+ omax = ssb.sa + ss;
-+ break;
-+ }
-+ if (vmax < cmin - _margin || vmin > cmax + _margin || omax < otmin - _margin || omin > otmax + _margin)
-+ continue;
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ dbgout->setenv(1, reinterpret_cast<void *>(j));
-+#endif
-+ if (omin > otmax)
-+ _ranges[i].weightedAxis(i, vmin - _margin, vmax + _margin, 0, 0, 0, 0, 0,
-+ sqr(_margin - omin + otmax) * _marginWt, false);
-+ else if (omax < otmin)
-+ _ranges[i].weightedAxis(i, vmin - _margin, vmax + _margin, 0, 0, 0, 0, 0,
-+ sqr(_margin - otmin + omax) * _marginWt, false);
-+ else
-+ _ranges[i].exclude_with_margins(vmin, vmax, i);
-+ anyhits = true;
-+ }
-+ if (anyhits)
-+ isCol = true;
-+ }
-+ else // no sub-boxes
-+ {
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ dbgout->setenv(1, reinterpret_cast<void *>(-1));
-+#endif
-+ isCol = true;
-+ if (omin > otmax)
-+ _ranges[i].weightedAxis(i, vmin - _margin, vmax + _margin, 0, 0, 0, 0, 0,
-+ sqr(_margin - omin + otmax) * _marginWt, false);
-+ else if (omax < otmin)
-+ _ranges[i].weightedAxis(i, vmin - _margin, vmax + _margin, 0, 0, 0, 0, 0,
-+ sqr(_margin - otmin + omax) * _marginWt, false);
-+ else
-+ _ranges[i].exclude_with_margins(vmin, vmax, i);
-+
-+ }
-+ }
-+ }
-+ bool res = true;
-+ if (cslot->exclGlyph() > 0 && gc.check(cslot->exclGlyph()) && !isExclusion)
-+ {
-+ // Set up the bogus slot representing the exclusion glyph.
-+ Slot *exclSlot = seg->newSlot();
-+ exclSlot->setGlyph(seg, cslot->exclGlyph());
-+ Position exclOrigin(slot->origin() + cslot->exclOffset());
-+ exclSlot->origin(exclOrigin);
-+ res &= mergeSlot(seg, exclSlot, currShift, isAfter, sameCluster, isCol, true, dbgout );
-+ seg->freeSlot(exclSlot);
-+ }
-+ hasCol |= isCol;
-+ return res;
-+
-+} // end of ShiftCollider::mergeSlot
-+
-+
-+// Figure out where to move the target glyph to, and return the amount to shift by.
-+Position ShiftCollider::resolve(GR_MAYBE_UNUSED Segment *seg, bool &isCol, GR_MAYBE_UNUSED json * const dbgout)
-+{
-+ float tbase;
-+ float totalCost = (float)(std::numeric_limits<float>::max() / 2);
-+ Position resultPos = Position(0, 0);
-+#if !defined GRAPHITE2_NTRACING
-+ int bestAxis = -1;
-+ if (dbgout)
-+ {
-+ outputJsonDbgStartSlot(dbgout, seg);
-+ *dbgout << "vectors" << json::array;
-+ }
-+#endif
-+ isCol = true;
-+ for (int i = 0; i < 4; ++i)
-+ {
-+ float bestCost = -1;
-+ float bestPos;
-+ // Calculate the margin depending on whether we are moving diagonally or not:
-+ switch (i) {
-+ case 0 : // x direction
-+ tbase = _currOffset.x;
-+ break;
-+ case 1 : // y direction
-+ tbase = _currOffset.y;
-+ break;
-+ case 2 : // sum (negatively-sloped diagonals)
-+ tbase = _currOffset.x + _currOffset.y;
-+ break;
-+ case 3 : // diff (positively-sloped diagonals)
-+ tbase = _currOffset.x - _currOffset.y;
-+ break;
-+ }
-+ Position testp;
-+ bestPos = _ranges[i].closest(0, bestCost) - tbase; // Get the best relative position
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ outputJsonDbgOneVector(dbgout, seg, i, tbase, bestCost, bestPos) ;
-+#endif
-+ if (bestCost >= 0.0f)
-+ {
-+ isCol = false;
-+ switch (i) {
-+ case 0 : testp = Position(bestPos, _currShift.y); break;
-+ case 1 : testp = Position(_currShift.x, bestPos); break;
-+ case 2 : testp = Position(0.5f * (_currShift.x - _currShift.y + bestPos), 0.5f * (_currShift.y - _currShift.x + bestPos)); break;
-+ case 3 : testp = Position(0.5f * (_currShift.x + _currShift.y + bestPos), 0.5f * (_currShift.x + _currShift.y - bestPos)); break;
-+ }
-+ if (bestCost < totalCost - 0.01f)
-+ {
-+ totalCost = bestCost;
-+ resultPos = testp;
-+#if !defined GRAPHITE2_NTRACING
-+ bestAxis = i;
-+#endif
-+ }
-+ }
-+ } // end of loop over 4 directions
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ outputJsonDbgEndSlot(dbgout, resultPos, bestAxis, isCol);
-+#endif
-+
-+ return resultPos;
-+
-+} // end of ShiftCollider::resolve
-+
-+
-+#if !defined GRAPHITE2_NTRACING
-+
-+void ShiftCollider::outputJsonDbg(json * const dbgout, Segment *seg, int axis)
-+{
-+ int axisMax = axis;
-+ if (axis < 0) // output all axes
-+ {
-+ *dbgout << "gid" << _target->gid()
-+ << "limit" << _limit
-+ << "target" << json::object
-+ << "origin" << _target->origin()
-+ << "margin" << _margin
-+ << "bbox" << seg->theGlyphBBoxTemporary(_target->gid())
-+ << "slantbox" << seg->getFace()->glyphs().slant(_target->gid())
-+ << json::close; // target object
-+ *dbgout << "ranges" << json::array;
-+ axis = 0;
-+ axisMax = 3;
-+ }
-+ for (int iAxis = axis; iAxis <= axisMax; ++iAxis)
-+ {
-+ *dbgout << json::flat << json::array << _ranges[iAxis].position();
-+ for (Zones::const_iterator s = _ranges[iAxis].begin(), e = _ranges[iAxis].end(); s != e; ++s)
-+ *dbgout << json::flat << json::array
-+ << Position(s->x, s->xm) << s->sm << s->smx << s->c
-+ << json::close;
-+ *dbgout << json::close;
-+ }
-+ if (axis < axisMax) // looped through the _ranges array for all axes
-+ *dbgout << json::close; // ranges array
-+}
-+
-+void ShiftCollider::outputJsonDbgStartSlot(json * const dbgout, Segment *seg)
-+{
-+ *dbgout << json::object // slot - not closed till the end of the caller method
-+ << "slot" << objectid(dslot(seg, _target))
-+ << "gid" << _target->gid()
-+ << "limit" << _limit
-+ << "target" << json::object
-+ << "origin" << _origin
-+ << "currShift" << _currShift
-+ << "currOffset" << seg->collisionInfo(_target)->offset()
-+ << "bbox" << seg->theGlyphBBoxTemporary(_target->gid())
-+ << "slantBox" << seg->getFace()->glyphs().slant(_target->gid())
-+ << "fix" << "shift";
-+ *dbgout << json::close; // target object
-+}
-+
-+void ShiftCollider::outputJsonDbgEndSlot(GR_MAYBE_UNUSED json * const dbgout,
-+ Position resultPos, int bestAxis, bool isCol)
-+{
-+ *dbgout << json::close // vectors array
-+ << "result" << resultPos
-+ //<< "scraping" << _scraping[bestAxis]
-+ << "bestAxis" << bestAxis
-+ << "stillBad" << isCol
-+ << json::close; // slot object
-+}
-+
-+void ShiftCollider::outputJsonDbgOneVector(json * const dbgout, Segment *seg, int axis,
-+ float tleft, float bestCost, float bestVal)
-+{
-+ const char * label;
-+ switch (axis)
-+ {
-+ case 0: label = "x"; break;
-+ case 1: label = "y"; break;
-+ case 2: label = "sum (NE-SW)"; break;
-+ case 3: label = "diff (NW-SE)"; break;
-+ default: label = "???"; break;
-+ }
-+
-+ *dbgout << json::object // vector
-+ << "direction" << label
-+ << "targetMin" << tleft;
-+
-+ outputJsonDbgRemovals(dbgout, axis, seg);
-+
-+ *dbgout << "ranges";
-+ outputJsonDbg(dbgout, seg, axis);
-+
-+ *dbgout << "bestCost" << bestCost
-+ << "bestVal" << bestVal + tleft
-+ << json::close; // vectors object
-+}
-+
-+void ShiftCollider::outputJsonDbgRemovals(json * const dbgout, int axis, Segment *seg)
-+{
-+ *dbgout << "removals" << json::array;
-+ _ranges[axis].jsonDbgOut(seg);
-+ *dbgout << json::close; // removals array
-+}
-+
-+#endif // !defined GRAPHITE2_NTRACING
-+
-+
-+//// KERN-COLLIDER ////
-+
-+inline
-+static float localmax (float al, float au, float bl, float bu, float x)
-+{
-+ if (al < bl)
-+ { if (au < bu) return au < x ? au : x; }
-+ else if (au > bu) return bl < x ? bl : x;
-+ return x;
-+}
-+
-+inline
-+static float localmin(float al, float au, float bl, float bu, float x)
-+{
-+ if (bl > al)
-+ { if (bu > au) return bl > x ? bl : x; }
-+ else if (au > bu) return al > x ? al : x;
-+ return x;
-+}
-+
-+// Return the given edge of the glyph at height y, taking any slant box into account.
-+static float get_edge(Segment *seg, const Slot *s, const Position &shift, float y, float width, bool isRight)
-+{
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+ unsigned short gid = s->gid();
-+ float sx = s->origin().x + shift.x;
-+ float sy = s->origin().y + shift.y;
-+ uint8 numsub = gc.numSubBounds(gid);
-+ float res = isRight ? (float)-1e38 : (float)1e38;
-+
-+ if (numsub > 0)
-+ {
-+ for (int i = 0; i < numsub; ++i)
-+ {
-+ const BBox &sbb = gc.getSubBoundingBBox(gid, i);
-+ const SlantBox &ssb = gc.getSubBoundingSlantBox(gid, i);
-+ if (sy + sbb.yi > y + width / 2 || sy + sbb.ya < y - width / 2)
-+ continue;
-+ if (isRight)
-+ {
-+ float x = sx + sbb.xa;
-+ if (x > res)
-+ {
-+ float td = sx - sy + ssb.da + y;
-+ float ts = sx + sy + ssb.sa - y;
-+ x = localmax(td - width / 2, td + width / 2, ts - width / 2, ts + width / 2, x);
-+ if (x > res)
-+ res = x;
-+ }
-+ }
-+ else
-+ {
-+ float x = sx + sbb.xi;
-+ if (x < res)
-+ {
-+ float td = sx - sy + ssb.di + y;
-+ float ts = sx + sy + ssb.si - y;
-+ x = localmin(td - width / 2, td + width / 2, ts - width / 2, ts + width / 2, x);
-+ if (x < res)
-+ res = x;
-+ }
-+ }
-+ }
-+ }
-+ else
-+ {
-+ const BBox &bb = gc.getBoundingBBox(gid);
-+ const SlantBox &sb = gc.getBoundingSlantBox(gid);
-+ float td = sx - sy + y;
-+ float ts = sx + sy - y;
-+ if (isRight)
-+ res = localmax(td + sb.da - width / 2, td + sb.da + width / 2, ts + sb.sa - width / 2, ts + sb.sa + width / 2, sx + bb.xa);
-+ else
-+ res = localmin(td + sb.di - width / 2, td + sb.di + width / 2, ts + sb.si - width / 2, ts + sb.si + width / 2, sx + bb.xi);
-+ }
-+ return res;
-+}
-+
-+
-+bool KernCollider::initSlot(Segment *seg, Slot *aSlot, const Rect &limit, float margin,
-+ const Position &currShift, const Position &offsetPrev, int dir,
-+ float ymin, float ymax, GR_MAYBE_UNUSED json * const dbgout)
-+{
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+ const Slot *base = aSlot;
-+ // const Slot *last = aSlot;
-+ const Slot *s;
-+ int numSlices;
-+ while (base->attachedTo())
-+ base = base->attachedTo();
-+ if (margin < 10) margin = 10;
-+
-+ _limit = limit;
-+ _offsetPrev = offsetPrev; // kern from a previous pass
-+
-+ // Calculate the height of the glyph and how many horizontal slices to use.
-+ if (_maxy >= 1e37f)
-+ {
-+ _maxy = ymax;
-+ _miny = ymin;
-+ _sliceWidth = margin / 1.5f;
-+ numSlices = int((_maxy - _miny + 2) / (_sliceWidth / 1.5f) + 1.f); // +2 helps with rounding errors
-+ _edges.clear();
-+ _edges.insert(_edges.begin(), numSlices, (dir & 1) ? 1e38f : -1e38f);
-+ _xbound = (dir & 1) ? (float)1e38f : (float)-1e38f;
-+ }
-+ else if (_maxy != ymax || _miny != ymin)
-+ {
-+ if (_miny != ymin)
-+ {
-+ numSlices = int((ymin - _miny) / _sliceWidth - 1);
-+ _miny += numSlices * _sliceWidth;
-+ if (numSlices < 0)
-+ _edges.insert(_edges.begin(), -numSlices, (dir & 1) ? 1e38f : -1e38f);
-+ else if ((unsigned)numSlices < _edges.size()) // this shouldn't fire since we always grow the range
-+ {
-+ Vector<float>::iterator e = _edges.begin();
-+ while (numSlices--)
-+ ++e;
-+ _edges.erase(_edges.begin(), e);
-+ }
-+ }
-+ if (_maxy != ymax)
-+ {
-+ numSlices = int((ymax - _miny) / _sliceWidth + 1);
-+ _maxy = numSlices * _sliceWidth + _miny;
-+ if (numSlices > (int)_edges.size())
-+ _edges.insert(_edges.end(), numSlices - _edges.size(), (dir & 1) ? 1e38f : -1e38f);
-+ else if (numSlices < (int)_edges.size()) // this shouldn't fire since we always grow the range
-+ {
-+ while ((int)_edges.size() > numSlices)
-+ _edges.pop_back();
-+ }
-+ }
-+ }
-+ numSlices = _edges.size();
-+
-+#if !defined GRAPHITE2_NTRACING
-+ // Debugging
-+ _seg = seg;
-+ _slotNear.clear();
-+ _slotNear.insert(_slotNear.begin(), numSlices, NULL);
-+ _nearEdges.clear();
-+ _nearEdges.insert(_nearEdges.begin(), numSlices, (dir & 1) ? -1e38f : +1e38f);
-+#endif
-+
-+ // Determine the trailing edge of each slice (ie, left edge for a RTL glyph).
-+ for (s = base; s; s = s->nextInCluster(s))
-+ {
-+ SlotCollision *c = seg->collisionInfo(s);
-+ if (!gc.check(s->gid()))
-+ return false;
-+ const BBox &bs = gc.getBoundingBBox(s->gid());
-+ float x = s->origin().x + c->shift().x + ((dir & 1) ? bs.xi : bs.xa);
-+ // Loop over slices.
-+ // Note smin might not be zero if glyph s is not at the bottom of the cluster; similarly for smax.
-+ float toffset = c->shift().y - _miny + 1 + s->origin().y;
-+ int smin = max(0, int((bs.yi + toffset) / _sliceWidth));
-+ int smax = min(numSlices - 1, int((bs.ya + toffset) / _sliceWidth + 1));
-+ for (int i = smin; i <= smax; ++i)
-+ {
-+ float t;
-+ float y = _miny - 1 + (i + .5f) * _sliceWidth; // vertical center of slice
-+ if ((dir & 1) && x < _edges[i])
-+ {
-+ t = get_edge(seg, s, c->shift(), y, _sliceWidth, false);
-+ if (t < _edges[i])
-+ {
-+ _edges[i] = t;
-+ if (t < _xbound)
-+ _xbound = t;
-+ }
-+ }
-+ else if (!(dir & 1) && x > _edges[i])
-+ {
-+ t = get_edge(seg, s, c->shift(), y, _sliceWidth, true);
-+ if (t > _edges[i])
-+ {
-+ _edges[i] = t;
-+ if (t > _xbound)
-+ _xbound = t;
-+ }
-+ }
-+ }
-+ }
-+ _mingap = (float)1e38;
-+ _target = aSlot;
-+ _margin = margin;
-+ _currShift = currShift;
-+ return true;
-+} // end of KernCollider::initSlot
-+
-+
-+// Determine how much the target slot needs to kern away from the given slot.
-+// In other words, merge information from given slot's position with what the target slot knows
-+// about how it can kern.
-+// Return false if we know there is no collision, true if we think there might be one.
-+bool KernCollider::mergeSlot(Segment *seg, Slot *slot, const Position &currShift, float currSpace, int dir, GR_MAYBE_UNUSED json * const dbgout)
-+{
-+ int rtl = (dir & 1) * 2 - 1;
-+ if (!seg->getFace()->glyphs().check(slot->gid()))
-+ return false;
-+ const Rect &bb = seg->theGlyphBBoxTemporary(slot->gid());
-+ const float sx = slot->origin().x + currShift.x;
-+ float x = sx + (rtl > 0 ? bb.tr.x : bb.bl.x);
-+ // this isn't going to reduce _mingap so skip
-+ if ((rtl > 0 && x < _xbound - _mingap - currSpace) || (rtl <= 0 && x > _xbound + _mingap + currSpace))
-+ return false;
-+
-+ const float sy = slot->origin().y + currShift.y;
-+ int smin = max(0, int((bb.bl.y + (1 - _miny + sy)) / _sliceWidth + 1));
-+ int smax = min((int)_edges.size() - 1, int((bb.tr.y + (1 - _miny + sy)) / _sliceWidth + 1));
-+ bool collides = false;
-+
-+ for (int i = smin; i <= smax; ++i)
-+ {
-+ float t;
-+ float y = (float)(_miny - 1 + (i + .5f) * _sliceWidth); // vertical center of slice
-+ if (x * rtl > _edges[i] * rtl - _mingap - currSpace)
-+ {
-+ // 2 * currSpace to account for the space that is already separating them and the space we want to add
-+ float m = get_edge(seg, slot, currShift, y, _sliceWidth, rtl > 0) + 2 * rtl * currSpace;
-+ t = rtl * (_edges[i] - m);
-+ // Check slices above and below (if any).
-+ if (i < (int)_edges.size() - 1) t = min(t, rtl * (_edges[i+1] - m));
-+ if (i > 0) t = min(t, rtl * (_edges[i-1] - m));
-+ // _mingap is positive to shrink
-+ if (t < _mingap)
-+ {
-+ _mingap = t;
-+ collides = true;
-+ }
-+#if !defined GRAPHITE2_NTRACING
-+ // Debugging - remember the closest neighboring edge for this slice.
-+ if (rtl * m > rtl * _nearEdges[i])
-+ {
-+ _slotNear[i] = slot;
-+ _nearEdges[i] = m;
-+ }
-+#endif
-+ }
-+ }
-+ return collides; // note that true is not a necessarily reliable value
-+
-+} // end of KernCollider::mergeSlot
-+
-+
-+// Return the amount to kern by.
-+Position KernCollider::resolve(GR_MAYBE_UNUSED Segment *seg, GR_MAYBE_UNUSED Slot *slot,
-+ int dir, float margin, GR_MAYBE_UNUSED json * const dbgout)
-+{
-+ float resultNeeded = (1 - 2 * (dir & 1)) * (_mingap - margin);
-+ float result = min(_limit.tr.x - _offsetPrev.x, max(resultNeeded, _limit.bl.x - _offsetPrev.x));
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ {
-+ *dbgout << json::object // slot
-+ << "slot" << objectid(dslot(seg, _target))
-+ << "gid" << _target->gid()
-+ << "margin" << _margin
-+ << "limit" << _limit
-+ << "miny" << _miny
-+ << "maxy" << _maxy
-+ << "slicewidth" << _sliceWidth
-+ << "target" << json::object
-+ << "origin" << _target->origin()
-+ //<< "currShift" << _currShift
-+ << "offsetPrev" << _offsetPrev
-+ << "bbox" << seg->theGlyphBBoxTemporary(_target->gid())
-+ << "slantBox" << seg->getFace()->glyphs().slant(_target->gid())
-+ << "fix" << "kern"
-+ << json::close; // target object
-+
-+ *dbgout << "slices" << json::array;
-+ for (int is = 0; is < (int)_edges.size(); is++)
-+ {
-+ *dbgout << json::flat << json::object
-+ << "i" << is
-+ << "targetEdge" << _edges[is]
-+ << "neighbor" << objectid(dslot(seg, _slotNear[is]))
-+ << "nearEdge" << _nearEdges[is]
-+ << json::close;
-+ }
-+ *dbgout << json::close; // slices array
-+
-+ *dbgout
-+ << "xbound" << _xbound
-+ << "minGap" << _mingap
-+ << "needed" << resultNeeded
-+ << "result" << result
-+ << "stillBad" << (result != resultNeeded)
-+ << json::close; // slot object
-+ }
-+#endif
-+
-+ return Position(result, 0.);
-+
-+} // end of KernCollider::resolve
-+
-+void KernCollider::shift(const Position &mv, int dir)
-+{
-+ for (Vector<float>::iterator e = _edges.begin(); e != _edges.end(); ++e)
-+ *e += mv.x;
-+ _xbound += (1 - 2 * (dir & 1)) * mv.x;
-+}
-+
-+//// SLOT-COLLISION ////
-+
-+// Initialize the collision attributes for the given slot.
-+SlotCollision::SlotCollision(Segment *seg, Slot *slot)
-+{
-+ initFromSlot(seg, slot);
-+}
-+
-+void SlotCollision::initFromSlot(Segment *seg, Slot *slot)
-+{
-+ // Initialize slot attributes from glyph attributes.
-+ // The order here must match the order in the grcompiler code,
-+ // GrcSymbolTable::AssignInternalGlyphAttrIDs.
-+ uint16 gid = slot->gid();
-+ uint16 aCol = seg->silf()->aCollision(); // flags attr ID
-+ const GlyphFace * glyphFace = seg->getFace()->glyphs().glyphSafe(gid);
-+ if (!glyphFace)
-+ return;
-+ const sparse &p = glyphFace->attrs();
-+ _flags = p[aCol];
-+ _limit = Rect(Position(p[aCol+1], p[aCol+2]),
-+ Position(p[aCol+3], p[aCol+4]));
-+ _margin = p[aCol+5];
-+ _marginWt = p[aCol+6];
-+
-+ _seqClass = p[aCol+7];
-+ _seqProxClass = p[aCol+8];
-+ _seqOrder = p[aCol+9];
-+ _seqAboveXoff = p[aCol+10];
-+ _seqAboveWt = p[aCol+11];
-+ _seqBelowXlim = p[aCol+12];
-+ _seqBelowWt = p[aCol+13];
-+ _seqValignHt = p[aCol+14];
-+ _seqValignWt = p[aCol+15];
-+
-+ // These attributes do not have corresponding glyph attribute:
-+ _exclGlyph = 0;
-+ _exclOffset = Position(0, 0);
-+}
-+
-+float SlotCollision::getKern(int dir) const
-+{
-+ if ((_flags & SlotCollision::COLL_KERN) != 0)
-+ return float(_shift.x * ((dir & 1) ? -1 : 1));
-+ else
-+ return 0;
-+}
-+
-diff --git a/gfx/graphite2/src/Decompressor.cpp b/gfx/graphite2/src/Decompressor.cpp
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/Decompressor.cpp
-@@ -0,0 +1,113 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2015, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#include <cassert>
-+
-+#include "inc/Decompressor.h"
-+#include "inc/Compression.h"
-+
-+using namespace lz4;
-+
-+namespace {
-+
-+inline
-+u32 read_literal(u8 const * &s, u8 const * const e, u32 l) {
-+ if (l == 15 && s != e)
-+ {
-+ u8 b = 0;
-+ do { l += b = *s++; } while(b==0xff && s != e);
-+ }
-+ return l;
-+}
-+
-+bool read_sequence(u8 const * &src, u8 const * const end, u8 const * &literal, u32 & literal_len, u32 & match_len, u32 & match_dist)
-+{
-+ u8 const token = *src++;
-+
-+ literal_len = read_literal(src, end, token >> 4);
-+ literal = src;
-+ src += literal_len;
-+
-+ if (src > end - 2)
-+ return false;
-+
-+ match_dist = *src++;
-+ match_dist |= *src++ << 8;
-+ match_len = read_literal(src, end, token & 0xf);
-+
-+ return src <= end-5;
-+}
-+
-+}
-+
-+int lz4::decompress(void const *in, size_t in_size, void *out, size_t out_size)
-+{
-+ if (out_size <= in_size || in_size < sizeof(unsigned long)+1)
-+ return -1;
-+
-+ u8 const * src = static_cast<u8 const *>(in),
-+ * literal = 0,
-+ * const src_end = src + in_size;
-+
-+ u8 * dst = static_cast<u8*>(out),
-+ * const dst_end = dst + out_size;
-+
-+ u32 literal_len = 0,
-+ match_len = 0,
-+ match_dist = 0;
-+
-+ while (read_sequence(src, src_end, literal, literal_len, match_len, match_dist))
-+ {
-+ if (literal_len != 0)
-+ {
-+ // Copy in literal. At this point the last full sequence must be at
-+ // least MINMATCH + 5 from the end of the output buffer.
-+ if (dst + align(literal_len) > dst_end - (MINMATCH+5))
-+ return -1;
-+ dst = overrun_copy(dst, literal, literal_len);
-+ }
-+
-+ // Copy, possibly repeating, match from earlier in the
-+ // decoded output.
-+ u8 const * const pcpy = dst - match_dist;
-+ if (pcpy < static_cast<u8*>(out)
-+ || dst + match_len + MINMATCH > dst_end - 5)
-+ return -1;
-+ if (dst > pcpy+sizeof(unsigned long)
-+ && dst + align(match_len + MINMATCH) <= dst_end)
-+ dst = overrun_copy(dst, pcpy, match_len + MINMATCH);
-+ else
-+ dst = safe_copy(dst, pcpy, match_len + MINMATCH);
-+ }
-+
-+ if (literal + literal_len > src_end
-+ || dst + literal_len > dst_end)
-+ return -1;
-+ dst = fast_copy(dst, literal, literal_len);
-+
-+ return dst - (u8*)out;
-+}
-+
-diff --git a/gfx/graphite2/src/Face.cpp b/gfx/graphite2/src/Face.cpp
---- a/gfx/graphite2/src/Face.cpp
-+++ b/gfx/graphite2/src/Face.cpp
-@@ -23,28 +23,39 @@ Alternatively, the contents of this file
- Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- #include <cstring>
- #include "graphite2/Segment.h"
- #include "inc/CmapCache.h"
- #include "inc/debug.h"
-+#include "inc/Decompressor.h"
- #include "inc/Endian.h"
- #include "inc/Face.h"
- #include "inc/FileFace.h"
- #include "inc/GlyphFace.h"
- #include "inc/json.h"
- #include "inc/SegCacheStore.h"
- #include "inc/Segment.h"
- #include "inc/NameTable.h"
- #include "inc/Error.h"
-
- using namespace graphite2;
-
-+namespace
-+{
-+enum compression
-+{
-+ NONE,
-+ LZ4
-+};
-+
-+}
-+
- Face::Face(const void* appFaceHandle/*non-NULL*/, const gr_face_ops & ops)
- : m_appFaceHandle(appFaceHandle),
- m_pFileFace(NULL),
- m_pGlyphFaceCache(NULL),
- m_cmap(NULL),
- m_pNames(NULL),
- m_logger(NULL),
- m_error(0), m_errcntxt(0),
-@@ -79,55 +90,59 @@ float Face::default_glyph_advance(const
-
- bool Face::readGlyphs(uint32 faceOptions)
- {
- Error e;
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::category _glyph_cat(tele.glyph);
- #endif
- error_context(EC_READGLYPHS);
-+ m_pGlyphFaceCache = new GlyphCache(*this, faceOptions);
-+
-+ if (e.test(!m_pGlyphFaceCache, E_OUTOFMEM)
-+ || e.test(m_pGlyphFaceCache->numGlyphs() == 0, E_NOGLYPHS)
-+ || e.test(m_pGlyphFaceCache->unitsPerEm() == 0, E_BADUPEM))
-+ {
-+ return error(e);
-+ }
-+
- if (faceOptions & gr_face_cacheCmap)
- m_cmap = new CachedCmap(*this);
- else
- m_cmap = new DirectCmap(*this);
--
-- m_pGlyphFaceCache = new GlyphCache(*this, faceOptions);
-- if (e.test(!m_pGlyphFaceCache, E_OUTOFMEM)
-- || e.test(m_pGlyphFaceCache->numGlyphs() == 0, E_NOGLYPHS)
-- || e.test(m_pGlyphFaceCache->unitsPerEm() == 0, E_BADUPEM)
-- || e.test(!m_cmap, E_OUTOFMEM) || e.test(!*m_cmap, E_BADCMAP))
-- {
-+ if (e.test(!m_cmap, E_OUTOFMEM) || e.test(!*m_cmap, E_BADCMAP))
- return error(e);
-- }
-
- if (faceOptions & gr_face_preloadGlyphs)
- nameTable(); // preload the name table along with the glyphs.
-
- return true;
- }
-
- bool Face::readGraphite(const Table & silf)
- {
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::category _silf_cat(tele.silf);
- #endif
- Error e;
- error_context(EC_READSILF);
- const byte * p = silf;
-- if (e.test(!p, E_NOSILF)) return error(e);
-+ if (e.test(!p, E_NOSILF) || e.test(silf.size() < 20, E_BADSIZE)) return error(e);
-
- const uint32 version = be::read<uint32>(p);
- if (e.test(version < 0x00020000, E_TOOOLD)) return error(e);
- if (version >= 0x00030000)
- be::skip<uint32>(p); // compilerVersion
- m_numSilf = be::read<uint16>(p);
-+
- be::skip<uint16>(p); // reserved
-
- bool havePasses = false;
- m_silfs = new Silf[m_numSilf];
-+ if (e.test(!m_silfs, E_OUTOFMEM)) return error(e);
- for (int i = 0; i < m_numSilf; i++)
- {
- error_context(EC_ASILF + (i << 8));
- const uint32 offset = be::read<uint32>(p),
- next = i == m_numSilf - 1 ? silf.size() : be::peek<uint32>(p);
- if (e.test(next > silf.size() || offset >= next, E_BADSIZE))
- return error(e);
-
-@@ -153,29 +168,38 @@ bool Face::runGraphite(Segment *seg, con
- if (dbgout)
- {
- *dbgout << json::object
- << "id" << objectid(seg)
- << "passes" << json::array;
- }
- #endif
-
-- bool res = aSilf->runGraphite(seg, 0, aSilf->justificationPass(), true);
-+// if ((seg->dir() & 1) != aSilf->dir())
-+// seg->reverseSlots();
-+ if ((seg->dir() & 3) == 3 && aSilf->bidiPass() == 0xFF)
-+ seg->doMirror(aSilf->aMirror());
-+ bool res = aSilf->runGraphite(seg, 0, aSilf->positionPass(), true);
- if (res)
-- res = aSilf->runGraphite(seg, aSilf->positionPass(), aSilf->numPasses(), false);
-+ {
-+ seg->associateChars(0, seg->charInfoCount());
-+ if (aSilf->flags() & 0x20)
-+ res &= seg->initCollisions();
-+ res &= aSilf->runGraphite(seg, aSilf->positionPass(), aSilf->numPasses(), false);
-+ }
-
- #if !defined GRAPHITE2_NTRACING
- if (dbgout)
- {
-+ seg->positionSlots(0, 0, 0, aSilf->dir());
- *dbgout << json::item
- << json::close // Close up the passes array
- << "output" << json::array;
- for(Slot * s = seg->first(); s; s = s->next())
- *dbgout << dslot(seg, s);
-- seg->finalise(0); // Call this here to fix up charinfo back indexes.
- *dbgout << json::close
- << "advance" << seg->advance()
- << "chars" << json::array;
- for(size_t i = 0, n = seg->charInfoCount(); i != n; ++i)
- *dbgout << json::flat << *seg->charinfo(i);
- *dbgout << json::close // Close up the chars array
- << json::close; // Close up the segment object
- }
-@@ -208,17 +232,19 @@ uint16 Face::findPseudo(uint32 uid) cons
- }
-
- uint16 Face::getGlyphMetric(uint16 gid, uint8 metric) const
- {
- switch (metrics(metric))
- {
- case kgmetAscent : return m_ascent;
- case kgmetDescent : return m_descent;
-- default: return glyphs().glyph(gid)->getMetric(metric);
-+ default:
-+ if (gid >= glyphs().numGlyphs()) return 0;
-+ return glyphs().glyph(gid)->getMetric(metric);
- }
- }
-
- void Face::takeFileFace(FileFace* pFileFace GR_MAYBE_UNUSED/*takes ownership*/)
- {
- #ifndef GRAPHITE2_NFILEFACE
- if (m_pFileFace==pFileFace)
- return;
-@@ -240,30 +266,100 @@ NameTable * Face::nameTable() const
- uint16 Face::languageForLocale(const char * locale) const
- {
- nameTable();
- if (m_pNames)
- return m_pNames->getLanguageId(locale);
- return 0;
- }
-
--Face::Table::Table(const Face & face, const Tag n) throw()
--: _f(&face)
-+
-+
-+Face::Table::Table(const Face & face, const Tag n, uint32 version) throw()
-+: _f(&face), _compressed(false)
- {
- size_t sz = 0;
-- _p = reinterpret_cast<const byte *>((*_f->m_ops.get_table)(_f->m_appFaceHandle, n, &sz));
-+ _p = static_cast<const byte *>((*_f->m_ops.get_table)(_f->m_appFaceHandle, n, &sz));
- _sz = uint32(sz);
-+
- if (!TtfUtil::CheckTable(n, _p, _sz))
- {
- this->~Table(); // Make sure we release the table buffer even if the table filed it's checks
-- _p = 0; _sz = 0;
-+ return;
- }
-+
-+ if (be::peek<uint32>(_p) >= version)
-+ decompress();
-+}
-+
-+void Face::Table::releaseBuffers()
-+{
-+ if (_compressed)
-+ free(const_cast<byte *>(_p));
-+ else if (_p && _f->m_ops.release_table)
-+ (*_f->m_ops.release_table)(_f->m_appFaceHandle, _p);
-+ _p = 0; _sz = 0;
- }
-
- Face::Table & Face::Table::operator = (const Table & rhs) throw()
- {
- if (_p == rhs._p) return *this;
-
- this->~Table();
- new (this) Table(rhs);
- return *this;
- }
-
-+Error Face::Table::decompress()
-+{
-+ Error e;
-+ if (e.test(_sz < 5 * sizeof(uint32), E_BADSIZE))
-+ return e;
-+ byte * uncompressed_table = 0;
-+ size_t uncompressed_size = 0;
-+
-+ const byte * p = _p;
-+ const uint32 version = be::read<uint32>(p); // Table version number.
-+
-+ // The scheme is in the top 5 bits of the 1st uint32.
-+ const uint32 hdr = be::read<uint32>(p);
-+ switch(compression(hdr >> 27))
-+ {
-+ case NONE: return e;
-+
-+ case LZ4:
-+ {
-+ uncompressed_size = hdr & 0x07ffffff;
-+ uncompressed_table = gralloc<byte>(uncompressed_size);
-+ if (!e.test(!uncompressed_table, E_OUTOFMEM))
-+ // coverity[forward_null : FALSE] - uncompressed_table has been checked so can't be null
-+ // coverity[checked_return : FALSE] - we test e later
-+ e.test(lz4::decompress(p, _sz - 2*sizeof(uint32), uncompressed_table, uncompressed_size) != signed(uncompressed_size), E_SHRINKERFAILED);
-+ break;
-+ }
-+
-+ default:
-+ e.error(E_BADSCHEME);
-+ };
-+
-+ // Check the uncompressed version number against the original.
-+ if (!e)
-+ // coverity[forward_null : FALSE] - uncompressed_table has already been tested so can't be null
-+ // coverity[checked_return : FALSE] - we test e later
-+ e.test(be::peek<uint32>(uncompressed_table) != version, E_SHRINKERFAILED);
-+
-+ // Tell the provider to release the compressed form since were replacing
-+ // it anyway.
-+ releaseBuffers();
-+
-+ if (e)
-+ {
-+ free(uncompressed_table);
-+ uncompressed_table = 0;
-+ uncompressed_size = 0;
-+ }
-+
-+ _p = uncompressed_table;
-+ _sz = uncompressed_size;
-+ _compressed = true;
-+
-+ return e;
-+}
-diff --git a/gfx/graphite2/src/FeatureMap.cpp b/gfx/graphite2/src/FeatureMap.cpp
---- a/gfx/graphite2/src/FeatureMap.cpp
-+++ b/gfx/graphite2/src/FeatureMap.cpp
-@@ -126,60 +126,61 @@ bool FeatureMap::readFeats(const Face &
- unsigned short bits = 0; //to cause overflow on first Feature
-
- for (int i = 0, ie = m_numFeats; i != ie; i++)
- {
- const uint32 label = version < 0x00020000 ? be::read<uint16>(p) : be::read<uint32>(p);
- const uint16 num_settings = be::read<uint16>(p);
- if (version >= 0x00020000)
- be::skip<uint16>(p);
-- const byte * const feat_setts = feat_start + be::read<uint32>(p);
-+ const uint32 settings_offset = be::read<uint32>(p);
- const uint16 flags = be::read<uint16>(p),
- uiName = be::read<uint16>(p);
-
-- if (feat_setts + num_settings * FEATURE_SETTING_SIZE > feat_end)
-+ if (settings_offset > size_t(feat_end - feat_start)
-+ || settings_offset + num_settings * FEATURE_SETTING_SIZE > size_t(feat_end - feat_start))
- {
- free(defVals);
- return false;
- }
-
- FeatureSetting *uiSet;
- uint32 maxVal;
- if (num_settings != 0)
- {
- uiSet = gralloc<FeatureSetting>(num_settings);
- if (!uiSet)
- {
- free(defVals);
- return false;
- }
-- maxVal = readFeatureSettings(feat_setts, uiSet, num_settings);
-+ maxVal = readFeatureSettings(feat_start + settings_offset, uiSet, num_settings);
- defVals[i] = uiSet[0].value();
- }
- else
- {
- uiSet = 0;
- maxVal = 0xffffffff;
- defVals[i] = 0;
- }
-
- ::new (m_feats + i) FeatureRef (face, bits, maxVal,
- label, uiName, flags,
- uiSet, num_settings);
- }
-- m_defaultFeatures = new Features(bits/(sizeof(uint32)*8) + 1, *this);
-+ new (&m_defaultFeatures) Features(bits/(sizeof(uint32)*8) + 1, *this);
- m_pNamedFeats = new NameAndFeatureRef[m_numFeats];
-- if (!m_defaultFeatures || !m_pNamedFeats)
-+ if (!m_pNamedFeats)
- {
- free(defVals);
- return false;
- }
- for (int i = 0; i < m_numFeats; ++i)
- {
-- m_feats[i].applyValToFeature(defVals[i], *m_defaultFeatures);
-+ m_feats[i].applyValToFeature(defVals[i], m_defaultFeatures);
- m_pNamedFeats[i] = m_feats+i;
- }
-
- free(defVals);
-
- qsort(m_pNamedFeats, m_numFeats, sizeof(NameAndFeatureRef), &cmpNameAndFeatures);
-
- return true;
-@@ -209,17 +210,17 @@ bool SillMap::readSill(const Face & face
- if (sill.size() < m_numLanguages * 8U + 12) return false;
-
- for (int i = 0; i < m_numLanguages; i++)
- {
- uint32 langid = be::read<uint32>(p);
- uint16 numSettings = be::read<uint16>(p);
- uint16 offset = be::read<uint16>(p);
- if (offset + 8U * numSettings > sill.size() && numSettings > 0) return false;
-- Features* feats = new Features(*m_FeatureMap.m_defaultFeatures);
-+ Features* feats = new Features(m_FeatureMap.m_defaultFeatures);
- if (!feats) return false;
- const byte *pLSet = sill + offset;
-
- // Apply langauge specific settings
- for (int j = 0; j < numSettings; j++)
- {
- uint32 name = be::read<uint32>(pLSet);
- uint16 val = be::read<uint16>(pLSet);
-@@ -245,17 +246,17 @@ Features* SillMap::cloneFeatures(uint32
- // the number of languages in a font is usually small e.g. 8 in Doulos
- // so this loop is not very expensive
- for (uint16 i = 0; i < m_numLanguages; i++)
- {
- if (m_langFeats[i].m_lang == langname)
- return new Features(*m_langFeats[i].m_pFeatures);
- }
- }
-- return new Features (*m_FeatureMap.m_defaultFeatures);
-+ return new Features (m_FeatureMap.m_defaultFeatures);
- }
-
-
-
- const FeatureRef *FeatureMap::findFeatureRef(uint32 name) const
- {
- NameAndFeatureRef *it;
-
-diff --git a/gfx/graphite2/src/FileFace.cpp b/gfx/graphite2/src/FileFace.cpp
---- a/gfx/graphite2/src/FileFace.cpp
-+++ b/gfx/graphite2/src/FileFace.cpp
-@@ -55,18 +55,22 @@ FileFace::FileFace(const char *filename)
- if (fread(_header_tbl, 1, tbl_len, _file) != tbl_len) return;
- if (!TtfUtil::CheckHeader(_header_tbl)) return;
- }
-
- // Get the table directory
- if (!TtfUtil::GetTableDirInfo(_header_tbl, tbl_offset, tbl_len)) return;
- _table_dir = (TtfUtil::Sfnt::OffsetSubTable::Entry*)gralloc<char>(tbl_len);
- if (fseek(_file, tbl_offset, SEEK_SET)) return;
-- if (_table_dir)
-- if (fread(_table_dir, 1, tbl_len, _file) != tbl_len) return;
-+ if (_table_dir && fread(_table_dir, 1, tbl_len, _file) != tbl_len)
-+ {
-+ free(_table_dir);
-+ _table_dir = NULL;
-+ }
-+ return;
- }
-
- FileFace::~FileFace()
- {
- free(_table_dir);
- free(_header_tbl);
- if (_file)
- fclose(_file);
-@@ -78,17 +82,17 @@ const void *FileFace::get_table_fn(const
- if (appFaceHandle == 0) return 0;
- const FileFace & file_face = *static_cast<const FileFace *>(appFaceHandle);
-
- void *tbl;
- size_t tbl_offset, tbl_len;
- if (!TtfUtil::GetTableInfo(name, file_face._header_tbl, file_face._table_dir, tbl_offset, tbl_len))
- return 0;
-
-- if (tbl_offset + tbl_len > file_face._file_len
-+ if (tbl_offset > file_face._file_len || tbl_len > file_face._file_len - tbl_offset
- || fseek(file_face._file, tbl_offset, SEEK_SET) != 0)
- return 0;
-
- tbl = malloc(tbl_len);
- if (fread(tbl, 1, tbl_len, file_face._file) != tbl_len)
- {
- free(tbl);
- return 0;
-diff --git a/gfx/graphite2/src/GlyphCache.cpp b/gfx/graphite2/src/GlyphCache.cpp
---- a/gfx/graphite2/src/GlyphCache.cpp
-+++ b/gfx/graphite2/src/GlyphCache.cpp
-@@ -26,16 +26,17 @@ of the License or (at your option) any l
- */
- #include "graphite2/Font.h"
-
- #include "inc/Main.h"
- #include "inc/Face.h" //for the tags
- #include "inc/GlyphCache.h"
- #include "inc/GlyphFace.h"
- #include "inc/Endian.h"
-+#include "inc/bits.h"
-
- using namespace graphite2;
-
- namespace
- {
- // Iterator over version 1 or 2 glat entries which consist of a series of
- // +-+-+-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+-+-+-+-+
- // v1 |k|n|v1 |v2 |...|vN | or v2 | k | n |v1 |v2 |...|vN |
-@@ -56,99 +57,127 @@ namespace
- if (_n == run()) advance_entry();
- return *this;
- }
- _glat_iterator<W> operator ++ (int) { _glat_iterator<W> tmp(*this); operator++(); return tmp; }
-
- // This is strictly a >= operator. A true == operator could be
- // implemented that test for overlap but it would be more expensive a
- // test.
-- bool operator == (const _glat_iterator<W> & rhs) { return _v >= rhs._e; }
-+ bool operator == (const _glat_iterator<W> & rhs) { return _v >= rhs._e - 1; }
- bool operator != (const _glat_iterator<W> & rhs) { return !operator==(rhs); }
-
- value_type operator * () const {
- return value_type(key(), be::peek<uint16>(_v));
- }
-
- protected:
- const byte * _e, * _v;
-- ptrdiff_t _n;
-+ size_t _n;
- };
-
- typedef _glat_iterator<uint8> glat_iterator;
- typedef _glat_iterator<uint16> glat2_iterator;
- }
-
-+const SlantBox SlantBox::empty = {0,0,0,0};
-+
-
- class GlyphCache::Loader
- {
- public:
- Loader(const Face & face, const bool dumb_font); //return result indicates success. Do not use if failed.
-
- operator bool () const throw();
- unsigned short int units_per_em() const throw();
- unsigned short int num_glyphs() const throw();
- unsigned short int num_attrs() const throw();
-+ bool has_boxes() const throw();
-
-- const GlyphFace * read_glyph(unsigned short gid, GlyphFace &) const throw();
-+ const GlyphFace * read_glyph(unsigned short gid, GlyphFace &, int *numsubs) const throw();
-+ GlyphBox * read_box(uint16 gid, GlyphBox *curr, const GlyphFace & face) const throw();
-
- CLASS_NEW_DELETE;
- private:
- Face::Table _head,
- _hhea,
- _hmtx,
- _glyf,
- _loca,
- m_pGlat,
- m_pGloc;
-
- bool _long_fmt;
-+ bool _has_boxes;
- unsigned short _num_glyphs_graphics, //i.e. boundary box and advance
- _num_glyphs_attributes,
- _num_attrs; // number of glyph attributes per glyph
- };
-
-
-
- GlyphCache::GlyphCache(const Face & face, const uint32 face_options)
- : _glyph_loader(new Loader(face, bool(face_options & gr_face_dumbRendering))),
- _glyphs(_glyph_loader && *_glyph_loader ? grzeroalloc<const GlyphFace *>(_glyph_loader->num_glyphs()) : 0),
-+ _boxes(_glyph_loader && _glyph_loader->has_boxes() ? grzeroalloc<GlyphBox *>(_glyph_loader->num_glyphs()) : 0),
- _num_glyphs(_glyphs ? _glyph_loader->num_glyphs() : 0),
- _num_attrs(_glyphs ? _glyph_loader->num_attrs() : 0),
- _upem(_glyphs ? _glyph_loader->units_per_em() : 0)
- {
- if ((face_options & gr_face_preloadGlyphs) && _glyph_loader && _glyphs)
- {
-+ int numsubs = 0;
- GlyphFace * const glyphs = new GlyphFace [_num_glyphs];
- if (!glyphs)
- return;
-
- // The 0 glyph is definately required.
-- _glyphs[0] = _glyph_loader->read_glyph(0, glyphs[0]);
-+ _glyphs[0] = _glyph_loader->read_glyph(0, glyphs[0], &numsubs);
-
- // glyphs[0] has the same address as the glyphs array just allocated,
- // thus assigning the &glyphs[0] to _glyphs[0] means _glyphs[0] points
- // to the entire array.
- const GlyphFace * loaded = _glyphs[0];
- for (uint16 gid = 1; loaded && gid != _num_glyphs; ++gid)
-- _glyphs[gid] = loaded = _glyph_loader->read_glyph(gid, glyphs[gid]);
-+ _glyphs[gid] = loaded = _glyph_loader->read_glyph(gid, glyphs[gid], &numsubs);
-
- if (!loaded)
- {
- _glyphs[0] = 0;
- delete [] glyphs;
- }
-+ else if (numsubs > 0)
-+ {
-+ GlyphBox * boxes = (GlyphBox *)gralloc<char>(_num_glyphs * sizeof(GlyphBox) + numsubs * 8 * sizeof(float));
-+ GlyphBox * currbox = boxes;
-+
-+ for (uint16 gid = 0; currbox && gid != _num_glyphs; ++gid)
-+ {
-+ _boxes[gid] = currbox;
-+ currbox = _glyph_loader->read_box(gid, currbox, *_glyphs[gid]);
-+ }
-+ if (!currbox)
-+ {
-+ free(boxes);
-+ _boxes[0] = 0;
-+ }
-+ }
- delete _glyph_loader;
- _glyph_loader = 0;
- }
-
- if (_glyphs && glyph(0) == 0)
- {
- free(_glyphs);
- _glyphs = 0;
-+ if (_boxes)
-+ {
-+ free(_boxes);
-+ _boxes = 0;
-+ }
- _num_glyphs = _num_attrs = _upem = 0;
- }
- }
-
-
- GlyphCache::~GlyphCache()
- {
- if (_glyphs)
-@@ -158,91 +187,130 @@ GlyphCache::~GlyphCache()
- const GlyphFace * * g = _glyphs;
- for(unsigned short n = _num_glyphs; n; --n, ++g)
- delete *g;
- }
- else
- delete [] _glyphs[0];
- free(_glyphs);
- }
-+ if (_boxes)
-+ {
-+ if (_glyph_loader)
-+ {
-+ GlyphBox * * g = _boxes;
-+ for (uint16 n = _num_glyphs; n; --n, ++g)
-+ free(*g);
-+ }
-+ else
-+ free(_boxes[0]);
-+ free(_boxes);
-+ }
- delete _glyph_loader;
- }
-
- const GlyphFace *GlyphCache::glyph(unsigned short glyphid) const //result may be changed by subsequent call with a different glyphid
- {
- const GlyphFace * & p = _glyphs[glyphid];
- if (p == 0 && _glyph_loader)
- {
-+ int numsubs = 0;
- GlyphFace * g = new GlyphFace();
-- if (g) p = _glyph_loader->read_glyph(glyphid, *g);
-+ if (g) p = _glyph_loader->read_glyph(glyphid, *g, &numsubs);
- if (!p)
- {
- delete g;
- return *_glyphs;
- }
-+ if (_boxes)
-+ {
-+ _boxes[glyphid] = (GlyphBox *)gralloc<char>(sizeof(GlyphBox) + 8 * numsubs * sizeof(float));
-+ if (!_glyph_loader->read_box(glyphid, _boxes[glyphid], *_glyphs[glyphid]))
-+ {
-+ free(_boxes[glyphid]);
-+ _boxes[glyphid] = 0;
-+ }
-+ }
- }
- return p;
- }
-
-
-
- GlyphCache::Loader::Loader(const Face & face, const bool dumb_font)
- : _head(face, Tag::head),
- _hhea(face, Tag::hhea),
- _hmtx(face, Tag::hmtx),
- _glyf(face, Tag::glyf),
- _loca(face, Tag::loca),
- _long_fmt(false),
-+ _has_boxes(false),
- _num_glyphs_graphics(0),
- _num_glyphs_attributes(0),
- _num_attrs(0)
- {
- if (!operator bool())
- return;
-
- const Face::Table maxp = Face::Table(face, Tag::maxp);
- if (!maxp) { _head = Face::Table(); return; }
-
- _num_glyphs_graphics = TtfUtil::GlyphCount(maxp);
- // This will fail if the number of glyphs is wildly out of range.
-- if (_glyf && TtfUtil::LocaLookup(_num_glyphs_graphics-1, _loca, _loca.size(), _head) == size_t(-1))
-+ if (_glyf && TtfUtil::LocaLookup(_num_glyphs_graphics-1, _loca, _loca.size(), _head) == size_t(-2))
- {
- _head = Face::Table();
- return;
- }
-
- if (!dumb_font)
- {
-- if ((m_pGlat = Face::Table(face, Tag::Glat)) == NULL
-+ if ((m_pGlat = Face::Table(face, Tag::Glat, 0x00030000)) == NULL
- || (m_pGloc = Face::Table(face, Tag::Gloc)) == NULL
- || m_pGloc.size() < 6)
- {
- _head = Face::Table();
- return;
- }
- const byte * p = m_pGloc;
-- const int version = be::read<uint32>(p);
-+ int version = be::read<uint32>(p);
- const uint16 flags = be::read<uint16>(p);
- _num_attrs = be::read<uint16>(p);
- // We can accurately calculate the number of attributed glyphs by
- // subtracting the length of the attribids array (numAttribs long if present)
- // and dividing by either 2 or 4 depending on shor or lonf format
- _long_fmt = flags & 1;
-- _num_glyphs_attributes = (m_pGloc.size()
-+ int tmpnumgattrs = (m_pGloc.size()
- - (p - m_pGloc)
- - sizeof(uint16)*(flags & 0x2 ? _num_attrs : 0))
- / (_long_fmt ? sizeof(uint32) : sizeof(uint16)) - 1;
-
-- if (version != 0x00010000
-+ if (version >= 0x00020000 || tmpnumgattrs < 0 || tmpnumgattrs > 65535
- || _num_attrs == 0 || _num_attrs > 0x3000 // is this hard limit appropriate?
-- || _num_glyphs_graphics > _num_glyphs_attributes)
-+ || _num_glyphs_graphics > tmpnumgattrs)
- {
- _head = Face::Table();
- return;
- }
-+
-+ _num_glyphs_attributes = static_cast<unsigned short>(tmpnumgattrs);
-+ p = m_pGlat;
-+ version = be::read<uint32>(p);
-+ if (version >= 0x00040000) // reject Glat tables that are too new
-+ {
-+ _head = Face::Table();
-+ return;
-+ }
-+ else if (version >= 0x00030000)
-+ {
-+ unsigned int glatflags = be::read<uint32>(p);
-+ _has_boxes = glatflags & 1;
-+ // delete this once the compiler is fixed
-+ _has_boxes = true;
-+ }
- }
- }
-
- inline
- GlyphCache::Loader::operator bool () const throw()
- {
- return _head && _hhea && _hmtx && !(bool(_glyf) != bool(_loca));
- }
-@@ -260,34 +328,44 @@ unsigned short int GlyphCache::Loader::n
- }
-
- inline
- unsigned short int GlyphCache::Loader::num_attrs() const throw()
- {
- return _num_attrs;
- }
-
--const GlyphFace * GlyphCache::Loader::read_glyph(unsigned short glyphid, GlyphFace & glyph) const throw()
-+inline
-+bool GlyphCache::Loader::has_boxes () const throw()
-+{
-+ return _has_boxes;
-+}
-+
-+const GlyphFace * GlyphCache::Loader::read_glyph(unsigned short glyphid, GlyphFace & glyph, int *numsubs) const throw()
- {
- Rect bbox;
- Position advance;
-
- if (glyphid < _num_glyphs_graphics)
- {
- int nLsb;
- unsigned int nAdvWid;
- if (_glyf)
- {
- int xMin, yMin, xMax, yMax;
- size_t locidx = TtfUtil::LocaLookup(glyphid, _loca, _loca.size(), _head);
- void *pGlyph = TtfUtil::GlyfLookup(_glyf, locidx, _glyf.size());
-
- if (pGlyph && TtfUtil::GlyfBox(pGlyph, xMin, yMin, xMax, yMax))
-+ {
-+ if ((xMin > xMax) || (yMin > yMax))
-+ return 0;
- bbox = Rect(Position(static_cast<float>(xMin), static_cast<float>(yMin)),
- Position(static_cast<float>(xMax), static_cast<float>(yMax)));
-+ }
- }
- if (TtfUtil::HorMetrics(glyphid, _hmtx, _hmtx.size(), _hhea, nLsb, nAdvWid))
- advance = Position(static_cast<float>(nAdvWid), 0);
- }
-
- if (glyphid < _num_glyphs_attributes)
- {
- const byte * gloc = m_pGloc;
-@@ -307,35 +385,95 @@ const GlyphFace * GlyphCache::Loader::re
- glocs = be::read<uint16>(gloc);
- gloce = be::peek<uint16>(gloc);
- }
-
- if (glocs >= m_pGlat.size() || gloce > m_pGlat.size())
- return 0;
-
- const uint32 glat_version = be::peek<uint32>(m_pGlat);
-+ if (glat_version == 0x00030000)
-+ {
-+ const byte * p = m_pGlat + glocs;
-+ uint16 bmap = be::read<uint16>(p);
-+ int num = bit_set_count((uint32)bmap);
-+ if (numsubs) *numsubs += num;
-+ glocs += 6 + 8 * num;
-+ if (glocs > gloce)
-+ return 0;
-+ }
- if (glat_version < 0x00020000)
- {
- if (gloce - glocs < 2*sizeof(byte)+sizeof(uint16)
- || gloce - glocs > _num_attrs*(2*sizeof(byte)+sizeof(uint16)))
-- {
-- return 0;
-- }
--
-+ return 0;
- new (&glyph) GlyphFace(bbox, advance, glat_iterator(m_pGlat + glocs), glat_iterator(m_pGlat + gloce));
- }
- else
- {
-- if (gloce - glocs < 3*sizeof(uint16)
-- || gloce - glocs > _num_attrs*3*sizeof(uint16))
-- {
-- return 0;
-- }
--
-+ if (gloce - glocs < 3*sizeof(uint16) // can a glyph have no attributes? why not?
-+ || gloce - glocs > _num_attrs*3*sizeof(uint16)
-+ || glocs > m_pGlat.size() - 2*sizeof(uint16))
-+ return 0;
- new (&glyph) GlyphFace(bbox, advance, glat2_iterator(m_pGlat + glocs), glat2_iterator(m_pGlat + gloce));
- }
--
- if (!glyph.attrs() || glyph.attrs().capacity() > _num_attrs)
- return 0;
- }
--
- return &glyph;
- }
-+
-+inline float scale_to(uint8 t, float zmin, float zmax)
-+{
-+ return (zmin + t * (zmax - zmin) / 255);
-+}
-+
-+Rect readbox(Rect &b, uint8 zxmin, uint8 zymin, uint8 zxmax, uint8 zymax)
-+{
-+ return Rect(Position(scale_to(zxmin, b.bl.x, b.tr.x), scale_to(zymin, b.bl.y, b.tr.y)),
-+ Position(scale_to(zxmax, b.bl.x, b.tr.x), scale_to(zymax, b.bl.y, b.tr.y)));
-+}
-+
-+GlyphBox * GlyphCache::Loader::read_box(uint16 gid, GlyphBox *curr, const GlyphFace & glyph) const throw()
-+{
-+ if (gid >= _num_glyphs_attributes) return 0;
-+
-+ const byte * gloc = m_pGloc;
-+ size_t glocs = 0, gloce = 0;
-+
-+ be::skip<uint32>(gloc);
-+ be::skip<uint16>(gloc,2);
-+ if (_long_fmt)
-+ {
-+ be::skip<uint32>(gloc, gid);
-+ glocs = be::read<uint32>(gloc);
-+ gloce = be::peek<uint32>(gloc);
-+ }
-+ else
-+ {
-+ be::skip<uint16>(gloc, gid);
-+ glocs = be::read<uint16>(gloc);
-+ gloce = be::peek<uint16>(gloc);
-+ }
-+
-+ if (glocs >= m_pGlat.size() || gloce > m_pGlat.size())
-+ return 0;
-+
-+ const byte * p = m_pGlat + glocs;
-+ uint16 bmap = be::read<uint16>(p);
-+ int num = bit_set_count((uint32)bmap);
-+
-+ Rect bbox = glyph.theBBox();
-+ Rect diamax(Position(bbox.bl.x + bbox.bl.y, bbox.bl.x - bbox.tr.y),
-+ Position(bbox.tr.x + bbox.tr.y, bbox.tr.x - bbox.bl.y));
-+ Rect diabound = readbox(diamax, p[0], p[2], p[1], p[3]);
-+ ::new (curr) GlyphBox(num, bmap, &diabound);
-+ be::skip<uint8>(p, 4);
-+
-+ for (int i = 0; i < num * 2; ++i)
-+ {
-+ Rect box = readbox((i & 1) ? diamax : bbox, p[0], p[2], p[1], p[3]);
-+ curr->addSubBox(i >> 1, i & 1, &box);
-+ be::skip<uint8>(p, 4);
-+ }
-+ return (GlyphBox *)((char *)(curr) + sizeof(GlyphBox) + 2 * num * sizeof(Rect));
-+}
-+
-diff --git a/gfx/graphite2/src/Intervals.cpp b/gfx/graphite2/src/Intervals.cpp
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/Intervals.cpp
-@@ -0,0 +1,294 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#include <algorithm>
-+#include <cmath>
-+#include <limits>
-+
-+#include "inc/Intervals.h"
-+#include "inc/Segment.h"
-+#include "inc/Slot.h"
-+#include "inc/debug.h"
-+#include "inc/bits.h"
-+
-+using namespace graphite2;
-+
-+#include <cmath>
-+
-+inline
-+Zones::Exclusion Zones::Exclusion::split_at(float p) {
-+ Exclusion r(*this);
-+ r.xm = x = p;
-+ return r;
-+}
-+
-+inline
-+void Zones::Exclusion::left_trim(float p) {
-+ x = p;
-+}
-+
-+inline
-+Zones::Exclusion & Zones::Exclusion::operator += (Exclusion const & rhs) {
-+ c += rhs.c; sm += rhs.sm; smx += rhs.smx; open = false;
-+ return *this;
-+}
-+
-+inline
-+uint8 Zones::Exclusion::outcode(float val) const {
-+ float p = val;
-+ return ((p >= xm) << 1) | (p < x);
-+}
-+
-+void Zones::exclude_with_margins(float xmin, float xmax, int axis) {
-+ remove(xmin, xmax);
-+ weightedAxis(axis, xmin-_margin_len, xmin, 0, 0, _margin_weight, xmin-_margin_len, 0, 0, false);
-+ weightedAxis(axis, xmax, xmax+_margin_len, 0, 0, _margin_weight, xmax+_margin_len, 0, 0, false);
-+}
-+
-+namespace
-+{
-+
-+inline
-+bool separated(float a, float b) {
-+ return a != b;
-+ //return std::fabs(a-b) > std::numeric_limits<float>::epsilon(); // std::epsilon may not work. but 0.5 fails exising 64 bit tests
-+ //return std::fabs(a-b) > 0.5f;
-+}
-+
-+}
-+
-+void Zones::insert(Exclusion e)
-+{
-+#if !defined GRAPHITE2_NTRACING
-+ addDebug(&e);
-+#endif
-+ e.x = max(e.x, _pos);
-+ e.xm = min(e.xm, _posm);
-+ if (e.x >= e.xm) return;
-+
-+ for (iterator i = _exclusions.begin(), ie = _exclusions.end(); i != ie && e.x < e.xm; ++i)
-+ {
-+ const uint8 oca = e.outcode(i->x),
-+ ocb = e.outcode(i->xm);
-+ if ((oca & ocb) != 0) continue;
-+
-+ switch (oca ^ ocb) // What kind of overlap?
-+ {
-+ case 0: // e completely covers i
-+ // split e at i.x into e1,e2
-+ // split e2 at i.mx into e2,e3
-+ // drop e1 ,i+e2, e=e3
-+ *i += e;
-+ e.left_trim(i->xm);
-+ break;
-+ case 1: // e overlaps on the rhs of i
-+ // split i at e->x into i1,i2
-+ // split e at i.mx into e1,e2
-+ // trim i1, insert i2+e1, e=e2
-+ if (!separated(i->xm, e.x)) break;
-+ if (separated(i->x,e.x)) { i = _exclusions.insert(i,i->split_at(e.x)); ++i; }
-+ *i += e;
-+ e.left_trim(i->xm);
-+ break;
-+ case 2: // e overlaps on the lhs of i
-+ // split e at i->x into e1,e2
-+ // split i at e.mx into i1,i2
-+ // drop e1, insert e2+i1, trim i2
-+ if (!separated(e.xm, i->x)) return;
-+ if (separated(e.xm, i->xm)) i = _exclusions.insert(i,i->split_at(e.xm));
-+ *i += e;
-+ return;
-+ case 3: // i completely covers e
-+ // split i at e.x into i1,i2
-+ // split i2 at e.mx into i2,i3
-+ // insert i1, insert e+i2
-+ if (separated(e.xm, i->xm)) i = _exclusions.insert(i,i->split_at(e.xm));
-+ i = _exclusions.insert(i, i->split_at(e.x));
-+ *++i += e;
-+ return;
-+ }
-+
-+ ie = _exclusions.end();
-+ }
-+}
-+
-+
-+void Zones::remove(float x, float xm)
-+{
-+#if !defined GRAPHITE2_NTRACING
-+ removeDebug(x, xm);
-+#endif
-+ x = max(x, _pos);
-+ xm = min(xm, _posm);
-+ if (x >= xm) return;
-+
-+ for (iterator i = _exclusions.begin(), ie = _exclusions.end(); i != ie; ++i)
-+ {
-+ const uint8 oca = i->outcode(x),
-+ ocb = i->outcode(xm);
-+ if ((oca & ocb) != 0) continue;
-+
-+ switch (oca ^ ocb) // What kind of overlap?
-+ {
-+ case 0: // i completely covers e
-+ if (separated(i->x, x)) { i = _exclusions.insert(i,i->split_at(x)); ++i; }
-+ GR_FALLTHROUGH;
-+ // no break
-+ case 1: // i overlaps on the rhs of e
-+ i->left_trim(xm);
-+ return;
-+ case 2: // i overlaps on the lhs of e
-+ i->xm = x;
-+ if (separated(i->x, i->xm)) break;
-+ GR_FALLTHROUGH;
-+ // no break
-+ case 3: // e completely covers i
-+ i = _exclusions.erase(i);
-+ --i;
-+ break;
-+ }
-+
-+ ie = _exclusions.end();
-+ }
-+}
-+
-+
-+Zones::const_iterator Zones::find_exclusion_under(float x) const
-+{
-+ int l = 0, h = _exclusions.size();
-+
-+ while (l < h)
-+ {
-+ int const p = (l+h) >> 1;
-+ switch (_exclusions[p].outcode(x))
-+ {
-+ case 0 : return _exclusions.begin()+p;
-+ case 1 : h = p; break;
-+ case 2 :
-+ case 3 : l = p+1; break;
-+ }
-+ }
-+
-+ return _exclusions.begin()+l;
-+}
-+
-+
-+float Zones::closest(float origin, float & cost) const
-+{
-+ float best_c = std::numeric_limits<float>::max(),
-+ best_x = 0;
-+
-+ const const_iterator start = find_exclusion_under(origin);
-+
-+ // Forward scan looking for lowest cost
-+ for (const_iterator i = start, ie = _exclusions.end(); i != ie; ++i)
-+ if (i->track_cost(best_c, best_x, origin)) break;
-+
-+ // Backward scan looking for lowest cost
-+ // We start from the exclusion to the immediate left of start since we've
-+ // already tested start with the right most scan above.
-+ for (const_iterator i = start-1, ie = _exclusions.begin()-1; i != ie; --i)
-+ if (i->track_cost(best_c, best_x, origin)) break;
-+
-+ cost = (best_c == std::numeric_limits<float>::max() ? -1 : best_c);
-+ return best_x;
-+}
-+
-+
-+// Cost and test position functions
-+
-+bool Zones::Exclusion::track_cost(float & best_cost, float & best_pos, float origin) const {
-+ const float p = test_position(origin),
-+ localc = cost(p - origin);
-+ if (open && localc > best_cost) return true;
-+
-+ if (localc < best_cost)
-+ {
-+ best_cost = localc;
-+ best_pos = p;
-+ }
-+ return false;
-+}
-+
-+inline
-+float Zones::Exclusion::cost(float p) const {
-+ return (sm * p - 2 * smx) * p + c;
-+}
-+
-+
-+float Zones::Exclusion::test_position(float origin) const {
-+ if (sm < 0)
-+ {
-+ // sigh, test both ends and perhaps the middle too!
-+ float res = x;
-+ float cl = cost(x);
-+ if (x < origin && xm > origin)
-+ {
-+ float co = cost(origin);
-+ if (co < cl)
-+ {
-+ cl = co;
-+ res = origin;
-+ }
-+ }
-+ float cr = cost(xm);
-+ return cl > cr ? xm : res;
-+ }
-+ else
-+ {
-+ float zerox = smx / sm + origin;
-+ if (zerox < x) return x;
-+ else if (zerox > xm) return xm;
-+ else return zerox;
-+ }
-+}
-+
-+
-+#if !defined GRAPHITE2_NTRACING
-+
-+void Zones::jsonDbgOut(Segment *seg) const {
-+
-+ if (_dbg)
-+ {
-+ for (Zones::idebugs s = dbgs_begin(), e = dbgs_end(); s != e; ++s)
-+ {
-+ *_dbg << json::flat << json::array
-+ << objectid(dslot(seg, (Slot *)(s->_env[0])))
-+ << reinterpret_cast<ptrdiff_t>(s->_env[1]);
-+ if (s->_isdel)
-+ *_dbg << "remove" << Position(s->_excl.x, s->_excl.xm);
-+ else
-+ *_dbg << "exclude" << json::flat << json::array
-+ << s->_excl.x << s->_excl.xm
-+ << s->_excl.sm << s->_excl.smx << s->_excl.c
-+ << json::close;
-+ *_dbg << json::close;
-+ }
-+ }
-+}
-+
-+#endif
-+
-diff --git a/gfx/graphite2/src/Justifier.cpp b/gfx/graphite2/src/Justifier.cpp
---- a/gfx/graphite2/src/Justifier.cpp
-+++ b/gfx/graphite2/src/Justifier.cpp
-@@ -26,17 +26,17 @@ of the License or (at your option) any l
- */
-
- #include "inc/Segment.h"
- #include "graphite2/Font.h"
- #include "inc/debug.h"
- #include "inc/CharInfo.h"
- #include "inc/Slot.h"
- #include "inc/Main.h"
--#include <math.h>
-+#include <cmath>
-
- using namespace graphite2;
-
- class JustifyTotal {
- public:
- JustifyTotal() : m_numGlyphs(0), m_tStretch(0), m_tShrink(0), m_tStep(0), m_tWeight(0) {}
- void accumulate(Slot *s, Segment *seg, int level);
- int weight() const { return m_tWeight; }
-@@ -55,37 +55,44 @@ void JustifyTotal::accumulate(Slot *s, S
- {
- ++m_numGlyphs;
- m_tStretch += s->getJustify(seg, level, 0);
- m_tShrink += s->getJustify(seg, level, 1);
- m_tStep += s->getJustify(seg, level, 2);
- m_tWeight += s->getJustify(seg, level, 3);
- }
-
--float Segment::justify(Slot *pSlot, const Font *font, float width, GR_MAYBE_UNUSED justFlags flags, Slot *pFirst, Slot *pLast)
-+float Segment::justify(Slot *pSlot, const Font *font, float width, GR_MAYBE_UNUSED justFlags jflags, Slot *pFirst, Slot *pLast)
- {
- Slot *s, *end;
- float currWidth = 0.0;
- const float scale = font ? font->scale() : 1.0f;
- Position res;
-
- if (width < 0 && !(silf()->flags()))
- return width;
-
-+ if ((m_dir & 1) != m_silf->dir() && m_silf->bidiPass() != m_silf->numPasses())
-+ {
-+ reverseSlots();
-+ s = pFirst;
-+ pFirst = pLast;
-+ pLast = s;
-+ }
- if (!pFirst) pFirst = pSlot;
- while (!pFirst->isBase()) pFirst = pFirst->attachedTo();
- if (!pLast) pLast = last();
- while (!pLast->isBase()) pLast = pLast->attachedTo();
- const float base = pFirst->origin().x / scale;
- width = width / scale;
-- if ((flags & gr_justEndInline) == 0)
-+ if ((jflags & gr_justEndInline) == 0)
- {
- do {
- Rect bbox = theGlyphBBoxTemporary(pLast->glyph());
-- if (bbox.bl.x != 0. || bbox.bl.y != 0. || bbox.tr.x != 0. || bbox.tr.y == 0.)
-+ if (bbox.bl.x != 0.f || bbox.bl.y != 0.f || bbox.tr.x != 0.f || bbox.tr.y == 0.f)
- break;
- pLast = pLast->prev();
- } while (pLast != pFirst);
- }
-
- end = pLast->nextSibling();
- pFirst = pFirst->nextSibling();
-
-@@ -111,28 +118,27 @@ float Segment::justify(Slot *pSlot, cons
- s->setJustify(this, 0, 3, 1);
- s->setJustify(this, 0, 2, 1);
- s->setJustify(this, 0, 0, -1);
- }
- }
- ++numLevels;
- }
-
-- JustifyTotal *stats = new JustifyTotal[numLevels];
-- if (!stats) return -1.0;
-+ Vector<JustifyTotal> stats(numLevels);
- for (s = pFirst; s != end; s = s->nextSibling())
- {
- float w = s->origin().x / scale + s->advance() - base;
- if (w > currWidth) currWidth = w;
- for (int j = 0; j < numLevels; ++j)
- stats[j].accumulate(s, this, j);
- s->just(0);
- }
-
-- for (int i = (width < 0.0) ? -1 : numLevels - 1; i >= 0; --i)
-+ for (int i = (width < 0.0f) ? -1 : numLevels - 1; i >= 0; --i)
- {
- float diff;
- float error = 0.;
- float diffpw;
- int tWeight = stats[i].weight();
-
- do {
- error = 0.;
-@@ -154,29 +160,29 @@ float Segment::justify(Slot *pSlot, cons
- }
- else
- {
- float max = uint16(s->getJustify(this, i, 1));
- if (i == 0) max += s->just();
- if (-pref > max) pref = -max;
- else tWeight += w;
- }
-- int actual = step ? int(pref / step) * step : int(pref);
-+ int actual = int(pref / step) * step;
-
- if (actual)
- {
- error += diffpw * w - actual;
- if (i == 0)
- s->just(s->just() + actual);
- else
- s->setJustify(this, i, 4, actual);
- }
- }
- currWidth += diff - error;
-- } while (i == 0 && int(abs(error)) > 0 && tWeight);
-+ } while (i == 0 && int(std::abs(error)) > 0 && tWeight);
- }
-
- Slot *oldFirst = m_first;
- Slot *oldLast = m_last;
- if (silf()->flags() & 1)
- {
- m_first = pSlot = addLineEnd(pSlot);
- m_last = pLast = addLineEnd(end);
-@@ -192,41 +198,44 @@ float Segment::justify(Slot *pSlot, cons
- #if !defined GRAPHITE2_NTRACING
- json * const dbgout = m_face->logger();
- if (dbgout)
- *dbgout << json::object
- << "justifies" << objectid(this)
- << "passes" << json::array;
- #endif
-
-- if (m_silf->justificationPass() != m_silf->positionPass() && (width >= 0. || (silf()->flags() & 1)))
-+ if (m_silf->justificationPass() != m_silf->positionPass() && (width >= 0.f || (silf()->flags() & 1)))
- m_silf->runGraphite(this, m_silf->justificationPass(), m_silf->positionPass());
-
- #if !defined GRAPHITE2_NTRACING
- if (dbgout)
- {
- *dbgout << json::item << json::close; // Close up the passes array
-- positionSlots(NULL, pSlot, pLast);
-+ positionSlots(NULL, pSlot, pLast, m_dir);
- Slot *lEnd = pLast->nextSibling();
- *dbgout << "output" << json::array;
- for(Slot * t = pSlot; t != lEnd; t = t->next())
- *dbgout << dslot(this, t);
- *dbgout << json::close << json::close;
- }
- #endif
-
-- res = positionSlots(font, pSlot, pLast);
-+ res = positionSlots(font, pSlot, pLast, m_dir);
-
- if (silf()->flags() & 1)
- {
- delLineEnd(m_first);
- delLineEnd(m_last);
- }
- m_first = oldFirst;
- m_last = oldLast;
-+
-+ if ((m_dir & 1) != m_silf->dir() && m_silf->bidiPass() != m_silf->numPasses())
-+ reverseSlots();
- return res.x;
- }
-
- Slot *Segment::addLineEnd(Slot *nSlot)
- {
- Slot *eSlot = newSlot();
- if (!eSlot) return NULL;
- const uint16 gid = silf()->endLineGlyphid();
-diff --git a/gfx/graphite2/src/Pass.cpp b/gfx/graphite2/src/Pass.cpp
---- a/gfx/graphite2/src/Pass.cpp
-+++ b/gfx/graphite2/src/Pass.cpp
-@@ -26,91 +26,120 @@ of the License or (at your option) any l
- */
- #include "inc/Main.h"
- #include "inc/debug.h"
- #include "inc/Endian.h"
- #include "inc/Pass.h"
- #include <cstring>
- #include <cstdlib>
- #include <cassert>
-+#include <cmath>
- #include "inc/Segment.h"
- #include "inc/Code.h"
- #include "inc/Rule.h"
- #include "inc/Error.h"
-+#include "inc/Collider.h"
-
- using namespace graphite2;
- using vm::Machine;
- typedef Machine::Code Code;
-
-+enum KernCollison
-+{
-+ None = 0,
-+ CrossSpace = 1,
-+ InWord = 2,
-+ reserved = 3
-+};
-
- Pass::Pass()
- : m_silf(0),
- m_cols(0),
- m_rules(0),
- m_ruleMap(0),
- m_startStates(0),
- m_transitions(0),
- m_states(0),
-- m_flags(0),
-+ m_codes(0),
-+ m_progs(0),
-+ m_numCollRuns(0),
-+ m_kernColls(0),
- m_iMaxLoop(0),
- m_numGlyphs(0),
- m_numRules(0),
- m_numStates(0),
- m_numTransition(0),
- m_numSuccess(0),
-+ m_successStart(0),
- m_numColumns(0),
- m_minPreCtxt(0),
-- m_maxPreCtxt(0)
-+ m_maxPreCtxt(0),
-+ m_colThreshold(0),
-+ m_isReverseDir(false)
- {
- }
-
- Pass::~Pass()
- {
- free(m_cols);
- free(m_startStates);
- free(m_transitions);
- free(m_states);
- free(m_ruleMap);
-
-- delete [] m_rules;
-+ if (m_rules) delete [] m_rules;
-+ if (m_codes) delete [] m_codes;
-+ free(m_progs);
- }
-
--bool Pass::readPass(const byte * const pass_start, size_t pass_length, size_t subtable_base, GR_MAYBE_UNUSED Face & face, Error &e)
-+bool Pass::readPass(const byte * const pass_start, size_t pass_length, size_t subtable_base,
-+ GR_MAYBE_UNUSED Face & face, passtype pt, GR_MAYBE_UNUSED uint32 version, Error &e)
- {
-- const byte * p = pass_start,
-- * const pass_end = p + pass_length;
-+ const byte * p = pass_start,
-+ * const pass_end = p + pass_length;
- size_t numRanges;
-
- if (e.test(pass_length < 40, E_BADPASSLENGTH)) return face.error(e);
- // Read in basic values
-- m_flags = be::read<byte>(p);
-+ const byte flags = be::read<byte>(p);
-+ if (e.test((flags & 0x1f) &&
-+ (pt < PASS_TYPE_POSITIONING || !m_silf->aCollision() || !face.glyphs().hasBoxes()),
-+ E_BADCOLLISIONPASS))
-+ return face.error(e);
-+ m_numCollRuns = flags & 0x7;
-+ m_kernColls = (flags >> 3) & 0x3;
-+ m_isReverseDir = (flags >> 5) & 0x1;
- m_iMaxLoop = be::read<byte>(p);
-+ if (m_iMaxLoop < 1) m_iMaxLoop = 1;
- be::skip<byte>(p,2); // skip maxContext & maxBackup
- m_numRules = be::read<uint16>(p);
-+ if (e.test(!m_numRules && m_numCollRuns == 0, E_BADEMPTYPASS)) return face.error(e);
- be::skip<uint16>(p); // fsmOffset - not sure why we would want this
- const byte * const pcCode = pass_start + be::read<uint32>(p) - subtable_base,
- * const rcCode = pass_start + be::read<uint32>(p) - subtable_base,
- * const aCode = pass_start + be::read<uint32>(p) - subtable_base;
- be::skip<uint32>(p);
- m_numStates = be::read<uint16>(p);
- m_numTransition = be::read<uint16>(p);
- m_numSuccess = be::read<uint16>(p);
- m_numColumns = be::read<uint16>(p);
- numRanges = be::read<uint16>(p);
- be::skip<uint16>(p, 3); // skip searchRange, entrySelector & rangeShift.
- assert(p - pass_start == 40);
- // Perform some sanity checks.
- if ( e.test(m_numTransition > m_numStates, E_BADNUMTRANS)
- || e.test(m_numSuccess > m_numStates, E_BADNUMSUCCESS)
- || e.test(m_numSuccess + m_numTransition < m_numStates, E_BADNUMSTATES)
-- || e.test(numRanges == 0, E_NORANGES))
-+ || e.test(m_numRules && numRanges == 0, E_NORANGES)
-+ || e.test(m_numColumns > 0x7FFF, E_BADNUMCOLUMNS))
- return face.error(e);
-
- m_successStart = m_numStates - m_numSuccess;
-- if (e.test(p + numRanges * 6 - 4 > pass_end, E_BADPASSLENGTH)) return face.error(e);
-+ // test for beyond end - 1 to account for reading uint16
-+ if (e.test(p + numRanges * 6 - 2 > pass_end, E_BADPASSLENGTH)) return face.error(e);
- m_numGlyphs = be::peek<uint16>(p + numRanges * 6 - 4) + 1;
- // Calculate the start of various arrays.
- const byte * const ranges = p;
- be::skip<uint16>(p, numRanges*3);
- const byte * const o_rule_map = p;
- be::skip<uint16>(p, m_numSuccess + 1);
-
- // More sanity checks
-@@ -126,108 +155,141 @@ bool Pass::readPass(const byte * const p
- m_maxPreCtxt = be::read<uint8>(p);
- if (e.test(m_minPreCtxt > m_maxPreCtxt, E_BADCTXTLENBOUNDS)) return face.error(e);
- const byte * const start_states = p;
- be::skip<int16>(p, m_maxPreCtxt - m_minPreCtxt + 1);
- const uint16 * const sort_keys = reinterpret_cast<const uint16 *>(p);
- be::skip<uint16>(p, m_numRules);
- const byte * const precontext = p;
- be::skip<byte>(p, m_numRules);
-- be::skip<byte>(p); // skip reserved byte
-
-- if (e.test(p + sizeof(uint16) > pass_end, E_BADCTXTLENS)) return face.error(e);
-+ if (e.test(p + sizeof(uint16) + sizeof(uint8) > pass_end, E_BADCTXTLENS)) return face.error(e);
-+ m_colThreshold = be::read<uint8>(p);
-+ if (m_colThreshold == 0) m_colThreshold = 10; // A default
- const size_t pass_constraint_len = be::read<uint16>(p);
-+
- const uint16 * const o_constraint = reinterpret_cast<const uint16 *>(p);
- be::skip<uint16>(p, m_numRules + 1);
- const uint16 * const o_actions = reinterpret_cast<const uint16 *>(p);
- be::skip<uint16>(p, m_numRules + 1);
- const byte * const states = p;
-+ if (e.test(p + 2u*m_numTransition*m_numColumns >= pass_end, E_BADPASSLENGTH)) return face.error(e);
- be::skip<int16>(p, m_numTransition*m_numColumns);
-- be::skip<byte>(p); // skip reserved byte
-- if (e.test(p != pcCode, E_BADPASSCCODEPTR) || e.test(p >= pass_end, E_BADPASSLENGTH)) return face.error(e);
-+ be::skip<uint8>(p);
-+ if (e.test(p != pcCode, E_BADPASSCCODEPTR)) return face.error(e);
- be::skip<byte>(p, pass_constraint_len);
-- if (e.test(p != rcCode, E_BADRULECCODEPTR) || e.test(p >= pass_end, E_BADPASSLENGTH)
-+ if (e.test(p != rcCode, E_BADRULECCODEPTR)
- || e.test(size_t(rcCode - pcCode) != pass_constraint_len, E_BADCCODELEN)) return face.error(e);
- be::skip<byte>(p, be::peek<uint16>(o_constraint + m_numRules));
-- if (e.test(p != aCode, E_BADACTIONCODEPTR) || e.test(p >= pass_end, E_BADPASSLENGTH)) return face.error(e);
-+ if (e.test(p != aCode, E_BADACTIONCODEPTR)) return face.error(e);
- be::skip<byte>(p, be::peek<uint16>(o_actions + m_numRules));
-
- // We should be at the end or within the pass
- if (e.test(p > pass_end, E_BADPASSLENGTH)) return face.error(e);
-
- // Load the pass constraint if there is one.
- if (pass_constraint_len)
- {
- face.error_context(face.error_context() + 1);
- m_cPConstraint = vm::Machine::Code(true, pcCode, pcCode + pass_constraint_len,
-- precontext[0], be::peek<uint16>(sort_keys), *m_silf, face);
-+ precontext[0], be::peek<uint16>(sort_keys), *m_silf, face, PASS_TYPE_UNKNOWN);
- if (e.test(!m_cPConstraint, E_OUTOFMEM)
-- || e.test(m_cPConstraint.status(), m_cPConstraint.status() + E_CODEFAILURE))
-+ || e.test(!m_cPConstraint, m_cPConstraint.status() + E_CODEFAILURE))
- return face.error(e);
- face.error_context(face.error_context() - 1);
- }
-- if (!readRanges(ranges, numRanges, e)) return face.error(e);
-- if (!readRules(rule_map, numEntries, precontext, sort_keys,
-- o_constraint, rcCode, o_actions, aCode, face, e)) return false;
-+ if (m_numRules)
-+ {
-+ if (!readRanges(ranges, numRanges, e)) return face.error(e);
-+ if (!readRules(rule_map, numEntries, precontext, sort_keys,
-+ o_constraint, rcCode, o_actions, aCode, face, pt, e)) return false;
-+ }
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::category _states_cat(face.tele.states);
- #endif
-- return readStates(start_states, states, o_rule_map, face, e);
-+ return m_numRules ? readStates(start_states, states, o_rule_map, face, e) : true;
- }
-
-
- bool Pass::readRules(const byte * rule_map, const size_t num_entries,
- const byte *precontext, const uint16 * sort_key,
- const uint16 * o_constraint, const byte *rc_data,
- const uint16 * o_action, const byte * ac_data,
-- Face & face, Error &e)
-+ Face & face, passtype pt, Error &e)
- {
- const byte * const ac_data_end = ac_data + be::peek<uint16>(o_action + m_numRules);
- const byte * const rc_data_end = rc_data + be::peek<uint16>(o_constraint + m_numRules);
-
-- if (e.test(!(m_rules = new Rule [m_numRules]), E_OUTOFMEM)) return face.error(e);
- precontext += m_numRules;
- sort_key += m_numRules;
- o_constraint += m_numRules;
- o_action += m_numRules;
-
- // Load rules.
- const byte * ac_begin = 0, * rc_begin = 0,
- * ac_end = ac_data + be::peek<uint16>(o_action),
- * rc_end = rc_data + be::peek<uint16>(o_constraint);
-+
-+ // Allocate pools
-+ m_rules = new Rule [m_numRules];
-+ m_codes = new Code [m_numRules*2];
-+ const size_t prog_pool_sz = vm::Machine::Code::estimateCodeDataOut(ac_end - ac_data + rc_end - rc_data);
-+ m_progs = gralloc<byte>(prog_pool_sz);
-+ byte * prog_pool_free = m_progs,
-+ * prog_pool_end = m_progs + prog_pool_sz;
-+ if (e.test(!(m_rules && m_codes && m_progs), E_OUTOFMEM)) return face.error(e);
-+
- Rule * r = m_rules + m_numRules - 1;
- for (size_t n = m_numRules; n; --n, --r, ac_end = ac_begin, rc_end = rc_begin)
- {
- face.error_context((face.error_context() & 0xFFFF00) + EC_ARULE + ((n - 1) << 24));
- r->preContext = *--precontext;
- r->sort = be::peek<uint16>(--sort_key);
- #ifndef NDEBUG
- r->rule_idx = n - 1;
- #endif
- if (r->sort > 63 || r->preContext >= r->sort || r->preContext > m_maxPreCtxt || r->preContext < m_minPreCtxt)
- return false;
- ac_begin = ac_data + be::peek<uint16>(--o_action);
- --o_constraint;
- rc_begin = be::peek<uint16>(o_constraint) ? rc_data + be::peek<uint16>(o_constraint) : rc_end;
-
- if (ac_begin > ac_end || ac_begin > ac_data_end || ac_end > ac_data_end
-- || rc_begin > rc_end || rc_begin > rc_data_end || rc_end > rc_data_end)
-+ || rc_begin > rc_end || rc_begin > rc_data_end || rc_end > rc_data_end
-+ || vm::Machine::Code::estimateCodeDataOut(ac_end - ac_begin + rc_end - rc_begin) > size_t(prog_pool_end - prog_pool_free))
- return false;
-- r->action = new vm::Machine::Code(false, ac_begin, ac_end, r->preContext, r->sort, *m_silf, face);
-- r->constraint = new vm::Machine::Code(true, rc_begin, rc_end, r->preContext, r->sort, *m_silf, face);
-+ r->action = new (m_codes+n*2-2) vm::Machine::Code(false, ac_begin, ac_end, r->preContext, r->sort, *m_silf, face, pt, &prog_pool_free);
-+ r->constraint = new (m_codes+n*2-1) vm::Machine::Code(true, rc_begin, rc_end, r->preContext, r->sort, *m_silf, face, pt, &prog_pool_free);
-
- if (e.test(!r->action || !r->constraint, E_OUTOFMEM)
- || e.test(r->action->status() != Code::loaded, r->action->status() + E_CODEFAILURE)
- || e.test(r->constraint->status() != Code::loaded, r->constraint->status() + E_CODEFAILURE)
- || e.test(!r->constraint->immutable(), E_MUTABLECCODE))
- return face.error(e);
- }
-
-+ byte * moved_progs = static_cast<byte *>(realloc(m_progs, prog_pool_free - m_progs));
-+ if (e.test(!moved_progs, E_OUTOFMEM))
-+ {
-+ if (prog_pool_free - m_progs == 0) m_progs = 0;
-+ return face.error(e);
-+ }
-+
-+ if (moved_progs != m_progs)
-+ {
-+ for (Code * c = m_codes, * const ce = c + m_numRules*2; c != ce; ++c)
-+ {
-+ c->externalProgramMoved(moved_progs - m_progs);
-+ }
-+ m_progs = moved_progs;
-+ }
-+
- // Load the rule entries map
- face.error_context((face.error_context() & 0xFFFF00) + EC_APASS);
-+ //TODO: Coverty: 1315804: FORWARD_NULL
- RuleEntry * re = m_ruleMap = gralloc<RuleEntry>(num_entries);
- if (e.test(!re, E_OUTOFMEM)) return face.error(e);
- for (size_t n = num_entries; n; --n, ++re)
- {
- const ptrdiff_t rn = be::read<uint16>(rule_map);
- if (e.test(rn >= m_numRules, E_BADRULENUM)) return face.error(e);
- re->rule = m_rules + rn;
- }
-@@ -320,43 +382,69 @@ bool Pass::readRanges(const byte * range
-
- if (e.test(ci != ci_end, E_BADRANGE))
- return false;
- }
- return true;
- }
-
-
--void Pass::runGraphite(Machine & m, FiniteStateMachine & fsm) const
-+bool Pass::runGraphite(vm::Machine & m, FiniteStateMachine & fsm, bool reverse) const
- {
- Slot *s = m.slotMap().segment.first();
-- if (!s || !testPassConstraint(m)) return;
-- Slot *currHigh = s->next();
-+ if (!s || !testPassConstraint(m)) return true;
-+ if (reverse)
-+ {
-+ m.slotMap().segment.reverseSlots();
-+ s = m.slotMap().segment.first();
-+ }
-+ if (m_numRules)
-+ {
-+ Slot *currHigh = s->next();
-
- #if !defined GRAPHITE2_NTRACING
-- if (fsm.dbgout) *fsm.dbgout << "rules" << json::array;
-- json::closer rules_array_closer(fsm.dbgout);
-+ if (fsm.dbgout) *fsm.dbgout << "rules" << json::array;
-+ json::closer rules_array_closer(fsm.dbgout);
- #endif
-
-- m.slotMap().highwater(currHigh);
-- int lc = m_iMaxLoop;
-- do
-+ m.slotMap().highwater(currHigh);
-+ int lc = m_iMaxLoop;
-+ do
-+ {
-+ findNDoRule(s, m, fsm);
-+ if (s && (s == m.slotMap().highwater() || m.slotMap().highpassed() || --lc == 0)) {
-+ if (!lc)
-+ s = m.slotMap().highwater();
-+ lc = m_iMaxLoop;
-+ if (s)
-+ m.slotMap().highwater(s->next());
-+ }
-+ } while (s);
-+ }
-+ //TODO: Use enums for flags
-+ const bool collisions = m_numCollRuns || m_kernColls;
-+
-+ if (!collisions || !m.slotMap().segment.hasCollisionInfo())
-+ return true;
-+
-+ if (m_numCollRuns)
- {
-- findNDoRule(s, m, fsm);
-- if (s && (m.slotMap().highpassed() || s == m.slotMap().highwater() || --lc == 0)) {
-- if (!lc)
-- {
--// if (dbgout) *dbgout << json::item << json::flat << rule_event(-1, s, 1);
-- s = m.slotMap().highwater();
-- }
-- lc = m_iMaxLoop;
-- if (s)
-- m.slotMap().highwater(s->next());
-+ if (!(m.slotMap().segment.flags() & Segment::SEG_INITCOLLISIONS))
-+ {
-+ m.slotMap().segment.positionSlots(0, 0, 0, m.slotMap().dir(), true);
-+// m.slotMap().segment.flags(m.slotMap().segment.flags() | Segment::SEG_INITCOLLISIONS);
- }
-- } while (s);
-+ if (!collisionShift(&m.slotMap().segment, m.slotMap().dir(), fsm.dbgout))
-+ return false;
-+ }
-+ if ((m_kernColls) && !collisionKern(&m.slotMap().segment, m.slotMap().dir(), fsm.dbgout))
-+ return false;
-+ if (collisions && !collisionFinish(&m.slotMap().segment, fsm.dbgout))
-+ return false;
-+ return true;
- }
-
- bool Pass::runFSM(FiniteStateMachine& fsm, Slot * slot) const
- {
- fsm.reset(slot, m_maxPreCtxt);
- if (fsm.slots.context() < m_minPreCtxt)
- return false;
-
-@@ -419,18 +507,18 @@ void Pass::findNDoRule(Slot * & slot, Ma
- {
- if (fsm.rules.size() != 0)
- {
- *fsm.dbgout << json::item << json::object;
- dumpRuleEventConsidered(fsm, *r);
- if (r != re)
- {
- const int adv = doAction(r->rule->action, slot, m);
-- dumpRuleEventOutput(fsm, *r->rule, slot);
-- if (r->rule->action->deletes()) fsm.slots.collectGarbage();
-+ dumpRuleEventOutput(fsm, m, *r->rule, slot);
-+ if (r->rule->action->deletes()) fsm.slots.collectGarbage(slot);
- adjustSlot(adv, slot, fsm.slots);
- *fsm.dbgout << "cursor" << objectid(dslot(&fsm.slots.segment, slot))
- << json::close; // Close RuelEvent object
-
- return;
- }
- else
- {
-@@ -442,47 +530,49 @@ void Pass::findNDoRule(Slot * & slot, Ma
- }
- }
- else
- #endif
- {
- if (r != re)
- {
- const int adv = doAction(r->rule->action, slot, m);
-- if (r->rule->action->deletes()) fsm.slots.collectGarbage();
-+ if (r->rule->action->deletes()) fsm.slots.collectGarbage(slot);
- adjustSlot(adv, slot, fsm.slots);
- return;
- }
- }
- }
-
- slot = slot->next();
-+ return;
- }
-
- #if !defined GRAPHITE2_NTRACING
-
- void Pass::dumpRuleEventConsidered(const FiniteStateMachine & fsm, const RuleEntry & re) const
- {
- *fsm.dbgout << "considered" << json::array;
- for (const RuleEntry *r = fsm.rules.begin(); r != &re; ++r)
- {
-- if (r->rule->preContext > fsm.slots.context()) continue;
-- *fsm.dbgout << json::flat << json::object
-- << "id" << r->rule - m_rules
-+ if (r->rule->preContext > fsm.slots.context())
-+ continue;
-+ *fsm.dbgout << json::flat << json::object
-+ << "id" << r->rule - m_rules
- << "failed" << true
- << "input" << json::flat << json::object
- << "start" << objectid(dslot(&fsm.slots.segment, input_slot(fsm.slots, -r->rule->preContext)))
- << "length" << r->rule->sort
- << json::close // close "input"
- << json::close; // close Rule object
- }
- }
-
-
--void Pass::dumpRuleEventOutput(const FiniteStateMachine & fsm, const Rule & r, Slot * const last_slot) const
-+void Pass::dumpRuleEventOutput(const FiniteStateMachine & fsm, Machine & m, const Rule & r, Slot * const last_slot) const
- {
- *fsm.dbgout << json::item << json::flat << json::object
- << "id" << &r - m_rules
- << "failed" << false
- << "input" << json::flat << json::object
- << "start" << objectid(dslot(&fsm.slots.segment, input_slot(fsm.slots, 0)))
- << "length" << r.sort - r.preContext
- << json::close // close "input"
-@@ -490,17 +580,17 @@ void Pass::dumpRuleEventOutput(const Fin
- << json::close // close considered array
- << "output" << json::object
- << "range" << json::flat << json::object
- << "start" << objectid(dslot(&fsm.slots.segment, input_slot(fsm.slots, 0)))
- << "end" << objectid(dslot(&fsm.slots.segment, last_slot))
- << json::close // close "input"
- << "slots" << json::array;
- const Position rsb_prepos = last_slot ? last_slot->origin() : fsm.slots.segment.advance();
-- fsm.slots.segment.positionSlots(0);
-+ fsm.slots.segment.positionSlots(0, 0, 0, m.slotMap().dir());
-
- for(Slot * slot = output_slot(fsm.slots, 0); slot != last_slot; slot = slot->next())
- *fsm.dbgout << dslot(&fsm.slots.segment, slot);
- *fsm.dbgout << json::close // close "slots"
- << "postshift" << (last_slot ? last_slot->origin() : fsm.slots.segment.advance()) - rsb_prepos
- << json::close; // close "output" object
-
- }
-@@ -546,22 +636,26 @@ bool Pass::testConstraint(const Rule & r
- if (!ret || m.status() != Machine::finished)
- return false;
- }
-
- return true;
- }
-
-
--void SlotMap::collectGarbage()
-+void SlotMap::collectGarbage(Slot * &aSlot)
- {
- for(Slot **s = begin(), *const *const se = end() - 1; s != se; ++s) {
- Slot *& slot = *s;
- if(slot->isDeleted() || slot->isCopied())
-+ {
-+ if (slot == aSlot)
-+ aSlot = slot->prev() ? slot->prev() : slot->next();
- segment.freeSlot(slot);
-+ }
- }
- }
-
-
-
- int Pass::doAction(const Code *codeptr, Slot * & slot_out, vm::Machine & m) const
- {
- assert(codeptr);
-@@ -581,40 +675,412 @@ int Pass::doAction(const Code *codeptr,
-
- slot_out = *map;
- return ret;
- }
-
-
- void Pass::adjustSlot(int delta, Slot * & slot_out, SlotMap & smap) const
- {
-- if (delta < 0)
-+ if (!slot_out)
- {
-- if (!slot_out)
-+ if (smap.highpassed() || slot_out == smap.highwater())
- {
- slot_out = smap.segment.last();
- ++delta;
-- if (smap.highpassed() && !smap.highwater())
-+ if (!smap.highwater())
- smap.highpassed(false);
- }
-+ else
-+ {
-+ slot_out = smap.segment.first();
-+ --delta;
-+ }
-+ }
-+ if (delta < 0)
-+ {
- while (++delta <= 0 && slot_out)
- {
- if (smap.highpassed() && smap.highwater() == slot_out)
- smap.highpassed(false);
- slot_out = slot_out->prev();
- }
- }
- else if (delta > 0)
- {
-- if (!slot_out)
-- {
-- slot_out = smap.segment.first();
-- --delta;
-- }
- while (--delta >= 0 && slot_out)
- {
- slot_out = slot_out->next();
- if (slot_out == smap.highwater() && slot_out)
- smap.highpassed(true);
- }
- }
- }
-
-+bool Pass::collisionShift(Segment *seg, int dir, json * const dbgout) const
-+{
-+ ShiftCollider shiftcoll(dbgout);
-+ // bool isfirst = true;
-+ bool hasCollisions = false;
-+ Slot *start = seg->first(); // turn on collision fixing for the first slot
-+ Slot *end = NULL;
-+ bool moved = false;
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << "collisions" << json::array
-+ << json::flat << json::object << "num-loops" << m_numCollRuns << json::close;
-+#endif
-+
-+ while (start)
-+ {
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout) *dbgout << json::object << "phase" << "1" << "moves" << json::array;
-+#endif
-+ hasCollisions = false;
-+ end = NULL;
-+ // phase 1 : position shiftable glyphs, ignoring kernable glyphs
-+ for (Slot *s = start; s; s = s->next())
-+ {
-+ const SlotCollision * c = seg->collisionInfo(s);
-+ if (start && (c->flags() & (SlotCollision::COLL_FIX | SlotCollision::COLL_KERN)) == SlotCollision::COLL_FIX
-+ && !resolveCollisions(seg, s, start, shiftcoll, false, dir, moved, hasCollisions, dbgout))
-+ return false;
-+ if (s != start && (c->flags() & SlotCollision::COLL_END))
-+ {
-+ end = s->next();
-+ break;
-+ }
-+ }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::close << json::close; // phase-1
-+#endif
-+
-+ // phase 2 : loop until happy.
-+ for (int i = 0; i < m_numCollRuns - 1; ++i)
-+ {
-+ if (hasCollisions || moved)
-+ {
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::object << "phase" << "2a" << "loop" << i << "moves" << json::array;
-+#endif
-+ // phase 2a : if any shiftable glyphs are in collision, iterate backwards,
-+ // fixing them and ignoring other non-collided glyphs. Note that this handles ONLY
-+ // glyphs that are actually in collision from phases 1 or 2b, and working backwards
-+ // has the intended effect of breaking logjams.
-+ if (hasCollisions)
-+ {
-+ hasCollisions = false;
-+ #if 0
-+ moved = true;
-+ for (Slot *s = start; s != end; s = s->next())
-+ {
-+ SlotCollision * c = seg->collisionInfo(s);
-+ c->setShift(Position(0, 0));
-+ }
-+ #endif
-+ Slot *lend = end ? end->prev() : seg->last();
-+ Slot *lstart = start->prev();
-+ for (Slot *s = lend; s != lstart; s = s->prev())
-+ {
-+ SlotCollision * c = seg->collisionInfo(s);
-+ if (start && (c->flags() & (SlotCollision::COLL_FIX | SlotCollision::COLL_KERN | SlotCollision::COLL_ISCOL))
-+ == (SlotCollision::COLL_FIX | SlotCollision::COLL_ISCOL)) // ONLY if this glyph is still colliding
-+ {
-+ if (!resolveCollisions(seg, s, lend, shiftcoll, true, dir, moved, hasCollisions, dbgout))
-+ return false;
-+ c->setFlags(c->flags() | SlotCollision::COLL_TEMPLOCK);
-+ }
-+ }
-+ }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::close << json::close // phase 2a
-+ << json::object << "phase" << "2b" << "loop" << i << "moves" << json::array;
-+#endif
-+
-+ // phase 2b : redo basic diacritic positioning pass for ALL glyphs. Each successive loop adjusts
-+ // glyphs from their current adjusted position, which has the effect of gradually minimizing the
-+ // resulting adjustment; ie, the final result will be gradually closer to the original location.
-+ // Also it allows more flexibility in the final adjustment, since it is moving along the
-+ // possible 8 vectors from successively different starting locations.
-+ if (moved)
-+ {
-+ moved = false;
-+ for (Slot *s = start; s != end; s = s->next())
-+ {
-+ SlotCollision * c = seg->collisionInfo(s);
-+ if (start && (c->flags() & (SlotCollision::COLL_FIX | SlotCollision::COLL_TEMPLOCK
-+ | SlotCollision::COLL_KERN)) == SlotCollision::COLL_FIX
-+ && !resolveCollisions(seg, s, start, shiftcoll, false, dir, moved, hasCollisions, dbgout))
-+ return false;
-+ else if (c->flags() & SlotCollision::COLL_TEMPLOCK)
-+ c->setFlags(c->flags() & ~SlotCollision::COLL_TEMPLOCK);
-+ }
-+ }
-+ // if (!hasCollisions) // no, don't leave yet because phase 2b will continue to improve things
-+ // break;
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::close << json::close; // phase 2
-+#endif
-+ }
-+ }
-+ if (!end)
-+ break;
-+ start = NULL;
-+ for (Slot *s = end->prev(); s; s = s->next())
-+ {
-+ if (seg->collisionInfo(s)->flags() & SlotCollision::COLL_START)
-+ {
-+ start = s;
-+ break;
-+ }
-+ }
-+ }
-+ return true;
-+}
-+
-+bool Pass::collisionKern(Segment *seg, int dir, json * const dbgout) const
-+{
-+ KernCollider kerncoll(dbgout);
-+ Slot *start = seg->first();
-+ float ymin = 1e38f;
-+ float ymax = -1e38f;
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+
-+ // phase 3 : handle kerning of clusters
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::object << "phase" << "3" << "moves" << json::array;
-+#endif
-+
-+ for (Slot *s = seg->first(); s; s = s->next())
-+ {
-+ if (!gc.check(s->gid()))
-+ return false;
-+ const SlotCollision * c = seg->collisionInfo(s);
-+ const Rect &bbox = seg->theGlyphBBoxTemporary(s->gid());
-+ float y = s->origin().y + c->shift().y;
-+ ymax = max(y + bbox.tr.y, ymax);
-+ ymin = min(y + bbox.bl.y, ymin);
-+ if (start && (c->flags() & (SlotCollision::COLL_KERN | SlotCollision::COLL_FIX))
-+ == (SlotCollision::COLL_KERN | SlotCollision::COLL_FIX))
-+ resolveKern(seg, s, start, kerncoll, dir, ymin, ymax, dbgout);
-+ if (c->flags() & SlotCollision::COLL_END)
-+ start = NULL;
-+ if (c->flags() & SlotCollision::COLL_START)
-+ start = s;
-+ }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::close << json::close; // phase 3
-+#endif
-+ return true;
-+}
-+
-+bool Pass::collisionFinish(Segment *seg, GR_MAYBE_UNUSED json * const dbgout) const
-+{
-+ for (Slot *s = seg->first(); s; s = s->next())
-+ {
-+ SlotCollision *c = seg->collisionInfo(s);
-+ if (c->shift().x != 0 || c->shift().y != 0)
-+ {
-+ const Position newOffset = c->shift();
-+ const Position nullPosition(0, 0);
-+ c->setOffset(newOffset + c->offset());
-+ c->setShift(nullPosition);
-+ }
-+ }
-+// seg->positionSlots();
-+
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ *dbgout << json::close;
-+#endif
-+ return true;
-+}
-+
-+// Can slot s be kerned, or is it attached to something that can be kerned?
-+static bool inKernCluster(Segment *seg, Slot *s)
-+{
-+ SlotCollision *c = seg->collisionInfo(s);
-+ if (c->flags() & SlotCollision::COLL_KERN /** && c->flags() & SlotCollision::COLL_FIX **/ )
-+ return true;
-+ while (s->attachedTo())
-+ {
-+ s = s->attachedTo();
-+ c = seg->collisionInfo(s);
-+ if (c->flags() & SlotCollision::COLL_KERN /** && c->flags() & SlotCollision::COLL_FIX **/ )
-+ return true;
-+ }
-+ return false;
-+}
-+
-+// Fix collisions for the given slot.
-+// Return true if everything was fixed, false if there are still collisions remaining.
-+// isRev means be we are processing backwards.
-+bool Pass::resolveCollisions(Segment *seg, Slot *slotFix, Slot *start,
-+ ShiftCollider &coll, GR_MAYBE_UNUSED bool isRev, int dir, bool &moved, bool &hasCol,
-+ json * const dbgout) const
-+{
-+ Slot * nbor; // neighboring slot
-+ SlotCollision *cFix = seg->collisionInfo(slotFix);
-+ if (!coll.initSlot(seg, slotFix, cFix->limit(), cFix->margin(), cFix->marginWt(),
-+ cFix->shift(), cFix->offset(), dir, dbgout))
-+ return false;
-+ bool collides = false;
-+ // When we're processing forward, ignore kernable glyphs that preceed the target glyph.
-+ // When processing backward, don't ignore these until we pass slotFix.
-+ bool ignoreForKern = !isRev;
-+ bool rtl = dir & 1;
-+ Slot *base = slotFix;
-+ while (base->attachedTo())
-+ base = base->attachedTo();
-+ Position zero(0., 0.);
-+
-+ // Look for collisions with the neighboring glyphs.
-+ for (nbor = start; nbor; nbor = isRev ? nbor->prev() : nbor->next())
-+ {
-+ SlotCollision *cNbor = seg->collisionInfo(nbor);
-+ bool sameCluster = nbor->isChildOf(base);
-+ if (nbor != slotFix // don't process if this is the slot of interest
-+ && !(cNbor->flags() & SlotCollision::COLL_IGNORE) // don't process if ignoring
-+ && (nbor == base || sameCluster // process if in the same cluster as slotFix
-+ || !inKernCluster(seg, nbor) // or this cluster is not to be kerned
-+ || (rtl ^ ignoreForKern)) // or it comes before(ltr) or after(rtl)
-+ && (!isRev // if processing forwards then good to merge otherwise only:
-+ || !(cNbor->flags() & SlotCollision::COLL_FIX) // merge in immovable stuff
-+ || ((cNbor->flags() & SlotCollision::COLL_KERN) && !sameCluster) // ignore other kernable clusters
-+ || (cNbor->flags() & SlotCollision::COLL_ISCOL)) // test against other collided glyphs
-+ && !coll.mergeSlot(seg, nbor, cNbor->shift(), !ignoreForKern, sameCluster, collides, false, dbgout))
-+ return false;
-+ else if (nbor == slotFix)
-+ // Switching sides of this glyph - if we were ignoring kernable stuff before, don't anymore.
-+ ignoreForKern = !ignoreForKern;
-+
-+ if (nbor != start && (cNbor->flags() & (isRev ? SlotCollision::COLL_START : SlotCollision::COLL_END)))
-+ break;
-+ }
-+ bool isCol = false;
-+ if (collides || cFix->shift().x != 0.f || cFix->shift().y != 0.f)
-+ {
-+ Position shift = coll.resolve(seg, isCol, dbgout);
-+ // isCol has been set to true if a collision remains.
-+ if (std::fabs(shift.x) < 1e38f && std::fabs(shift.y) < 1e38f)
-+ {
-+ if (sqr(shift.x-cFix->shift().x) + sqr(shift.y-cFix->shift().y) >= m_colThreshold * m_colThreshold)
-+ moved = true;
-+ cFix->setShift(shift);
-+ if (slotFix->firstChild())
-+ {
-+ Rect bbox;
-+ Position here = slotFix->origin() + shift;
-+ float clusterMin = here.x;
-+ slotFix->firstChild()->finalise(seg, NULL, here, bbox, 0, clusterMin, rtl, false);
-+ }
-+ }
-+ }
-+ else
-+ {
-+ // This glyph is not colliding with anything.
-+#if !defined GRAPHITE2_NTRACING
-+ if (dbgout)
-+ {
-+ *dbgout << json::object
-+ << "missed" << objectid(dslot(seg, slotFix));
-+ coll.outputJsonDbg(dbgout, seg, -1);
-+ *dbgout << json::close;
-+ }
-+#endif
-+ }
-+
-+ // Set the is-collision flag bit.
-+ if (isCol)
-+ { cFix->setFlags(cFix->flags() | SlotCollision::COLL_ISCOL | SlotCollision::COLL_KNOWN); }
-+ else
-+ { cFix->setFlags((cFix->flags() & ~SlotCollision::COLL_ISCOL) | SlotCollision::COLL_KNOWN); }
-+ hasCol |= isCol;
-+ return true;
-+}
-+
-+float Pass::resolveKern(Segment *seg, Slot *slotFix, GR_MAYBE_UNUSED Slot *start, KernCollider &coll, int dir,
-+ float &ymin, float &ymax, json *const dbgout) const
-+{
-+ Slot *nbor; // neighboring slot
-+ float currSpace = 0.;
-+ bool collides = false;
-+ unsigned int space_count = 0;
-+ Slot *base = slotFix;
-+ while (base->attachedTo())
-+ base = base->attachedTo();
-+ SlotCollision *cFix = seg->collisionInfo(base);
-+ const GlyphCache &gc = seg->getFace()->glyphs();
-+
-+ if (base != slotFix)
-+ {
-+ cFix->setFlags(cFix->flags() | SlotCollision::COLL_KERN | SlotCollision::COLL_FIX);
-+ return 0;
-+ }
-+ bool seenEnd = (cFix->flags() & SlotCollision::COLL_END) != 0;
-+ bool isInit = false;
-+
-+ for (nbor = slotFix->next(); nbor; nbor = nbor->next())
-+ {
-+ if (nbor->isChildOf(base))
-+ continue;
-+ if (!gc.check(nbor->gid()))
-+ return 0.;
-+ const Rect &bb = seg->theGlyphBBoxTemporary(nbor->gid());
-+ SlotCollision *cNbor = seg->collisionInfo(nbor);
-+ if (bb.bl.y == 0.f && bb.tr.y == 0.f)
-+ {
-+ if (m_kernColls == InWord)
-+ break;
-+ // Add space for a space glyph.
-+ currSpace += nbor->advance();
-+ ++space_count;
-+ }
-+ else
-+ {
-+ space_count = 0;
-+ float y = nbor->origin().y + cNbor->shift().y;
-+ ymax = max(y + bb.tr.y, ymax);
-+ ymin = min(y + bb.bl.y, ymin);
-+ if (nbor != slotFix && !(cNbor->flags() & SlotCollision::COLL_IGNORE))
-+ {
-+ seenEnd = true;
-+ if (!isInit)
-+ {
-+ if (!coll.initSlot(seg, slotFix, cFix->limit(), cFix->margin(),
-+ cFix->shift(), cFix->offset(), dir, ymin, ymax, dbgout))
-+ return 0.;
-+ isInit = true;
-+ }
-+ collides |= coll.mergeSlot(seg, nbor, cNbor->shift(), currSpace, dir, dbgout);
-+ }
-+ }
-+ if (cNbor->flags() & SlotCollision::COLL_END)
-+ {
-+ if (seenEnd && space_count < 2)
-+ break;
-+ else
-+ seenEnd = true;
-+ }
-+ }
-+ if (collides)
-+ {
-+ Position mv = coll.resolve(seg, slotFix, dir, cFix->margin(), dbgout);
-+ coll.shift(mv, dir);
-+ Position delta = slotFix->advancePos() + mv - cFix->shift();
-+ slotFix->advance(delta);
-+ cFix->setShift(mv);
-+ return mv.x;
-+ }
-+ return 0.;
-+}
-+
-diff --git a/gfx/graphite2/src/Position.cpp b/gfx/graphite2/src/Position.cpp
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/Position.cpp
-@@ -0,0 +1,98 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#include "inc/Position.h"
-+#include <cmath>
-+
-+using namespace graphite2;
-+
-+bool Rect::hitTest(Rect &other)
-+{
-+ if (bl.x > other.tr.x) return false;
-+ if (tr.x < other.bl.x) return false;
-+ if (bl.y > other.tr.y) return false;
-+ if (tr.y < other.bl.y) return false;
-+ return true;
-+}
-+
-+Position Rect::overlap(Position &offset, Rect &other, Position &othero)
-+{
-+ float ax = (bl.x + offset.x) - (other.tr.x + othero.x);
-+ float ay = (bl.y + offset.y) - (other.tr.y + othero.y);
-+ float bx = (other.bl.x + othero.x) - (tr.x + offset.x);
-+ float by = (other.bl.y + othero.y) - (tr.y + offset.y);
-+ return Position((ax > bx ? ax : bx), (ay > by ? ay : by));
-+}
-+
-+float boundmin(float move, float lim1, float lim2, float &error)
-+{
-+ // error is always positive for easy comparison
-+ if (move < lim1 && move < lim2)
-+ { error = 0.; return move; }
-+ else if (lim1 < lim2)
-+ { error = std::fabs(move - lim1); return lim1; }
-+ else
-+ { error = std::fabs(move - lim2); return lim2; }
-+}
-+
-+#if 0
-+Position Rect::constrainedAvoid(Position &offset, Rect &box, Rect &sdbox, Position &other, Rect &obox, Rect &osdbox)
-+{
-+ // a = max, i = min, s = sum, d = diff
-+ float eax, eay, eix, eiy, eas, eis, ead, eid;
-+ float beste = INF;
-+ Position res;
-+ // calculate the movements in each direction and the error (amount of remaining overlap)
-+ // first param is movement, second and third are movement over the constraining box
-+ float ax = boundmin(obox.tr.x + other.x - box.bl.x - offset.x + 1, tr.x - offset.x, INF, &eax);
-+ float ay = boundmin(obox.tr.y + other.y - box.bl.y - offset.y + 1, tr.y - offset.y, INF, &eay);
-+ float ix = boundmin(obox.bl.x + other.x - box.tr.x - offset.x + 1, bl.x - offset.x, INF, &eix);
-+ float iy = boundmin(obox.bl.y + other.y - box.tr.y - offset.y + 1, bl.y - offset.y, INF, &eiy);
-+ float as = boundmin(ISQRT2 * (osdbox.tr.x + other.x + other.y - sdbox.bl.x - offset.x - offset.y) + 1, tr.x - offset.x, tr.y - offset.y, &eas);
-+ float is = boundmin(ISQRT2 * (osdbox.bl.x + other.x + other.y - sdbox.tr.x - offset.x - offset.y) + 1, bl.x - offset.x, bl.y - offset.y, &eis);
-+ float ad = boundmin(ISQRT2 * (osdbox.tr.y + other.x - other.y - sdbox.bl.y - offset.x + offset.y) + 1, tr.y - offset.y, tr.x - offset.x, &ead);
-+ float id = boundmin(ISQRT2 * (osdbox.bl.y + other.x - other.y - sdbox.tr.y - offset.x + offset.y) + 1, bl.y - offset.y, bl.x - offset.x, &eid);
-+
-+ if (eax < beste)
-+ { res = Position(ax, 0); beste = eax; }
-+ if (eay < beste)
-+ { res = Position(0, ay); beste = eay; }
-+ if (eix < beste)
-+ { res = Position(ix, 0); beste = eix; }
-+ if (eiy < beste)
-+ { res = Position(0, iy); beste = eiy; }
-+ if (SQRT2 * (eas) < beste)
-+ { res = Position(as, ad); beste = SQRT2 * (eas); }
-+ if (SQRT2 * (eis) < beste)
-+ { res = Position(is, is); beste = SQRT2 * (eis); }
-+ if (SQRT2 * (ead) < beste)
-+ { res = Position(ad, ad); beste = SQRT2 * (ead); }
-+ if (SQRT2 * (eid) < beste)
-+ { res = Position(id, id); beste = SQRT2 * (eid); }
-+ return res;
-+}
-+#endif
-+
-diff --git a/gfx/graphite2/src/SegCache.cpp b/gfx/graphite2/src/SegCache.cpp
---- a/gfx/graphite2/src/SegCache.cpp
-+++ b/gfx/graphite2/src/SegCache.cpp
-@@ -35,17 +35,17 @@ of the License or (at your option) any l
-
-
- using namespace graphite2;
-
- #ifndef GRAPHITE2_NSEGCACHE
-
- SegCache::SegCache(const SegCacheStore * store, const Features & feats)
- : m_prefixLength(ePrefixLength),
-- m_maxCachedSegLength(eMaxSpliceSize),
-+// m_maxCachedSegLength(eMaxSpliceSize),
- m_segmentCount(0),
- m_features(feats),
- m_totalAccessCount(0l), m_totalMisses(0l),
- m_purgeFactor(1.0f / (ePurgeFactor * store->maxSegmentCount()))
- {
- m_prefixes.raw = grzeroalloc<void*>(store->maxCmapGid() + 2);
- m_prefixes.range[SEG_CACHE_MIN_INDEX] = SEG_CACHE_UNSET_INDEX;
- m_prefixes.range[SEG_CACHE_MAX_INDEX] = SEG_CACHE_UNSET_INDEX;
-@@ -79,17 +79,17 @@ SegCache::~SegCache()
- {
- assert(m_prefixes.raw == NULL);
- }
-
- SegCacheEntry* SegCache::cache(SegCacheStore * store, const uint16* cmapGlyphs, size_t length, Segment * seg, size_t charOffset)
- {
- uint16 pos = 0;
- if (!length) return NULL;
-- assert(length < m_maxCachedSegLength);
-+// assert(length < m_maxCachedSegLength);
- SegCachePrefixArray pArray = m_prefixes;
- while (pos + 1 < m_prefixLength)
- {
- uint16 gid = (pos < length)? cmapGlyphs[pos] : 0;
- if (!pArray.array[gid].raw)
- {
- pArray.array[gid].raw = grzeroalloc<void*>(store->maxCmapGid() + 2);
- if (!pArray.array[gid].raw)
-diff --git a/gfx/graphite2/src/Segment.cpp b/gfx/graphite2/src/Segment.cpp
---- a/gfx/graphite2/src/Segment.cpp
-+++ b/gfx/graphite2/src/Segment.cpp
-@@ -31,48 +31,53 @@ of the License or (at your option) any l
- #include "inc/bits.h"
- #include "inc/Segment.h"
- #include "graphite2/Font.h"
- #include "inc/CharInfo.h"
- #include "inc/debug.h"
- #include "inc/Slot.h"
- #include "inc/Main.h"
- #include "inc/CmapCache.h"
--#include "inc/Bidi.h"
-+#include "inc/Collider.h"
- #include "graphite2/Segment.h"
-
-
- using namespace graphite2;
-
- Segment::Segment(unsigned int numchars, const Face* face, uint32 script, int textDir)
- : m_freeSlots(NULL),
- m_freeJustifies(NULL),
- m_charinfo(new CharInfo[numchars]),
-+ m_collisions(NULL),
- m_face(face),
- m_silf(face->chooseSilf(script)),
- m_first(NULL),
- m_last(NULL),
- m_bufSize(numchars + 10),
- m_numGlyphs(numchars),
- m_numCharinfo(numchars),
- m_passBits(m_silf->aPassBits() ? -1 : 0),
- m_defaultOriginal(0),
-- m_dir(textDir)
-+ m_dir(textDir),
-+ m_flags(((m_silf->flags() & 0x20) != 0) << 1)
- {
- freeSlot(newSlot());
- m_bufSize = log_binary(numchars)+1;
- }
-
- Segment::~Segment()
- {
- for (SlotRope::iterator i = m_slots.begin(); i != m_slots.end(); ++i)
- free(*i);
-- for (AttributeRope::iterator j = m_userAttrs.begin(); j != m_userAttrs.end(); ++j)
-- free(*j);
-+ for (AttributeRope::iterator i = m_userAttrs.begin(); i != m_userAttrs.end(); ++i)
-+ free(*i);
-+ for (JustifyRope::iterator i = m_justifies.begin(); i != m_justifies.end(); ++i)
-+ free(*i);
- delete[] m_charinfo;
-+ free(m_collisions);
- }
-
- #ifndef GRAPHITE2_NSEGCACHE
- SegmentScopeState Segment::setScope(Slot * firstSlot, Slot * lastSlot, size_t subLength)
- {
- SegmentScopeState state;
- state.numGlyphsOutsideScope = m_numGlyphs - subLength;
- state.realFirstSlot = m_first;
-@@ -159,28 +164,35 @@ void Segment::appendSlot(int id, int cid
- m_passBits &= theGlyph->attrs()[m_silf->aPassBits()]
- | (m_silf->numPasses() > 16 ? (theGlyph->attrs()[m_silf->aPassBits() + 1] << 16) : 0);
- }
-
- Slot *Segment::newSlot()
- {
- if (!m_freeSlots)
- {
-+ // check that the segment doesn't grow indefinintely
-+ if (m_numGlyphs > m_numCharinfo * MAX_SEG_GROWTH_FACTOR)
-+ return NULL;
- int numUser = m_silf->numUser();
- #if !defined GRAPHITE2_NTRACING
- if (m_face->logger()) ++numUser;
- #endif
- Slot *newSlots = grzeroalloc<Slot>(m_bufSize);
-- int16 *newAttrs = grzeroalloc<int16>(numUser * m_bufSize);
-- if (!newSlots || !newAttrs) return NULL;
-+ int16 *newAttrs = grzeroalloc<int16>(m_bufSize * numUser);
-+ if (!newSlots || !newAttrs)
-+ {
-+ free(newSlots);
-+ free(newAttrs);
-+ return NULL;
-+ }
- for (size_t i = 0; i < m_bufSize; i++)
- {
-+ ::new (newSlots + i) Slot(newAttrs + i * numUser);
- newSlots[i].next(newSlots + i + 1);
-- newSlots[i].userAttrs(newAttrs + i * numUser);
-- newSlots[i].setBidiClass(-1);
- }
- newSlots[m_bufSize - 1].next(NULL);
- newSlots[0].next(NULL);
- m_slots.push_back(newSlots);
- m_userAttrs.push_back(newAttrs);
- m_freeSlots = (m_bufSize > 1)? newSlots + 1 : NULL;
- return newSlots;
- }
-@@ -197,17 +209,17 @@ void Segment::freeSlot(Slot *aSlot)
- if (aSlot->attachedTo())
- aSlot->attachedTo()->removeChild(aSlot);
- while (aSlot->firstChild())
- {
- aSlot->firstChild()->attachTo(NULL);
- aSlot->removeChild(aSlot->firstChild());
- }
- // reset the slot incase it is reused
-- ::new (aSlot) Slot;
-+ ::new (aSlot) Slot(aSlot->userAttrs());
- memset(aSlot->userAttrs(), 0, m_silf->numUser() * sizeof(int16));
- // Update generation counter for debug
- #if !defined GRAPHITE2_NTRACING
- if (m_face->logger())
- ++aSlot->userAttrs()[m_silf->numUser()];
- #endif
- // update next pointer
- if (!m_freeSlots)
-@@ -301,16 +313,71 @@ void Segment::splice(size_t offset, size
- slot->set(*srcSlot, offset, m_silf->numUser(), m_silf->numJustLevels(), numChars);
- if (srcSlot->attachedTo()) slot->attachTo(indexmap[srcSlot->attachedTo()->index()]);
- if (srcSlot->nextSibling()) slot->m_sibling = indexmap[srcSlot->nextSibling()->index()];
- if (srcSlot->firstChild()) slot->m_child = indexmap[srcSlot->firstChild()->index()];
- }
- }
- #endif // GRAPHITE2_NSEGCACHE
-
-+// reverse the slots but keep diacritics in their same position after their bases
-+void Segment::reverseSlots()
-+{
-+ m_dir = m_dir ^ 64; // invert the reverse flag
-+ if (m_first == m_last) return; // skip 0 or 1 glyph runs
-+
-+ Slot *t = 0;
-+ Slot *curr = m_first;
-+ Slot *tlast;
-+ Slot *tfirst;
-+ Slot *out = 0;
-+
-+ while (curr && getSlotBidiClass(curr) == 16)
-+ curr = curr->next();
-+ if (!curr) return;
-+ tfirst = curr->prev();
-+ tlast = curr;
-+
-+ while (curr)
-+ {
-+ if (getSlotBidiClass(curr) == 16)
-+ {
-+ Slot *d = curr->next();
-+ while (d && getSlotBidiClass(d) == 16)
-+ d = d->next();
-+
-+ d = d ? d->prev() : m_last;
-+ Slot *p = out->next(); // one after the diacritics. out can't be null
-+ if (p)
-+ p->prev(d);
-+ else
-+ tlast = d;
-+ t = d->next();
-+ d->next(p);
-+ curr->prev(out);
-+ out->next(curr);
-+ }
-+ else // will always fire first time round the loop
-+ {
-+ if (out)
-+ out->prev(curr);
-+ t = curr->next();
-+ curr->next(out);
-+ out = curr;
-+ }
-+ curr = t;
-+ }
-+ out->prev(tfirst);
-+ if (tfirst)
-+ tfirst->next(out);
-+ else
-+ m_first = out;
-+ m_last = tlast;
-+}
-+
- void Segment::linkClusters(Slot *s, Slot * end)
- {
- end = end->next();
-
- for (; s != end && !s->isBase(); s = s->next());
- Slot * ls = s;
-
- if (m_dir & 1)
-@@ -330,39 +397,47 @@ void Segment::linkClusters(Slot *s, Slot
- if (!s->isBase()) continue;
-
- ls->sibling(s);
- ls = s;
- }
- }
- }
-
--Position Segment::positionSlots(const Font *font, Slot * iStart, Slot * iEnd)
-+Position Segment::positionSlots(const Font *font, Slot * iStart, Slot * iEnd, bool isRtl, bool isFinal)
- {
- Position currpos(0., 0.);
- float clusterMin = 0.;
- Rect bbox;
-
-+ if (currdir() != isRtl)
-+ {
-+ Slot *temp;
-+ reverseSlots();
-+ temp = iStart;
-+ iStart = iEnd;
-+ iEnd = temp;
-+ }
- if (!iStart) iStart = m_first;
- if (!iEnd) iEnd = m_last;
-
-- if (m_dir & 1)
-+ if (isRtl)
- {
- for (Slot * s = iEnd, * const end = iStart->prev(); s && s != end; s = s->prev())
- {
- if (s->isBase())
-- currpos = s->finalise(this, font, currpos, bbox, 0, clusterMin = currpos.x);
-+ currpos = s->finalise(this, font, currpos, bbox, 0, clusterMin = currpos.x, isRtl, isFinal);
- }
- }
- else
- {
- for (Slot * s = iStart, * const end = iEnd->next(); s && s != end; s = s->next())
- {
- if (s->isBase())
-- currpos = s->finalise(this, font, currpos, bbox, 0, clusterMin = currpos.x);
-+ currpos = s->finalise(this, font, currpos, bbox, 0, clusterMin = currpos.x, isRtl, isFinal);
- }
- }
- return currpos;
- }
-
-
- void Segment::associateChars(int offset, int numChars)
- {
-@@ -429,66 +504,28 @@ bool Segment::read_text(const Face *face
- {
- case gr_utf8: process_utf_data(*this, *face, addFeatures(*pFeats), utf8::const_iterator(pStart), nChars); break;
- case gr_utf16: process_utf_data(*this, *face, addFeatures(*pFeats), utf16::const_iterator(pStart), nChars); break;
- case gr_utf32: process_utf_data(*this, *face, addFeatures(*pFeats), utf32::const_iterator(pStart), nChars); break;
- }
- return true;
- }
-
--void Segment::prepare_pos(const Font * /*font*/)
-+void Segment::doMirror(uint16 aMirror)
- {
-- // copy key changeable metrics into slot (if any);
--}
--
--Slot *process_bidi(Slot *start, int level, int prelevel, int &nextLevel, int dirover, int isol, int &cisol, int &isolerr, int &embederr, int init, Segment *seg, uint8 aMirror, BracketPairStack &stack);
--void resolveImplicit(Slot *s, Segment *seg, uint8 aMirror);
--void resolveWhitespace(int baseLevel, Slot *s);
--Slot *resolveOrder(Slot * & s, const bool reordered, const int level = 0);
--
--void Segment::bidiPass(uint8 aBidi, int paradir, uint8 aMirror)
--{
-- if (slotCount() == 0)
-- return;
--
-- Slot *s;
-- int baseLevel = paradir ? 1 : 0;
-- unsigned int bmask = 0;
-- unsigned int ssize = 0;
-- for (s = first(); s; s = s->next())
-+ Slot * s;
-+ for (s = m_first; s; s = s->next())
- {
-- if (s->getBidiClass() == -1)
-- {
-- unsigned int bAttr = glyphAttr(s->gid(), aBidi);
-- s->setBidiClass((bAttr <= 22) * bAttr);
-- }
-- bmask |= (1 << s->getBidiClass());
-- s->setBidiLevel(baseLevel);
-- if (glyphAttr(s->gid(), aMirror) && s->getBidiClass() == 21)
-- ++ssize;
-- }
--
-- BracketPairStack bstack(ssize);
-- if (bmask & (paradir ? 0x2E7892 : 0x2E789C))
-- {
-- // O(8N) algorithm, with no working data beyond what is needed for processParens
-- int nextLevel = paradir;
-- int e, i, c;
-- process_bidi(first(), baseLevel, paradir, nextLevel, 0, 0, c = 0, i = 0, e = 0, 1, this, aMirror, bstack);
-- resolveImplicit(first(), this, aMirror);
-- resolveWhitespace(baseLevel, last());
-- s = resolveOrder(s = first(), baseLevel != 0);
-- if (s)
-- {
-- first(s); last(s->prev());
-- s->prev()->next(0); s->prev(0);
-- }
-- }
-- else if (!(dir() & 4) && baseLevel && aMirror)
-- {
-- for (s = first(); s; s = s->next())
-- {
-- unsigned short g = glyphAttr(s->gid(), aMirror);
-- if (g) s->setGlyph(this, g);
-- }
-+ unsigned short g = glyphAttr(s->gid(), aMirror);
-+ if (g && (!(dir() & 4) || !glyphAttr(s->gid(), aMirror + 1)))
-+ s->setGlyph(this, g);
- }
- }
-
-+bool Segment::initCollisions()
-+{
-+ m_collisions = grzeroalloc<SlotCollision>(slotCount());
-+ if (!m_collisions) return false;
-+
-+ for (Slot *p = m_first; p; p = p->next())
-+ ::new (collisionInfo(p)) SlotCollision(this, p);
-+ return true;
-+}
-diff --git a/gfx/graphite2/src/Silf.cpp b/gfx/graphite2/src/Silf.cpp
---- a/gfx/graphite2/src/Silf.cpp
-+++ b/gfx/graphite2/src/Silf.cpp
-@@ -46,23 +46,25 @@ Silf::Silf() throw()
- m_justs(0),
- m_numPasses(0),
- m_numJusts(0),
- m_sPass(0),
- m_pPass(0),
- m_jPass(0),
- m_bPass(0),
- m_flags(0),
-+ m_dir(0),
- m_aPseudo(0),
- m_aBreak(0),
- m_aUser(0),
- m_aBidi(0),
- m_aMirror(0),
- m_aPassBits(0),
- m_iMaxComp(0),
-+ m_aCollision(0),
- m_aLig(0),
- m_numPseudo(0),
- m_nClass(0),
- m_nLinear(0),
- m_gEndLine(0)
- {
- memset(&m_silfinfo, 0, sizeof m_silfinfo);
- }
-@@ -88,16 +90,20 @@ void Silf::releaseBuffers() throw()
-
-
- bool Silf::readGraphite(const byte * const silf_start, size_t lSilf, Face& face, uint32 version)
- {
- const byte * p = silf_start,
- * const silf_end = p + lSilf;
- Error e;
-
-+ if (e.test(version >= 0x00060000, E_BADSILFVERSION))
-+ {
-+ releaseBuffers(); return face.error(e);
-+ }
- if (version >= 0x00030000)
- {
- if (e.test(lSilf < 28, E_BADSIZE)) { releaseBuffers(); return face.error(e); }
- be::skip<int32>(p); // ruleVersion
- be::skip<uint16>(p,2); // passOffset & pseudosOffset
- }
- else if (e.test(lSilf < 20, E_BADSIZE)) { releaseBuffers(); return face.error(e); }
- const uint16 maxGlyph = be::read<uint16>(p);
-@@ -132,73 +138,88 @@ bool Silf::readGraphite(const byte * con
- for (uint8 i = 0; i < m_numJusts; i++)
- {
- ::new(m_justs + i) Justinfo(p[0], p[1], p[2], p[3]);
- be::skip<byte>(p,8);
- }
- }
-
- if (e.test(p + sizeof(uint16) + sizeof(uint8)*8 >= silf_end, E_BADENDJUSTS)) { releaseBuffers(); return face.error(e); }
-- m_aLig = be::read<uint16>(p);
-- m_aUser = be::read<uint8>(p);
-- m_iMaxComp = be::read<uint8>(p);
-- be::skip<byte>(p,5); // direction and 4 reserved bytes
-+ m_aLig = be::read<uint16>(p);
-+ m_aUser = be::read<uint8>(p);
-+ m_iMaxComp = be::read<uint8>(p);
-+ m_dir = be::read<uint8>(p) - 1;
-+ m_aCollision = be::read<uint8>(p);
-+ be::skip<byte>(p,3);
- be::skip<uint16>(p, be::read<uint8>(p)); // don't need critical features yet
- be::skip<byte>(p); // reserved
- if (e.test(p >= silf_end, E_BADCRITFEATURES)) { releaseBuffers(); return face.error(e); }
- be::skip<uint32>(p, be::read<uint8>(p)); // don't use scriptTag array.
- if (e.test(p + sizeof(uint16) + sizeof(uint32) >= silf_end, E_BADSCRIPTTAGS)) { releaseBuffers(); return face.error(e); }
- m_gEndLine = be::read<uint16>(p); // lbGID
- const byte * o_passes = p,
- * const passes_start = silf_start + be::read<uint32>(p);
-
- const size_t num_attrs = face.glyphs().numAttrs();
- if (e.test(m_aPseudo >= num_attrs, E_BADAPSEUDO)
- || e.test(m_aBreak >= num_attrs, E_BADABREAK)
- || e.test(m_aBidi >= num_attrs, E_BADABIDI)
- || e.test(m_aMirror>= num_attrs, E_BADAMIRROR)
-+ || e.test(m_aCollision && m_aCollision >= num_attrs - 5, E_BADACOLLISION)
- || e.test(m_numPasses > 128, E_BADNUMPASSES) || e.test(passes_start >= silf_end, E_BADPASSESSTART)
- || e.test(m_pPass < m_sPass, E_BADPASSBOUND) || e.test(m_pPass > m_numPasses, E_BADPPASS) || e.test(m_sPass > m_numPasses, E_BADSPASS)
- || e.test(m_jPass < m_pPass, E_BADJPASSBOUND) || e.test(m_jPass > m_numPasses, E_BADJPASS)
- || e.test((m_bPass != 0xFF && (m_bPass < m_jPass || m_bPass > m_numPasses)), E_BADBPASS)
- || e.test(m_aLig > 127, E_BADALIG))
- {
- releaseBuffers();
- return face.error(e);
- }
- be::skip<uint32>(p, m_numPasses);
- if (e.test(p + sizeof(uint16) >= passes_start, E_BADPASSESSTART)) { releaseBuffers(); return face.error(e); }
- m_numPseudo = be::read<uint16>(p);
- be::skip<uint16>(p, 3); // searchPseudo, pseudoSelector, pseudoShift
-- if (e.test(p + m_numPseudo*(sizeof(uint32) + sizeof(uint16)) >= passes_start, E_BADNUMPSEUDO))
-+ m_pseudos = new Pseudo[m_numPseudo];
-+ if (e.test(p + m_numPseudo*(sizeof(uint32) + sizeof(uint16)) >= passes_start, E_BADNUMPSEUDO)
-+ || e.test(!m_pseudos, E_OUTOFMEM))
- {
- releaseBuffers(); return face.error(e);
- }
-- m_pseudos = new Pseudo[m_numPseudo];
- for (int i = 0; i < m_numPseudo; i++)
- {
- m_pseudos[i].uid = be::read<uint32>(p);
- m_pseudos[i].gid = be::read<uint16>(p);
- }
-
- const size_t clen = readClassMap(p, passes_start - p, version, e);
-- if (e || e.test(p + clen > passes_start, E_BADPASSESSTART)) { releaseBuffers(); return face.error(e); }
-+ m_passes = new Pass[m_numPasses];
-+ if (e || e.test(p + clen > passes_start, E_BADPASSESSTART)
-+ || e.test(!m_passes, E_OUTOFMEM))
-+ { releaseBuffers(); return face.error(e); }
-
-- m_passes = new Pass[m_numPasses];
- for (size_t i = 0; i < m_numPasses; ++i)
- {
- const byte * const pass_start = silf_start + be::read<uint32>(o_passes),
- * const pass_end = silf_start + be::peek<uint32>(o_passes);
- face.error_context((face.error_context() & 0xFF00) + EC_ASILF + (i << 16));
-- if (e.test(pass_start > pass_end, E_BADPASSSTART) || e.test(pass_end > silf_end, E_BADPASSEND)) {
-+ if (e.test(pass_start > pass_end, E_BADPASSSTART)
-+ || e.test(pass_start < passes_start, E_BADPASSSTART)
-+ || e.test(pass_end > silf_end, E_BADPASSEND)) {
- releaseBuffers(); return face.error(e);
- }
-
-+ enum passtype pt = PASS_TYPE_UNKNOWN;
-+ if (i >= m_jPass) pt = PASS_TYPE_JUSTIFICATION;
-+ else if (i >= m_pPass) pt = PASS_TYPE_POSITIONING;
-+ else if (i >= m_sPass) pt = PASS_TYPE_SUBSTITUTE;
-+ else pt = PASS_TYPE_LINEBREAK;
-+
- m_passes[i].init(this);
-- if (!m_passes[i].readPass(pass_start, pass_end - pass_start, pass_start - silf_start, face, e))
-+ if (!m_passes[i].readPass(pass_start, pass_end - pass_start, pass_start - silf_start, face, pt,
-+ version, e))
- {
- releaseBuffers();
- return false;
- }
- }
-
- // fill in gr_faceinfo
- m_silfinfo.upem = face.glyphs().unitsPerEm();
-@@ -246,35 +267,38 @@ size_t Silf::readClassMap(const byte *p,
- uint32 max_off;
- if (version >= 0x00040000)
- max_off = readClassOffsets<uint32>(p, data_len, e);
- else
- max_off = readClassOffsets<uint16>(p, data_len, e);
-
- if (max_off == ERROROFFSET) return ERROROFFSET;
-
-+ if (e.test((int)max_off < m_nLinear + (m_nClass - m_nLinear) * 6, E_CLASSESTOOBIG))
-+ return ERROROFFSET;
-+
- // Check the linear offsets are sane, these must be monotonically increasing.
- for (const uint32 *o = m_classOffsets, * const o_end = o + m_nLinear; o != o_end; ++o)
- if (e.test(o[0] > o[1], E_BADCLASSOFFSET))
- return ERROROFFSET;
-
- // Fortunately the class data is all uint16s so we can decode these now
- m_classData = gralloc<uint16>(max_off);
- if (e.test(!m_classData, E_OUTOFMEM)) return ERROROFFSET;
- for (uint16 *d = m_classData, * const d_end = d + max_off; d != d_end; ++d)
- *d = be::read<uint16>(p);
-
- // Check the lookup class invariants for each non-linear class
- for (const uint32 *o = m_classOffsets + m_nLinear, * const o_end = m_classOffsets + m_nClass; o != o_end; ++o)
- {
- const uint16 * lookup = m_classData + *o;
-- if (e.test(*o > max_off - 4, E_HIGHCLASSOFFSET) // LookupClass doesn't stretch over max_off
-+ if (e.test(*o + 4 > max_off, E_HIGHCLASSOFFSET) // LookupClass doesn't stretch over max_off
- || e.test(lookup[0] == 0 // A LookupClass with no looks is a suspicious thing ...
-- || lookup[0] > (max_off - *o - 4)/2 // numIDs lookup pairs fits within (start of LookupClass' lookups array, max_off]
-- || lookup[3] != lookup[0] - lookup[1], E_BADCLASSLOOKUPINFO)) // rangeShift: numIDs - searchRange
-+ || lookup[0] * 2 + *o + 4 > max_off // numIDs lookup pairs fits within (start of LookupClass' lookups array, max_off]
-+ || lookup[3] + lookup[1] != lookup[0], E_BADCLASSLOOKUPINFO)) // rangeShift: numIDs - searchRange
- return ERROROFFSET;
- }
-
- return max_off;
- }
-
- uint16 Silf::findPseudo(uint32 uid) const
- {
-@@ -285,17 +309,17 @@ uint16 Silf::findPseudo(uint32 uid) cons
-
- uint16 Silf::findClassIndex(uint16 cid, uint16 gid) const
- {
- if (cid > m_nClass) return -1;
-
- const uint16 * cls = m_classData + m_classOffsets[cid];
- if (cid < m_nLinear) // output class being used for input, shouldn't happen
- {
-- for (unsigned int i = 0, n = m_classOffsets[cid + 1]; i < n; ++i, ++cls)
-+ for (unsigned int i = 0, n = m_classOffsets[cid + 1] - m_classOffsets[cid]; i < n; ++i, ++cls)
- if (*cls == gid) return i;
- return -1;
- }
- else
- {
- const uint16 * min = cls + 4, // lookups array
- * max = min + cls[0]*2; // lookups aray is numIDs (cls[0]) uint16 pairs long
- do
-@@ -326,90 +350,82 @@ uint16 Silf::getClassGlyph(uint16 cid, u
- }
- return 0;
- }
-
-
- bool Silf::runGraphite(Segment *seg, uint8 firstPass, uint8 lastPass, int dobidi) const
- {
- assert(seg != 0);
-- SlotMap map(*seg);
-+ SlotMap map(*seg, m_dir);
- FiniteStateMachine fsm(map, seg->getFace()->logger());
- vm::Machine m(map);
- unsigned int initSize = seg->slotCount();
- uint8 lbidi = m_bPass;
- #if !defined GRAPHITE2_NTRACING
- json * const dbgout = seg->getFace()->logger();
- #endif
-
- if (lastPass == 0)
- {
- if (firstPass == lastPass && lbidi == 0xFF)
- return true;
- lastPass = m_numPasses;
- }
-- if (firstPass <= lbidi && lastPass >= lbidi && dobidi)
-+ if ((firstPass < lbidi || (dobidi && firstPass == lbidi)) && (lastPass >= lbidi || (dobidi && lastPass + 1 == lbidi)))
- lastPass++;
- else
- lbidi = 0xFF;
-
- for (size_t i = firstPass; i < lastPass; ++i)
- {
- // bidi and mirroring
- if (i == lbidi)
- {
- #if !defined GRAPHITE2_NTRACING
- if (dbgout)
- {
- *dbgout << json::item << json::object
- << "id" << -1
- << "slots" << json::array;
-- seg->positionSlots(0);
-+ seg->positionSlots(0, 0, 0, m_dir);
- for(Slot * s = seg->first(); s; s = s->next())
- *dbgout << dslot(seg, s);
- *dbgout << json::close
- << "rules" << json::array << json::close
- << json::close;
- }
- #endif
--
-- if (!(seg->dir() & 2))
-- seg->bidiPass(m_aBidi, seg->dir() & 1, m_aMirror);
-- else if (m_aMirror)
-- {
-- Slot * s;
-- for (s = seg->first(); s; s = s->next())
-- {
-- unsigned short g = seg->glyphAttr(s->gid(), m_aMirror);
-- if (g && (!(seg->dir() & 4) || !seg->glyphAttr(s->gid(), m_aMirror + 1)))
-- s->setGlyph(seg, g);
-- }
-- }
-+ if (seg->currdir() != (m_dir & 1))
-+ seg->reverseSlots();
-+ if (m_aMirror && (seg->dir() & 3) == 3)
-+ seg->doMirror(m_aMirror);
- --i;
-+ lbidi = lastPass;
- --lastPass;
-- lbidi = 0xFF;
- continue;
- }
-
- #if !defined GRAPHITE2_NTRACING
- if (dbgout)
- {
- *dbgout << json::item << json::object
- << "id" << i+1
- << "slots" << json::array;
-- seg->positionSlots(0);
-+ seg->positionSlots(0, 0, 0, m_dir);
- for(Slot * s = seg->first(); s; s = s->next())
- *dbgout << dslot(seg, s);
- *dbgout << json::close;
- }
- #endif
-
- // test whether to reorder, prepare for positioning
-- if (i >= 32 || (seg->passBits() & (1 << i)) == 0)
-- m_passes[i].runGraphite(m, fsm);
-+ bool reverse = (lbidi == 0xFF) && (seg->currdir() != ((m_dir & 1) ^ m_passes[i].reverseDir()));
-+ if ((i >= 32 || (seg->passBits() & (1 << i)) == 0 || m_passes[i].collisionLoops())
-+ && !m_passes[i].runGraphite(m, fsm, reverse))
-+ return false;
- // only subsitution passes can change segment length, cached subsegments are short for their text
- if (m.status() != vm::Machine::finished
-- || (i < m_pPass && (seg->slotCount() > initSize * MAX_SEG_GROWTH_FACTOR
-- || (seg->slotCount() && seg->slotCount() * MAX_SEG_GROWTH_FACTOR < initSize))))
-+ || (seg->slotCount() && seg->slotCount() * MAX_SEG_GROWTH_FACTOR < initSize))
- return false;
- }
- return true;
- }
-diff --git a/gfx/graphite2/src/Slot.cpp b/gfx/graphite2/src/Slot.cpp
---- a/gfx/graphite2/src/Slot.cpp
-+++ b/gfx/graphite2/src/Slot.cpp
-@@ -24,34 +24,34 @@ Mozilla Public License (http://mozilla.o
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- #include "inc/Segment.h"
- #include "inc/Slot.h"
- #include "inc/Silf.h"
- #include "inc/CharInfo.h"
- #include "inc/Rule.h"
-+#include "inc/Collider.h"
-
-
- using namespace graphite2;
-
--Slot::Slot() :
-+Slot::Slot(int16 *user_attrs) :
- m_next(NULL), m_prev(NULL),
- m_glyphid(0), m_realglyphid(0), m_original(0), m_before(0), m_after(0),
- m_index(0), m_parent(NULL), m_child(NULL), m_sibling(NULL),
- m_position(0, 0), m_shift(0, 0), m_advance(0, 0),
- m_attach(0, 0), m_with(0, 0), m_just(0.),
-- m_flags(0), m_attLevel(0), m_bidiCls(-1), m_bidiLevel(0), m_justs(NULL)
-- // Do not set m_userAttr since it is set *before* new is called since this
-- // is used as a positional new to reset the GrSlot
-+ m_flags(0), m_attLevel(0), m_bidiCls(-1), m_bidiLevel(0),
-+ m_userAttr(user_attrs), m_justs(NULL)
- {
- }
-
- // take care, this does not copy any of the GrSlot pointer fields
--void Slot::set(const Slot & orig, int charOffset, size_t numUserAttr, size_t justLevels, size_t numChars)
-+void Slot::set(const Slot & orig, int charOffset, size_t sizeAttr, size_t justLevels, size_t numChars)
- {
- // leave m_next and m_prev unchanged
- m_glyphid = orig.m_glyphid;
- m_realglyphid = orig.m_realglyphid;
- m_original = orig.m_original + charOffset;
- if (charOffset + int(orig.m_before) < 0)
- m_before = 0;
- else
-@@ -68,95 +68,104 @@ void Slot::set(const Slot & orig, int ch
- m_advance = orig.m_advance;
- m_attach = orig.m_attach;
- m_with = orig.m_with;
- m_flags = orig.m_flags;
- m_attLevel = orig.m_attLevel;
- m_bidiCls = orig.m_bidiCls;
- m_bidiLevel = orig.m_bidiLevel;
- if (m_userAttr && orig.m_userAttr)
-- memcpy(m_userAttr, orig.m_userAttr, numUserAttr * sizeof(*m_userAttr));
-+ memcpy(m_userAttr, orig.m_userAttr, sizeAttr * sizeof(*m_userAttr));
- if (m_justs && orig.m_justs)
- memcpy(m_justs, orig.m_justs, SlotJustify::size_of(justLevels));
- }
-
- void Slot::update(int /*numGrSlots*/, int numCharInfo, Position &relpos)
- {
- m_before += numCharInfo;
- m_after += numCharInfo;
- m_position = m_position + relpos;
- }
-
--Position Slot::finalise(const Segment *seg, const Font *font, Position & base, Rect & bbox, uint8 attrLevel, float & clusterMin)
-+Position Slot::finalise(const Segment *seg, const Font *font, Position & base, Rect & bbox, uint8 attrLevel, float & clusterMin, bool rtl, bool isFinal)
- {
-+ SlotCollision *coll = NULL;
- if (attrLevel && m_attLevel > attrLevel) return Position(0, 0);
-- float scale = 1.0;
-- Position shift(m_shift.x * ((seg->dir() & 1) * -2 + 1) + m_just, m_shift.y);
-+ float scale = font ? font->scale() : 1.0f;
-+ Position shift(m_shift.x * (rtl * -2 + 1) + m_just, m_shift.y);
- float tAdvance = m_advance.x + m_just;
-+ if (isFinal && (coll = seg->collisionInfo(this)))
-+ {
-+ const Position &collshift = coll->offset();
-+ if (!(coll->flags() & SlotCollision::COLL_KERN) || rtl)
-+ shift = shift + collshift;
-+ }
- const GlyphFace * glyphFace = seg->getFace()->glyphs().glyphSafe(glyph());
- if (font)
- {
- scale = font->scale();
- shift *= scale;
- if (font->isHinted() && glyphFace)
-- tAdvance = (m_advance.x - glyphFace->theAdvance().x + m_just) * scale + font->advance(m_glyphid);
-+ tAdvance = (m_advance.x - glyphFace->theAdvance().x + m_just) * scale + font->advance(glyph());
- else
- tAdvance *= scale;
- }
- Position res;
-
- m_position = base + shift;
- if (!m_parent)
- {
- res = base + Position(tAdvance, m_advance.y * scale);
-- clusterMin = base.x;
-+ clusterMin = m_position.x;
- }
- else
- {
- float tAdv;
- m_position += (m_attach - m_with) * scale;
-- tAdv = m_advance.x >= 0.5 ? m_position.x + tAdvance - shift.x : 0.f;
-+ tAdv = m_advance.x >= 0.5f ? m_position.x + tAdvance - shift.x : 0.f;
- res = Position(tAdv, 0);
-- if ((m_advance.x >= 0.5 || m_position.x < 0) && m_position.x < clusterMin) clusterMin = m_position.x;
-+ if ((m_advance.x >= 0.5f || m_position.x < 0) && m_position.x < clusterMin) clusterMin = m_position.x;
- }
-
- if (glyphFace)
- {
- Rect ourBbox = glyphFace->theBBox() * scale + m_position;
- bbox = bbox.widen(ourBbox);
- }
-
- if (m_child && m_child != this && m_child->attachedTo() == this)
- {
-- Position tRes = m_child->finalise(seg, font, m_position, bbox, attrLevel, clusterMin);
-- if ((!m_parent || m_advance.x >= 0.5) && tRes.x > res.x) res = tRes;
-+ Position tRes = m_child->finalise(seg, font, m_position, bbox, attrLevel, clusterMin, rtl, isFinal);
-+ if ((!m_parent || m_advance.x >= 0.5f) && tRes.x > res.x) res = tRes;
- }
-
- if (m_parent && m_sibling && m_sibling != this && m_sibling->attachedTo() == m_parent)
- {
-- Position tRes = m_sibling->finalise(seg, font, base, bbox, attrLevel, clusterMin);
-+ Position tRes = m_sibling->finalise(seg, font, base, bbox, attrLevel, clusterMin, rtl, isFinal);
- if (tRes.x > res.x) res = tRes;
- }
-
- if (!m_parent && clusterMin < base.x)
- {
-- Position adj = Position(base.x - clusterMin, 0.);
-+ Position adj = Position(m_position.x - clusterMin, 0.);
- res += adj;
- m_position += adj;
- if (m_child) m_child->floodShift(adj);
- }
- return res;
- }
-
--int32 Slot::clusterMetric(const Segment *seg, uint8 metric, uint8 attrLevel)
-+int32 Slot::clusterMetric(const Segment *seg, uint8 metric, uint8 attrLevel, bool rtl)
- {
- Position base;
-+ if (glyph() >= seg->getFace()->glyphs().numGlyphs())
-+ return 0;
- Rect bbox = seg->theGlyphBBoxTemporary(glyph());
- float clusterMin = 0.;
-- Position res = finalise(seg, NULL, base, bbox, attrLevel, clusterMin);
-+ Position res = finalise(seg, NULL, base, bbox, attrLevel, clusterMin, rtl, false);
-
- switch (metrics(metric))
- {
- case kgmetLsb :
- return static_cast<uint32>(bbox.bl.x);
- case kgmetRsb :
- return static_cast<uint32>(res.x - bbox.tr.x);
- case kgmetBbTop :
-@@ -175,19 +184,20 @@ int32 Slot::clusterMetric(const Segment
- return static_cast<uint32>(res.x);
- case kgmetAdvHeight :
- return static_cast<uint32>(res.y);
- default :
- return 0;
- }
- }
-
-+#define SLOTGETCOLATTR(x) { SlotCollision *c = seg->collisionInfo(this); return c ? int(c-> x) : 0; }
-+
- int Slot::getAttr(const Segment *seg, attrCode ind, uint8 subindex) const
- {
-- if (!this) return 0;
- if (ind == gr_slatUserDefnV1)
- {
- ind = gr_slatUserDefn;
- subindex = 0;
- }
- else if (ind >= gr_slatJStretch && ind < gr_slatJStretch + 20 && ind != gr_slatJWidth)
- {
- int indx = ind - gr_slatJStretch;
-@@ -205,37 +215,66 @@ int Slot::getAttr(const Segment *seg, at
- case gr_slatAttYOff : return 0;
- case gr_slatAttWithX : return int(m_with.x);
- case gr_slatAttWithY : return int(m_with.y);
- case gr_slatAttWithXOff:
- case gr_slatAttWithYOff:return 0;
- case gr_slatAttLevel : return m_attLevel;
- case gr_slatBreak : return seg->charinfo(m_original)->breakWeight();
- case gr_slatCompRef : return 0;
-- case gr_slatDir : if (m_bidiCls == -1)
-- const_cast<Slot *>(this)->setBidiClass(seg->glyphAttr(gid(), seg->silf()->aBidi()));
-- return m_bidiCls;
-+ case gr_slatDir : return seg->dir() & 1;
- case gr_slatInsert : return isInsertBefore();
- case gr_slatPosX : return int(m_position.x); // but need to calculate it
- case gr_slatPosY : return int(m_position.y);
- case gr_slatShiftX : return int(m_shift.x);
- case gr_slatShiftY : return int(m_shift.y);
- case gr_slatMeasureSol: return -1; // err what's this?
- case gr_slatMeasureEol: return -1;
- case gr_slatJWidth: return int(m_just);
- case gr_slatUserDefn : return m_userAttr[subindex];
- case gr_slatSegSplit : return seg->charinfo(m_original)->flags() & 3;
- case gr_slatBidiLevel: return m_bidiLevel;
-- default : return 0;
-+ case gr_slatColFlags : { SlotCollision *c = seg->collisionInfo(this); return c ? c->flags() : 0; }
-+ case gr_slatColLimitblx : SLOTGETCOLATTR(limit().bl.x)
-+ case gr_slatColLimitbly : SLOTGETCOLATTR(limit().bl.y)
-+ case gr_slatColLimittrx : SLOTGETCOLATTR(limit().tr.x)
-+ case gr_slatColLimittry : SLOTGETCOLATTR(limit().tr.y)
-+ case gr_slatColShiftx : SLOTGETCOLATTR(offset().x)
-+ case gr_slatColShifty : SLOTGETCOLATTR(offset().y)
-+ case gr_slatColMargin : SLOTGETCOLATTR(margin())
-+ case gr_slatColMarginWt : SLOTGETCOLATTR(marginWt())
-+ case gr_slatColExclGlyph : SLOTGETCOLATTR(exclGlyph())
-+ case gr_slatColExclOffx : SLOTGETCOLATTR(exclOffset().x)
-+ case gr_slatColExclOffy : SLOTGETCOLATTR(exclOffset().y)
-+ case gr_slatSeqClass : SLOTGETCOLATTR(seqClass())
-+ case gr_slatSeqProxClass : SLOTGETCOLATTR(seqProxClass())
-+ case gr_slatSeqOrder : SLOTGETCOLATTR(seqOrder())
-+ case gr_slatSeqAboveXoff : SLOTGETCOLATTR(seqAboveXoff())
-+ case gr_slatSeqAboveWt : SLOTGETCOLATTR(seqAboveWt())
-+ case gr_slatSeqBelowXlim : SLOTGETCOLATTR(seqBelowXlim())
-+ case gr_slatSeqBelowWt : SLOTGETCOLATTR(seqBelowWt())
-+ case gr_slatSeqValignHt : SLOTGETCOLATTR(seqValignHt())
-+ case gr_slatSeqValignWt : SLOTGETCOLATTR(seqValignWt())
-+ default : return 0;
- }
- }
-
-+#define SLOTCOLSETATTR(x) { \
-+ SlotCollision *c = seg->collisionInfo(this); \
-+ if (c) { c-> x ; c->setFlags(c->flags() & ~SlotCollision::COLL_KNOWN); } \
-+ break; }
-+#define SLOTCOLSETCOMPLEXATTR(t, y, x) { \
-+ SlotCollision *c = seg->collisionInfo(this); \
-+ if (c) { \
-+ const t &s = c-> y; \
-+ c-> x ; c->setFlags(c->flags() & ~SlotCollision::COLL_KNOWN); } \
-+ break; }
-+
- void Slot::setAttr(Segment *seg, attrCode ind, uint8 subindex, int16 value, const SlotMap & map)
- {
-- if (!this) return;
- if (ind == gr_slatUserDefnV1)
- {
- ind = gr_slatUserDefn;
- subindex = 0;
- }
- else if (ind >= gr_slatJStretch && ind < gr_slatJStretch + 20 && ind != gr_slatJWidth)
- {
- int indx = ind - gr_slatJStretch;
-@@ -247,22 +286,22 @@ void Slot::setAttr(Segment *seg, attrCod
- case gr_slatAdvX : m_advance.x = value; break;
- case gr_slatAdvY : m_advance.y = value; break;
- case gr_slatAttTo :
- {
- const uint16 idx = uint16(value);
- if (idx < map.size() && map[idx])
- {
- Slot *other = map[idx];
-- if (other == this) break;
-+ if (other == this || other == m_parent) break;
- if (m_parent) m_parent->removeChild(this);
-- if (other->child(this))
-+ if (!other->isChildOf(this) && other->child(this))
- {
- attachTo(other);
-- if (((seg->dir() & 1) != 0) ^ (idx > subindex))
-+ if ((map.dir() != 0) ^ (idx > subindex))
- m_with = Position(advance(), 0);
- else // normal match to previous root
- m_attach = Position(other->advance(), 0);
- }
- }
- break;
- }
- case gr_slatAttX : m_attach.x = value; break;
-@@ -275,29 +314,52 @@ void Slot::setAttr(Segment *seg, attrCod
- case gr_slatAttWithYOff : break;
- case gr_slatAttLevel :
- m_attLevel = byte(value);
- break;
- case gr_slatBreak :
- seg->charinfo(m_original)->breakWeight(value);
- break;
- case gr_slatCompRef : break; // not sure what to do here
-- case gr_slatDir : m_bidiCls = value; break;
-+ case gr_slatDir : break;
- case gr_slatInsert :
- markInsertBefore(value? true : false);
- break;
- case gr_slatPosX : break; // can't set these here
- case gr_slatPosY : break;
- case gr_slatShiftX : m_shift.x = value; break;
- case gr_slatShiftY : m_shift.y = value; break;
- case gr_slatMeasureSol : break;
- case gr_slatMeasureEol : break;
- case gr_slatJWidth : just(value); break;
- case gr_slatSegSplit : seg->charinfo(m_original)->addflags(value & 3); break;
- case gr_slatUserDefn : m_userAttr[subindex] = value; break;
-+ case gr_slatColFlags : {
-+ SlotCollision *c = seg->collisionInfo(this);
-+ if (c)
-+ c->setFlags(value);
-+ break; }
-+ case gr_slatColLimitblx : SLOTCOLSETCOMPLEXATTR(Rect, limit(), setLimit(Rect(Position(value, s.bl.y), s.tr)))
-+ case gr_slatColLimitbly : SLOTCOLSETCOMPLEXATTR(Rect, limit(), setLimit(Rect(Position(s.bl.x, value), s.tr)))
-+ case gr_slatColLimittrx : SLOTCOLSETCOMPLEXATTR(Rect, limit(), setLimit(Rect(s.bl, Position(value, s.tr.y))))
-+ case gr_slatColLimittry : SLOTCOLSETCOMPLEXATTR(Rect, limit(), setLimit(Rect(s.bl, Position(s.tr.x, value))))
-+ case gr_slatColMargin : SLOTCOLSETATTR(setMargin(value))
-+ case gr_slatColMarginWt : SLOTCOLSETATTR(setMarginWt(value))
-+ case gr_slatColExclGlyph : SLOTCOLSETATTR(setExclGlyph(value))
-+ case gr_slatColExclOffx : SLOTCOLSETCOMPLEXATTR(Position, exclOffset(), setExclOffset(Position(value, s.y)))
-+ case gr_slatColExclOffy : SLOTCOLSETCOMPLEXATTR(Position, exclOffset(), setExclOffset(Position(s.x, value)))
-+ case gr_slatSeqClass : SLOTCOLSETATTR(setSeqClass(value))
-+ case gr_slatSeqProxClass : SLOTCOLSETATTR(setSeqProxClass(value))
-+ case gr_slatSeqOrder : SLOTCOLSETATTR(setSeqOrder(value))
-+ case gr_slatSeqAboveXoff : SLOTCOLSETATTR(setSeqAboveXoff(value))
-+ case gr_slatSeqAboveWt : SLOTCOLSETATTR(setSeqAboveWt(value))
-+ case gr_slatSeqBelowXlim : SLOTCOLSETATTR(setSeqBelowXlim(value))
-+ case gr_slatSeqBelowWt : SLOTCOLSETATTR(setSeqBelowWt(value))
-+ case gr_slatSeqValignHt : SLOTCOLSETATTR(setSeqValignHt(value))
-+ case gr_slatSeqValignWt : SLOTCOLSETATTR(setSeqValignWt(value))
- default :
- break;
- }
- }
-
- int Slot::getJustify(const Segment *seg, uint8 level, uint8 subindex) const
- {
- if (level && level >= seg->silf()->numJustLevels()) return 0;
-@@ -369,46 +431,54 @@ bool Slot::removeChild(Slot *ap)
- }
-
- bool Slot::removeSibling(Slot *ap)
- {
- if (this == ap || !m_sibling) return false;
- else if (ap == m_sibling)
- {
- m_sibling = m_sibling->nextSibling();
-+ ap->sibling(NULL);
- return true;
- }
- else
- return m_sibling->removeSibling(ap);
- return true;
- }
-
- void Slot::setGlyph(Segment *seg, uint16 glyphid, const GlyphFace * theGlyph)
- {
- m_glyphid = glyphid;
-+ m_bidiCls = -1;
- if (!theGlyph)
- {
- theGlyph = seg->getFace()->glyphs().glyphSafe(glyphid);
- if (!theGlyph)
- {
- m_realglyphid = 0;
- m_advance = Position(0.,0.);
- return;
- }
- }
- m_realglyphid = theGlyph->attrs()[seg->silf()->aPseudo()];
-+ if (m_realglyphid > seg->getFace()->glyphs().numGlyphs())
-+ m_realglyphid = 0;
- const GlyphFace *aGlyph = theGlyph;
- if (m_realglyphid)
- {
- aGlyph = seg->getFace()->glyphs().glyphSafe(m_realglyphid);
- if (!aGlyph) aGlyph = theGlyph;
- }
- m_advance = Position(aGlyph->theAdvance().x, 0.);
- if (seg->silf()->aPassBits())
-+ {
- seg->mergePassBits(theGlyph->attrs()[seg->silf()->aPassBits()]);
-+ if (seg->silf()->numPasses() > 16)
-+ seg->mergePassBits(theGlyph->attrs()[seg->silf()->aPassBits()+1] << 16);
-+ }
- }
-
- void Slot::floodShift(Position adj)
- {
- m_position += adj;
- if (m_child) m_child->floodShift(adj);
- if (m_sibling) m_sibling->floodShift(adj);
- }
-@@ -420,8 +490,35 @@ void SlotJustify::LoadSlot(const Slot *s
- Justinfo *justs = seg->silf()->justAttrs() + i;
- int16 *v = values + i * NUMJUSTPARAMS;
- v[0] = seg->glyphAttr(s->gid(), justs->attrStretch());
- v[1] = seg->glyphAttr(s->gid(), justs->attrShrink());
- v[2] = seg->glyphAttr(s->gid(), justs->attrStep());
- v[3] = seg->glyphAttr(s->gid(), justs->attrWeight());
- }
- }
-+
-+Slot * Slot::nextInCluster(const Slot *s) const
-+{
-+ Slot *base;
-+ if (s->firstChild())
-+ return s->firstChild();
-+ else if (s->nextSibling())
-+ return s->nextSibling();
-+ while ((base = s->attachedTo()))
-+ {
-+ // if (base->firstChild() == s && base->nextSibling())
-+ if (base->nextSibling())
-+ return base->nextSibling();
-+ s = base;
-+ }
-+ return NULL;
-+}
-+
-+bool Slot::isChildOf(const Slot *base) const
-+{
-+ if (m_parent == base)
-+ return true;
-+ else if (!m_parent)
-+ return false;
-+ else
-+ return m_parent->isChildOf(base);
-+}
-diff --git a/gfx/graphite2/src/Sparse.cpp b/gfx/graphite2/src/Sparse.cpp
---- a/gfx/graphite2/src/Sparse.cpp
-+++ b/gfx/graphite2/src/Sparse.cpp
-@@ -25,17 +25,17 @@ License, as published by the Free Softwa
- of the License or (at your option) any later version.
- */
- #include <cassert>
- #include "inc/Sparse.h"
- #include "inc/bits.h"
-
- using namespace graphite2;
-
--sparse::chunk sparse::empty_chunk = {0,0};
-+const sparse::chunk sparse::empty_chunk = {0,0};
-
- sparse::~sparse() throw()
- {
- if (m_array.map == &empty_chunk) return;
- free(m_array.values);
- }
-
-
-diff --git a/gfx/graphite2/src/TtfUtil.cpp b/gfx/graphite2/src/TtfUtil.cpp
---- a/gfx/graphite2/src/TtfUtil.cpp
-+++ b/gfx/graphite2/src/TtfUtil.cpp
-@@ -57,18 +57,20 @@ Description
- Forward declarations
- ***********************************************************************************************/
-
- /***********************************************************************************************
- Local Constants and static variables
- ***********************************************************************************************/
- namespace
- {
-+#ifdef ALL_TTFUTILS
- // max number of components allowed in composite glyphs
- const int kMaxGlyphComponents = 8;
-+#endif
-
- template <int R, typename T>
- inline float fixed_to_float(const T f) {
- return float(f)/float(2^R);
- }
-
- /*----------------------------------------------------------------------------------------------
- Table of standard Postscript glyph names. From Martin Hosken. Disagress with ttfdump.exe
-@@ -222,69 +224,79 @@ bool GetTableInfo(const Tag TableTag, co
- /*----------------------------------------------------------------------------------------------
- Check the specified table. Tests depend on the table type.
- Return true if successful, false otherwise.
- ----------------------------------------------------------------------------------------------*/
- bool CheckTable(const Tag TableId, const void * pTable, size_t lTableSize)
- {
- using namespace Sfnt;
-
-- if (pTable == 0) return false;
-+ if (pTable == 0 || lTableSize < 4) return false;
-
- switch(TableId)
- {
- case Tag::cmap: // cmap
- {
- const Sfnt::CharacterCodeMap * const pCmap
- = reinterpret_cast<const Sfnt::CharacterCodeMap *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::CharacterCodeMap))
-+ return false;
- return be::swap(pCmap->version) == 0;
- }
-
- case Tag::head: // head
- {
- const Sfnt::FontHeader * const pHead
- = reinterpret_cast<const Sfnt::FontHeader *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::FontHeader))
-+ return false;
- bool r = be::swap(pHead->version) == OneFix
- && be::swap(pHead->magic_number) == FontHeader::MagicNumber
- && be::swap(pHead->glyph_data_format)
- == FontHeader::GlypDataFormat
- && (be::swap(pHead->index_to_loc_format)
- == FontHeader::ShortIndexLocFormat
- || be::swap(pHead->index_to_loc_format)
- == FontHeader::LongIndexLocFormat)
- && sizeof(FontHeader) <= lTableSize;
- return r;
- }
-
- case Tag::post: // post
- {
- const Sfnt::PostScriptGlyphName * const pPost
- = reinterpret_cast<const Sfnt::PostScriptGlyphName *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::PostScriptGlyphName))
-+ return false;
- const fixed format = be::swap(pPost->format);
- bool r = format == PostScriptGlyphName::Format1
- || format == PostScriptGlyphName::Format2
- || format == PostScriptGlyphName::Format3
- || format == PostScriptGlyphName::Format25;
- return r;
- }
-
- case Tag::hhea: // hhea
- {
- const Sfnt::HorizontalHeader * pHhea =
- reinterpret_cast<const Sfnt::HorizontalHeader *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::HorizontalHeader))
-+ return false;
- bool r = be::swap(pHhea->version) == OneFix
- && be::swap(pHhea->metric_data_format) == 0
- && sizeof (Sfnt::HorizontalHeader) <= lTableSize;
- return r;
- }
-
- case Tag::maxp: // maxp
- {
- const Sfnt::MaximumProfile * pMaxp =
- reinterpret_cast<const Sfnt::MaximumProfile *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::MaximumProfile))
-+ return false;
- bool r = be::swap(pMaxp->version) == OneFix
- && sizeof(Sfnt::MaximumProfile) <= lTableSize;
- return r;
- }
-
- case Tag::OS_2: // OS/2
- {
- const Sfnt::Compatibility * pOs2
-@@ -319,16 +331,18 @@ bool CheckTable(const Tag TableId, const
- return false;
- break;
- }
-
- case Tag::name:
- {
- const Sfnt::FontNames * pName
- = reinterpret_cast<const Sfnt::FontNames *>(pTable);
-+ if (lTableSize < sizeof(Sfnt::FontNames))
-+ return false;
- return be::swap(pName->format) == 0;
- }
-
- default:
- break;
- }
-
- return true;
-@@ -791,27 +805,27 @@ bool HorMetrics(gid16 nGlyphId, const vo
- reinterpret_cast<const Sfnt::HorizontalMetric *>(pHmtx);
-
- const Sfnt::HorizontalHeader * phhea =
- reinterpret_cast<const Sfnt::HorizontalHeader *>(pHhea);
-
- size_t cLongHorMetrics = be::swap(phhea->num_long_hor_metrics);
- if (nGlyphId < cLongHorMetrics)
- { // glyph id is acceptable
-- if (nGlyphId * sizeof(Sfnt::HorizontalMetric) >= lHmtxSize) return false;
-+ if ((nGlyphId + 1) * sizeof(Sfnt::HorizontalMetric) > lHmtxSize) return false;
- nAdvWid = be::swap(phmtx[nGlyphId].advance_width);
- nLsb = be::swap(phmtx[nGlyphId].left_side_bearing);
- }
- else
- {
- // guard against bad glyph id
- size_t lLsbOffset = sizeof(Sfnt::HorizontalMetric) * cLongHorMetrics +
- sizeof(int16) * (nGlyphId - cLongHorMetrics); // offset in bytes
- // We test like this as LsbOffset is an offset not a length.
-- if (lLsbOffset > lHmtxSize - sizeof(int16))
-+ if (lLsbOffset >= lHmtxSize - sizeof(int16) || cLongHorMetrics == 0)
- {
- nLsb = 0;
- return false;
- }
- nAdvWid = be::swap(phmtx[cLongHorMetrics - 1].advance_width);
- nLsb = be::peek<int16>(reinterpret_cast<const byte *>(phmtx) + lLsbOffset);
- }
-
-@@ -833,31 +847,33 @@ const void * FindCmapSubtable(const void
- {
- if (be::swap(pTable->encoding[i].platform_id) == nPlatformId &&
- (nEncodingId == -1 || be::swap(pTable->encoding[i].platform_specific_id) == nEncodingId))
- {
- uint32 offset = be::swap(pTable->encoding[i].offset);
- const uint8 * pRtn = reinterpret_cast<const uint8 *>(pCmap) + offset;
- if (length)
- {
-- if (offset > length) return NULL;
-+ if (offset > length - 2) return NULL;
- uint16 format = be::read<uint16>(pRtn);
- if (format == 4)
- {
-+ if (offset > length - 4) return NULL;
- uint16 subTableLength = be::peek<uint16>(pRtn);
- if (i + 1 == csuPlatforms)
- {
- if (subTableLength > length - offset)
- return NULL;
- }
- else if (subTableLength > be::swap(pTable->encoding[i+1].offset))
- return NULL;
- }
- if (format == 12)
- {
-+ if (offset > length - 6) return NULL;
- uint32 subTableLength = be::peek<uint32>(pRtn);
- if (i + 1 == csuPlatforms)
- {
- if (subTableLength > length - offset)
- return NULL;
- }
- else if (subTableLength > be::swap(pTable->encoding[i+1].offset))
- return NULL;
-@@ -868,48 +884,80 @@ const void * FindCmapSubtable(const void
- }
-
- return 0;
- }
-
- /*----------------------------------------------------------------------------------------------
- Check the Microsoft Unicode subtable for expected values
- ----------------------------------------------------------------------------------------------*/
--bool CheckCmapSubtable4(const void * pCmapSubtable4)
-+bool CheckCmapSubtable4(const void * pCmapSubtable4, size_t table_len /*, unsigned int maxgid*/)
- {
- if (!pCmapSubtable4) return false;
- const Sfnt::CmapSubTable * pTable = reinterpret_cast<const Sfnt::CmapSubTable *>(pCmapSubtable4);
-- // Bob H says ome freeware TT fonts have version 1 (eg, CALIGULA.TTF)
-+ // Bob H say some freeware TT fonts have version 1 (eg, CALIGULA.TTF)
- // so don't check subtable version. 21 Mar 2002 spec changes version to language.
- if (be::swap(pTable->format) != 4) return false;
- const Sfnt::CmapSubTableFormat4 * pTable4 = reinterpret_cast<const Sfnt::CmapSubTableFormat4 *>(pCmapSubtable4);
- uint16 length = be::swap(pTable4->length);
-+ if (length > table_len)
-+ return false;
- if (length < sizeof(Sfnt::CmapSubTableFormat4))
- return false;
- uint16 nRanges = be::swap(pTable4->seg_count_x2) >> 1;
- if (length < sizeof(Sfnt::CmapSubTableFormat4) + 4 * nRanges * sizeof(uint16))
- return false;
- // check last range is properly terminated
- uint16 chEnd = be::peek<uint16>(pTable4->end_code + nRanges - 1);
-- return (chEnd == 0xFFFF);
-+ if (chEnd != 0xFFFF)
-+ return false;
-+#if 0
-+ int lastend = -1;
-+ for (int i = 0; i < nRanges; ++i)
-+ {
-+ uint16 end = be::peek<uint16>(pTable4->end_code + i);
-+ uint16 start = be::peek<uint16>(pTable4->end_code + nRanges + 1 + i);
-+ int16 delta = be::peek<int16>(pTable4->end_code + 2*nRanges + 1 + i);
-+ uint16 offset = be::peek<uint16>(pTable4->end_code + 3*nRanges + 1 + i);
-+ if (lastend >= end || lastend >= start)
-+ return false;
-+ if (offset)
-+ {
-+ const uint16 *gstart = pTable4->end_code + 3*nRanges + 1 + i + (offset >> 1);
-+ const uint16 *gend = gstart + end - start;
-+ if ((char *)gend >= (char *)pCmapSubtable4 + length)
-+ return false;
-+ while (gstart <= gend)
-+ {
-+ uint16 g = be::peek<uint16>(gstart++);
-+ if (g && ((g + delta) & 0xFFFF) > maxgid)
-+ return false;
-+ }
-+ }
-+ else if (((delta + end) & 0xFFFF) > maxgid)
-+ return false;
-+ lastend = end;
-+ }
-+#endif
-+ return true;
- }
-
- /*----------------------------------------------------------------------------------------------
- Return the Glyph ID for the given Unicode ID in the Microsoft Unicode subtable.
- (Actually this code only depends on subtable being format 4.)
- Return 0 if the Unicode ID is not in the subtable.
- ----------------------------------------------------------------------------------------------*/
- gid16 CmapSubtable4Lookup(const void * pCmapSubtabel4, unsigned int nUnicodeId, int rangeKey)
- {
- const Sfnt::CmapSubTableFormat4 * pTable = reinterpret_cast<const Sfnt::CmapSubTableFormat4 *>(pCmapSubtabel4);
-
- uint16 nSeg = be::swap(pTable->seg_count_x2) >> 1;
-
- uint16 n;
-- const uint16 * pLeft, * pMid;
-+ const uint16 * pLeft, * pMid;
- uint16 cMid, chStart, chEnd;
-
- if (rangeKey)
- {
- pMid = &(pTable->end_code[rangeKey]);
- chEnd = be::peek<uint16>(pMid);
- }
- else
-@@ -1027,29 +1075,41 @@ unsigned int CmapSubtable4NextCodepoint(
- if (pRangeKey)
- *pRangeKey = iRange + 1;
- return be::peek<uint16>(pStartCode + iRange + 1);
- }
-
- /*----------------------------------------------------------------------------------------------
- Check the Microsoft UCS-4 subtable for expected values.
- ----------------------------------------------------------------------------------------------*/
--bool CheckCmapSubtable12(const void *pCmapSubtable12)
-+bool CheckCmapSubtable12(const void *pCmapSubtable12, size_t table_len /*, unsigned int maxgid*/)
- {
- if (!pCmapSubtable12) return false;
- const Sfnt::CmapSubTable * pTable = reinterpret_cast<const Sfnt::CmapSubTable *>(pCmapSubtable12);
- if (be::swap(pTable->format) != 12)
- return false;
- const Sfnt::CmapSubTableFormat12 * pTable12 = reinterpret_cast<const Sfnt::CmapSubTableFormat12 *>(pCmapSubtable12);
- uint32 length = be::swap(pTable12->length);
-+ if (length > table_len)
-+ return false;
- if (length < sizeof(Sfnt::CmapSubTableFormat12))
- return false;
--
-- return (length == (sizeof(Sfnt::CmapSubTableFormat12) + (be::swap(pTable12->num_groups) - 1)
-- * sizeof(uint32) * 3));
-+ uint32 num_groups = be::swap(pTable12->num_groups);
-+ if (length != (sizeof(Sfnt::CmapSubTableFormat12) + (num_groups - 1) * sizeof(uint32) * 3))
-+ return false;
-+#if 0
-+ for (unsigned int i = 0; i < num_groups; ++i)
-+ {
-+ if (be::swap(pTable12->group[i].end_char_code) - be::swap(pTable12->group[i].start_char_code) + be::swap(pTable12->group[i].start_glyph_id) > maxgid)
-+ return false;
-+ if (i > 0 && be::swap(pTable12->group[i].start_char_code) <= be::swap(pTable12->group[i-1].end_char_code))
-+ return false;
-+ }
-+#endif
-+ return true;
- }
-
- /*----------------------------------------------------------------------------------------------
- Return the Glyph ID for the given Unicode ID in the Microsoft UCS-4 subtable.
- (Actually this code only depends on subtable being format 12.)
- Return 0 if the Unicode ID is not in the subtable.
- ----------------------------------------------------------------------------------------------*/
- gid16 CmapSubtable12Lookup(const void * pCmap310, unsigned int uUnicodeId, int rangeKey)
-@@ -1140,49 +1200,53 @@ unsigned int CmapSubtable12NextCodepoint
- Technically this method should return an unsigned long but it is unlikely the offset will
- exceed 2^31.
- ----------------------------------------------------------------------------------------------*/
- size_t LocaLookup(gid16 nGlyphId,
- const void * pLoca, size_t lLocaSize,
- const void * pHead) // throw (std::out_of_range)
- {
- const Sfnt::FontHeader * pTable = reinterpret_cast<const Sfnt::FontHeader *>(pHead);
-+ size_t res = -2;
-
- // CheckTable verifies the index_to_loc_format is valid
- if (be::swap(pTable->index_to_loc_format) == Sfnt::FontHeader::ShortIndexLocFormat)
- { // loca entries are two bytes and have been divided by two
-- if (nGlyphId < (lLocaSize >> 1) - 1) // allow sentinel value to be accessed
-+ if (lLocaSize > 1 && nGlyphId + 1u < lLocaSize >> 1) // allow sentinel value to be accessed
- {
- const uint16 * pShortTable = reinterpret_cast<const uint16 *>(pLoca);
-- return (be::peek<uint16>(pShortTable + nGlyphId) << 1);
-+ res = be::peek<uint16>(pShortTable + nGlyphId) << 1;
-+ if (res == static_cast<size_t>(be::peek<uint16>(pShortTable + nGlyphId + 1) << 1))
-+ return -1;
- }
- }
--
-- if (be::swap(pTable->index_to_loc_format) == Sfnt::FontHeader::LongIndexLocFormat)
-+ else if (be::swap(pTable->index_to_loc_format) == Sfnt::FontHeader::LongIndexLocFormat)
- { // loca entries are four bytes
-- if (nGlyphId < (lLocaSize >> 2) - 1)
-+ if (lLocaSize > 3 && nGlyphId + 1u < lLocaSize >> 2)
- {
- const uint32 * pLongTable = reinterpret_cast<const uint32 *>(pLoca);
-- return be::peek<uint32>(pLongTable + nGlyphId);
-+ res = be::peek<uint32>(pLongTable + nGlyphId);
-+ if (res == static_cast<size_t>(be::peek<uint32>(pLongTable + nGlyphId + 1)))
-+ return -1;
- }
- }
-
- // only get here if glyph id was bad
-- return -1;
-+ return res;
- //throw std::out_of_range("glyph id out of range for font");
- }
-
- /*----------------------------------------------------------------------------------------------
- Return a pointer into the glyf table based on the given offset (from LocaLookup).
- Return NULL on error.
- ----------------------------------------------------------------------------------------------*/
- void * GlyfLookup(const void * pGlyf, size_t nGlyfOffset, size_t nTableLen)
- {
- const uint8 * pByte = reinterpret_cast<const uint8 *>(pGlyf);
-- if (nGlyfOffset == size_t(-1) || nGlyfOffset >= nTableLen)
-+ if (nGlyfOffset + pByte < pByte || nGlyfOffset + sizeof(Sfnt::Glyph) >= nTableLen)
- return NULL;
- return const_cast<uint8 *>(pByte + nGlyfOffset);
- }
-
- /*----------------------------------------------------------------------------------------------
- Get the bounding box coordinates for a simple glyf entry (non-composite).
- Return true if successful, false otherwise.
- ----------------------------------------------------------------------------------------------*/
-@@ -1784,17 +1848,16 @@ bool GlyfContourEndPoints(gid16 nGlyphId
- cnPoints - count of points from largest end point obtained from GlyfContourEndPoints
- prgnX & prgnY - should point to buffers large enough to hold cnPoints integers
- The ranges are parallel so that coordinates for point(n) are found at offset n in
- both ranges. These points are in absolute coordinates.
- prgfOnCurve - should point to a buffer a large enough to hold cnPoints bytes (bool)
- This range is parallel to the prgnX & prgnY
- Return true if successful, false otherwise. On false, all points may be INT_MIN
- False may indicate a white space glyph, a multi-level composite, or a corrupt font
-- // TODO: doesn't support composite glyphs whose components are themselves components
- It's not clear from the TTF spec when the transforms should be applied. Should the
- transform be done before or after attachment point calcs? (current code - before)
- Should the transform be applied to other offsets? (currently - no; however commented
- out code is in place so that if CompoundGlyph::UnscaledOffset on the MS rasterizer is
- clear (typical) then yes, and if CompoundGlyph::ScaledOffset on the Apple rasterizer is
- clear (typical?) then no). See GetComponentTransform.
- It's also unclear where point numbering with attachment poinst starts
- (currently - first point number is relative to whole glyph, second point number is
-diff --git a/gfx/graphite2/src/call_machine.cpp b/gfx/graphite2/src/call_machine.cpp
---- a/gfx/graphite2/src/call_machine.cpp
-+++ b/gfx/graphite2/src/call_machine.cpp
-@@ -65,57 +65,60 @@ using namespace graphite2;
- using namespace vm;
-
- struct regbank {
- slotref is;
- slotref * map;
- SlotMap & smap;
- slotref * const map_base;
- const instr * & ip;
-+ uint8 direction;
- int8 flags;
- };
-
- typedef bool (* ip_t)(registers);
-
- // Pull in the opcode definitions
- // We pull these into a private namespace so these otherwise common names dont
- // pollute the toplevel namespace.
- namespace {
- #define smap reg.smap
- #define seg smap.segment
- #define is reg.is
- #define ip reg.ip
- #define map reg.map
- #define mapb reg.map_base
- #define flags reg.flags
-+#define dir reg.direction
-
- #include "inc/opcodes.h"
-
- #undef smap
- #undef seg
- #undef is
- #undef ip
- #undef map
- #undef mapb
- #undef flags
-+#undef dir
- }
-
- Machine::stack_t Machine::run(const instr * program,
- const byte * data,
- slotref * & map)
-
- {
- assert(program != 0);
-
- // Declare virtual machine registers
- const instr * ip = program-1;
- const byte * dp = data;
- stack_t * sp = _stack + Machine::STACK_GUARD,
- * const sb = sp;
-- regbank reg = {*map, map, _map, _map.begin()+_map.context(), ip, 0};
-+ regbank reg = {*map, map, _map, _map.begin()+_map.context(), ip, _map.dir(), 0};
-
- // Run the program
- while ((reinterpret_cast<ip_t>(*++ip))(dp, sp, sb, reg)) {}
- const stack_t ret = sp == _stack+STACK_GUARD+1 ? *sp-- : 0;
-
- check_final_stack(sp);
- map = reg.map;
- *map = reg.is;
-diff --git a/gfx/graphite2/src/direct_machine.cpp b/gfx/graphite2/src/direct_machine.cpp
---- a/gfx/graphite2/src/direct_machine.cpp
-+++ b/gfx/graphite2/src/direct_machine.cpp
-@@ -56,16 +56,17 @@ using namespace vm;
-
- namespace {
-
- const void * direct_run(const bool get_table_mode,
- const instr * program,
- const byte * data,
- Machine::stack_t * stack,
- slotref * & __map,
-+ uint8 _dir,
- SlotMap * __smap=0)
- {
- // We need to define and return to opcode table from within this function
- // other inorder to take the addresses of the instruction bodies.
- #include "inc/opcode_table.h"
- if (get_table_mode)
- return opcode_table;
-
-@@ -74,16 +75,17 @@ const void * direct_run(const bool
- const byte * dp = data;
- Machine::stack_t * sp = stack + Machine::STACK_GUARD,
- * const sb = sp;
- SlotMap & smap = *__smap;
- Segment & seg = smap.segment;
- slotref is = *__map,
- * map = __map,
- * const mapb = smap.begin()+smap.context();
-+ uint8 dir = _dir;
- int8 flags = 0;
-
- // start the program
- goto **ip;
-
- // Pull in the opcode definitions
- #include "inc/opcodes.h"
-
-@@ -104,14 +106,14 @@ const opcode_t * Machine::getOpcodeTable
-
- Machine::stack_t Machine::run(const instr * program,
- const byte * data,
- slotref * & is)
- {
- assert(program != 0);
-
- const stack_t *sp = static_cast<const stack_t *>(
-- direct_run(false, program, data, _stack, is, &_map));
-+ direct_run(false, program, data, _stack, is, _map.dir(), &_map));
- const stack_t ret = sp == _stack+STACK_GUARD+1 ? *sp-- : 0;
- check_final_stack(sp);
- return ret;
- }
-
-diff --git a/gfx/graphite2/src/files.mk b/gfx/graphite2/src/files.mk
---- a/gfx/graphite2/src/files.mk
-+++ b/gfx/graphite2/src/files.mk
-@@ -42,29 +42,32 @@
- $($(_NS)_BASE)/src/gr_char_info.cpp \
- $($(_NS)_BASE)/src/gr_face.cpp \
- $($(_NS)_BASE)/src/gr_features.cpp \
- $($(_NS)_BASE)/src/gr_font.cpp \
- $($(_NS)_BASE)/src/gr_logging.cpp \
- $($(_NS)_BASE)/src/gr_segment.cpp \
- $($(_NS)_BASE)/src/gr_slot.cpp \
- $($(_NS)_BASE)/src/json.cpp \
-- $($(_NS)_BASE)/src/Bidi.cpp \
- $($(_NS)_BASE)/src/CachedFace.cpp \
- $($(_NS)_BASE)/src/CmapCache.cpp \
- $($(_NS)_BASE)/src/Code.cpp \
-+ $($(_NS)_BASE)/src/Collider.cpp \
-+ $($(_NS)_BASE)/src/Decompressor.cpp \
- $($(_NS)_BASE)/src/Face.cpp \
- $($(_NS)_BASE)/src/FeatureMap.cpp \
- $($(_NS)_BASE)/src/FileFace.cpp \
- $($(_NS)_BASE)/src/Font.cpp \
- $($(_NS)_BASE)/src/GlyphCache.cpp \
- $($(_NS)_BASE)/src/GlyphFace.cpp \
-+ $($(_NS)_BASE)/src/Intervals.cpp \
- $($(_NS)_BASE)/src/Justifier.cpp \
- $($(_NS)_BASE)/src/NameTable.cpp \
- $($(_NS)_BASE)/src/Pass.cpp \
-+ $($(_NS)_BASE)/src/Position.cpp \
- $($(_NS)_BASE)/src/SegCache.cpp \
- $($(_NS)_BASE)/src/SegCacheEntry.cpp \
- $($(_NS)_BASE)/src/SegCacheStore.cpp \
- $($(_NS)_BASE)/src/Segment.cpp \
- $($(_NS)_BASE)/src/Silf.cpp \
- $($(_NS)_BASE)/src/Slot.cpp \
- $($(_NS)_BASE)/src/Sparse.cpp \
- $($(_NS)_BASE)/src/TtfUtil.cpp \
-@@ -73,25 +76,29 @@
- $(_NS)_PRIVATE_HEADERS = \
- $($(_NS)_BASE)/src/inc/bits.h \
- $($(_NS)_BASE)/src/inc/debug.h \
- $($(_NS)_BASE)/src/inc/json.h \
- $($(_NS)_BASE)/src/inc/CachedFace.h \
- $($(_NS)_BASE)/src/inc/CharInfo.h \
- $($(_NS)_BASE)/src/inc/CmapCache.h \
- $($(_NS)_BASE)/src/inc/Code.h \
-+ $($(_NS)_BASE)/src/inc/Collider.h \
-+ $($(_NS)_BASE)/src/inc/Compression.h \
-+ $($(_NS)_BASE)/src/inc/Decompressor.h \
- $($(_NS)_BASE)/src/inc/Endian.h \
- $($(_NS)_BASE)/src/inc/Error.h \
- $($(_NS)_BASE)/src/inc/Face.h \
- $($(_NS)_BASE)/src/inc/FeatureMap.h \
- $($(_NS)_BASE)/src/inc/FeatureVal.h \
- $($(_NS)_BASE)/src/inc/FileFace.h \
- $($(_NS)_BASE)/src/inc/Font.h \
- $($(_NS)_BASE)/src/inc/GlyphCache.h \
- $($(_NS)_BASE)/src/inc/GlyphFace.h \
-+ $($(_NS)_BASE)/src/inc/Intervals.h \
- $($(_NS)_BASE)/src/inc/List.h \
- $($(_NS)_BASE)/src/inc/locale2lcid.h \
- $($(_NS)_BASE)/src/inc/Machine.h \
- $($(_NS)_BASE)/src/inc/Main.h \
- $($(_NS)_BASE)/src/inc/NameTable.h \
- $($(_NS)_BASE)/src/inc/opcode_table.h \
- $($(_NS)_BASE)/src/inc/opcodes.h \
- $($(_NS)_BASE)/src/inc/Pass.h \
-diff --git a/gfx/graphite2/src/gr_face.cpp b/gfx/graphite2/src/gr_face.cpp
---- a/gfx/graphite2/src/gr_face.cpp
-+++ b/gfx/graphite2/src/gr_face.cpp
-@@ -41,17 +41,17 @@ extern json *global_log;
-
- namespace
- {
- bool load_face(Face & face, unsigned int options)
- {
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::category _misc_cat(face.tele.misc);
- #endif
-- Face::Table silf(face, Tag::Silf);
-+ Face::Table silf(face, Tag::Silf, 0x00050000);
- if (silf) options &= ~gr_face_dumbRendering;
- else if (!(options & gr_face_dumbRendering))
- return false;
-
- if (!face.readGlyphs(options))
- return false;
-
- if (silf)
-diff --git a/gfx/graphite2/src/gr_logging.cpp b/gfx/graphite2/src/gr_logging.cpp
---- a/gfx/graphite2/src/gr_logging.cpp
-+++ b/gfx/graphite2/src/gr_logging.cpp
-@@ -19,24 +19,25 @@
- Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
- internet at http://www.fsf.org/licenses/lgpl.html.
-
- Alternatively, the contents of this file may be used under the terms of the
- Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
--#include <stdio.h>
-+#include <cstdio>
-
- #include "graphite2/Log.h"
- #include "inc/debug.h"
- #include "inc/CharInfo.h"
- #include "inc/Slot.h"
- #include "inc/Segment.h"
- #include "inc/json.h"
-+#include "inc/Collider.h"
-
- #if defined _WIN32
- #include "windows.h"
- #endif
-
- using namespace graphite2;
-
- #if !defined GRAPHITE2_NTRACING
-@@ -179,16 +180,17 @@ json & graphite2::operator << (json & j,
-
-
- json & graphite2::operator << (json & j, const dslot & ds) throw()
- {
- assert(ds.first);
- assert(ds.second);
- const Segment & seg = *ds.first;
- const Slot & s = *ds.second;
-+ const SlotCollision *cslot = seg.collisionInfo(ds.second);
-
- j << json::object
- << "id" << objectid(ds)
- << "gid" << s.gid()
- << "charinfo" << json::flat << json::object
- << "original" << s.original()
- << "before" << s.before()
- << "after" << s.after()
-@@ -215,16 +217,38 @@ json & graphite2::operator << (json & j,
- j << json::close;
- if (s.firstChild())
- {
- j << "children" << json::flat << json::array;
- for (const Slot *c = s.firstChild(); c; c = c->nextSibling())
- j << objectid(dslot(&seg, c));
- j << json::close;
- }
-+ if (cslot)
-+ {
-+ // Note: the reason for using Positions to lump together related attributes is to make the
-+ // JSON output slightly more compact.
-+ j << "collision" << json::flat << json::object
-+// << "shift" << cslot->shift() -- not used pass level, only within the collision routine itself
-+ << "offset" << cslot->offset()
-+ << "limit" << cslot->limit()
-+ << "flags" << cslot->flags()
-+ << "margin" << Position(cslot->margin(), cslot->marginWt())
-+ << "exclude" << cslot->exclGlyph()
-+ << "excludeoffset" << cslot->exclOffset();
-+ if (cslot->seqOrder() != 0)
-+ {
-+ j << "seqclass" << Position(cslot->seqClass(), cslot->seqProxClass())
-+ << "seqorder" << cslot->seqOrder()
-+ << "seqabove" << Position(cslot->seqAboveXoff(), cslot->seqAboveWt())
-+ << "seqbelow" << Position(cslot->seqBelowXlim(), cslot->seqBelowWt())
-+ << "seqvalign" << Position(cslot->seqValignHt(), cslot->seqValignWt());
-+ }
-+ j << json::close;
-+ }
- return j << json::close;
- }
-
-
- graphite2::objectid::objectid(const dslot & ds) throw()
- {
- const Slot * const p = ds.second;
- uint32 s = reinterpret_cast<size_t>(p);
-diff --git a/gfx/graphite2/src/gr_segment.cpp b/gfx/graphite2/src/gr_segment.cpp
---- a/gfx/graphite2/src/gr_segment.cpp
-+++ b/gfx/graphite2/src/gr_segment.cpp
-@@ -43,21 +43,17 @@ namespace
- Segment* pRes=new Segment(nChars, face, script, dir);
-
-
- if (!pRes->read_text(face, pFeats, enc, pStart, nChars) || !pRes->runGraphite())
- {
- delete pRes;
- return NULL;
- }
-- // run the line break passes
-- // run the substitution passes
-- pRes->prepare_pos(font);
-- // run the positioning passes
-- pRes->finalise(font);
-+ pRes->finalise(font, true);
-
- return static_cast<gr_segment*>(pRes);
- }
-
-
- }
-
-
-diff --git a/gfx/graphite2/src/gr_slot.cpp b/gfx/graphite2/src/gr_slot.cpp
---- a/gfx/graphite2/src/gr_slot.cpp
-+++ b/gfx/graphite2/src/gr_slot.cpp
-@@ -98,21 +98,21 @@ float gr_slot_advance_X(const gr_slot* p
- if (face && font->isHinted())
- res = (res - face->glyphs().glyph(p->gid())->theAdvance().x) * scale + font->advance(p->gid());
- else
- res = res * scale;
- }
- return res;
- }
-
--float gr_slot_advance_Y(const gr_slot *p/*not NULL*/, const gr_face *face, const gr_font *font)
-+float gr_slot_advance_Y(const gr_slot *p/*not NULL*/, GR_MAYBE_UNUSED const gr_face *face, const gr_font *font)
- {
- assert(p);
- float res = p->advancePos().y;
-- if (font && (face || !face))
-+ if (font)
- return res * font->scale();
- else
- return res;
- }
-
- int gr_slot_before(const gr_slot* p/*not NULL*/)
- {
- assert(p);
-diff --git a/gfx/graphite2/src/inc/Code.h b/gfx/graphite2/src/inc/Code.h
---- a/gfx/graphite2/src/inc/Code.h
-+++ b/gfx/graphite2/src/inc/Code.h
-@@ -36,32 +36,41 @@ of the License or (at your option) any l
- #include "inc/Main.h"
- #include "inc/Machine.h"
-
- namespace graphite2 {
-
- class Silf;
- class Face;
-
-+enum passtype {
-+ PASS_TYPE_UNKNOWN = 0,
-+ PASS_TYPE_LINEBREAK,
-+ PASS_TYPE_SUBSTITUTE,
-+ PASS_TYPE_POSITIONING,
-+ PASS_TYPE_JUSTIFICATION
-+};
-+
- namespace vm {
-
- class Machine::Code
- {
- public:
- enum status_t
- {
- loaded,
- alloc_failed,
- invalid_opcode,
- unimplemented_opcode_used,
- out_of_range_data,
- jump_past_end,
- arguments_exhausted,
- missing_return,
-- nested_context_item
-+ nested_context_item,
-+ underfull_stack
- };
-
- private:
- class decoder;
-
- instr * _code;
- byte * _data;
- size_t _data_size,
-@@ -72,40 +81,51 @@ private:
- _modify,
- _delete;
- mutable bool _own;
-
- void release_buffers() throw ();
- void failure(const status_t) throw();
-
- public:
-+ static size_t estimateCodeDataOut(size_t num_bytecodes);
-+
- Code() throw();
- Code(bool is_constraint, const byte * bytecode_begin, const byte * const bytecode_end,
-- uint8 pre_context, uint16 rule_length, const Silf &, const Face &);
-+ uint8 pre_context, uint16 rule_length, const Silf &, const Face &,
-+ enum passtype pt, byte * * const _out = 0);
- Code(const Machine::Code &) throw();
- ~Code() throw();
-
- Code & operator=(const Code &rhs) throw();
-- operator bool () const throw();
-- status_t status() const throw();
-- bool constraint() const throw();
-- size_t dataSize() const throw();
-- size_t instructionCount() const throw();
-- bool immutable() const throw();
-- bool deletes() const throw();
-- size_t maxRef() const throw();
-+ operator bool () const throw() { return _code && status() == loaded; }
-+ status_t status() const throw() { return _status; }
-+ bool constraint() const throw() { return _constraint; }
-+ size_t dataSize() const throw() { return _data_size; }
-+ size_t instructionCount() const throw() { return _instr_count; }
-+ bool immutable() const throw() { return !(_delete || _modify); }
-+ bool deletes() const throw() { return _delete; }
-+ size_t maxRef() const throw() { return _max_ref; }
-+ void externalProgramMoved(ptrdiff_t) throw();
-
- int32 run(Machine &m, slotref * & map) const;
-
- CLASS_NEW_DELETE;
- };
-
-+inline
-+size_t Machine::Code::estimateCodeDataOut(size_t n_bc)
-+{
-+ return n_bc * (sizeof(instr)+sizeof(byte));
-+}
-+
-+
- inline Machine::Code::Code() throw()
- : _code(0), _data(0), _data_size(0), _instr_count(0), _max_ref(0),
-- _status(loaded), _constraint(false), _modify(false),_delete(false),
-+ _status(loaded), _constraint(false), _modify(false), _delete(false),
- _own(false)
- {
- }
-
- inline Machine::Code::Code(const Machine::Code &obj) throw ()
- : _code(obj._code),
- _data(obj._data),
- _data_size(obj._data_size),
-@@ -131,45 +151,19 @@ inline Machine::Code & Machine::Code::op
- _constraint = rhs._constraint;
- _modify = rhs._modify;
- _delete = rhs._delete;
- _own = rhs._own;
- rhs._own = false;
- return *this;
- }
-
--inline Machine::Code::operator bool () const throw () {
-- return _code && status() == loaded;
--}
--
--inline Machine::Code::status_t Machine::Code::status() const throw() {
-- return _status;
--}
--
--inline bool Machine::Code::constraint() const throw() {
-- return _constraint;
--}
--
--inline size_t Machine::Code::dataSize() const throw() {
-- return _data_size;
--}
--
--inline size_t Machine::Code::instructionCount() const throw() {
-- return _instr_count;
--}
--
--inline bool Machine::Code::immutable() const throw()
-+inline void Machine::Code::externalProgramMoved(ptrdiff_t dist) throw()
- {
-- return !(_delete || _modify);
--}
--
--inline bool Machine::Code::deletes() const throw()
--{
-- return _delete;
--}
--
--inline size_t Machine::Code::maxRef() const throw()
--{
-- return _max_ref;
-+ if (_code && !_own)
-+ {
-+ _code += dist / sizeof(instr);
-+ _data += dist;
-+ }
- }
-
- } // namespace vm
- } // namespace graphite2
-diff --git a/gfx/graphite2/src/inc/Collider.h b/gfx/graphite2/src/inc/Collider.h
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/inc/Collider.h
-@@ -0,0 +1,242 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#pragma once
-+
-+#include "inc/List.h"
-+#include "inc/Position.h"
-+#include "inc/Intervals.h"
-+#include "inc/debug.h"
-+
-+namespace graphite2 {
-+
-+class json;
-+class Slot;
-+class Segment;
-+
-+#define SLOTCOLSETUINTPROP(x, y) uint16 x() const { return _ ##x; } void y (uint16 v) { _ ##x = v; }
-+#define SLOTCOLSETINTPROP(x, y) int16 x() const { return _ ##x; } void y (int16 v) { _ ##x = v; }
-+#define SLOTCOLSETPOSITIONPROP(x, y) const Position &x() const { return _ ##x; } void y (const Position &v) { _ ##x = v; }
-+
-+// Slot attributes related to collision-fixing
-+class SlotCollision
-+{
-+public:
-+ enum {
-+ // COLL_TESTONLY = 0, // default - test other glyphs for collision with this one, but don't move this one
-+ COLL_FIX = 1, // fix collisions involving this glyph
-+ COLL_IGNORE = 2, // ignore this glyph altogether
-+ COLL_START = 4, // start of range of possible collisions
-+ COLL_END = 8, // end of range of possible collisions
-+ COLL_KERN = 16, // collisions with this glyph are fixed by adding kerning space after it
-+ COLL_ISCOL = 32, // this glyph has a collision
-+ COLL_KNOWN = 64, // we've figured out what's happening with this glyph
-+ COLL_TEMPLOCK = 128, // Lock glyphs that have been given priority positioning
-+ ////COLL_JUMPABLE = 128, // moving glyphs may jump this stationary glyph in any direction - DELETE
-+ ////COLL_OVERLAP = 256, // use maxoverlap to restrict - DELETE
-+ };
-+
-+ // Behavior for the collision.order attribute. To GDL this is an enum, to us it's a bitfield, with only 1 bit set
-+ // Allows for easier inversion.
-+ enum {
-+ SEQ_ORDER_LEFTDOWN = 1,
-+ SEQ_ORDER_RIGHTUP = 2,
-+ SEQ_ORDER_NOABOVE = 4,
-+ SEQ_ORDER_NOBELOW = 8,
-+ SEQ_ORDER_NOLEFT = 16,
-+ SEQ_ORDER_NORIGHT = 32
-+ };
-+
-+ SlotCollision(Segment *seg, Slot *slot);
-+ void initFromSlot(Segment *seg, Slot *slot);
-+
-+ const Rect &limit() const { return _limit; }
-+ void setLimit(const Rect &r) { _limit = r; }
-+ SLOTCOLSETPOSITIONPROP(shift, setShift)
-+ SLOTCOLSETPOSITIONPROP(offset, setOffset)
-+ SLOTCOLSETPOSITIONPROP(exclOffset, setExclOffset)
-+ SLOTCOLSETUINTPROP(margin, setMargin)
-+ SLOTCOLSETUINTPROP(marginWt, setMarginWt)
-+ SLOTCOLSETUINTPROP(flags, setFlags)
-+ SLOTCOLSETUINTPROP(exclGlyph, setExclGlyph)
-+ SLOTCOLSETUINTPROP(seqClass, setSeqClass)
-+ SLOTCOLSETUINTPROP(seqProxClass, setSeqProxClass)
-+ SLOTCOLSETUINTPROP(seqOrder, setSeqOrder)
-+ SLOTCOLSETINTPROP(seqAboveXoff, setSeqAboveXoff)
-+ SLOTCOLSETUINTPROP(seqAboveWt, setSeqAboveWt)
-+ SLOTCOLSETINTPROP(seqBelowXlim, setSeqBelowXlim)
-+ SLOTCOLSETUINTPROP(seqBelowWt, setSeqBelowWt)
-+ SLOTCOLSETUINTPROP(seqValignHt, setSeqValignHt)
-+ SLOTCOLSETUINTPROP(seqValignWt, setSeqValignWt)
-+
-+ float getKern(int dir) const;
-+
-+private:
-+ Rect _limit;
-+ Position _shift; // adjustment within the given pass
-+ Position _offset; // total adjustment for collisions
-+ Position _exclOffset;
-+ uint16 _margin;
-+ uint16 _marginWt;
-+ uint16 _flags;
-+ uint16 _exclGlyph;
-+ uint16 _seqClass;
-+ uint16 _seqProxClass;
-+ uint16 _seqOrder;
-+ int16 _seqAboveXoff;
-+ uint16 _seqAboveWt;
-+ int16 _seqBelowXlim;
-+ uint16 _seqBelowWt;
-+ uint16 _seqValignHt;
-+ uint16 _seqValignWt;
-+
-+}; // end of class SlotColllision
-+
-+struct BBox;
-+struct SlantBox;
-+
-+class ShiftCollider
-+{
-+public:
-+ typedef std::pair<float, float> fpair;
-+ typedef Vector<fpair> vfpairs;
-+ typedef vfpairs::iterator ivfpairs;
-+
-+ ShiftCollider(json *dbgout);
-+ ~ShiftCollider() throw() { };
-+
-+ bool initSlot(Segment *seg, Slot *aSlot, const Rect &constraint,
-+ float margin, float marginMin, const Position &currShift,
-+ const Position &currOffset, int dir, GR_MAYBE_UNUSED json * const dbgout);
-+ bool mergeSlot(Segment *seg, Slot *slot, const Position &currShift, bool isAfter,
-+ bool sameCluster, bool &hasCol, bool isExclusion, GR_MAYBE_UNUSED json * const dbgout);
-+ Position resolve(Segment *seg, bool &isCol, GR_MAYBE_UNUSED json * const dbgout);
-+ void addBox_slope(bool isx, const Rect &box, const BBox &bb, const SlantBox &sb, const Position &org, float weight, float m, bool minright, int mode);
-+ void removeBox(const Rect &box, const BBox &bb, const SlantBox &sb, const Position &org, int mode);
-+ const Position &origin() const { return _origin; }
-+
-+#if !defined GRAPHITE2_NTRACING
-+ void outputJsonDbg(json * const dbgout, Segment *seg, int axis);
-+ void outputJsonDbgStartSlot(json * const dbgout, Segment *seg);
-+ void outputJsonDbgEndSlot(json * const dbgout, Position resultPos, int bestAxis, bool isCol);
-+ void outputJsonDbgOneVector(json * const dbgout, Segment *seg, int axis, float tleft, float bestCost, float bestVal);
-+ void outputJsonDbgRawRanges(json * const dbgout, int axis);
-+ void outputJsonDbgRemovals(json * const dbgout, int axis, Segment *seg);
-+#endif
-+
-+ CLASS_NEW_DELETE;
-+
-+protected:
-+ Zones _ranges[4]; // possible movements in 4 directions (horizontally, vertically, diagonally);
-+ Slot * _target; // the glyph to fix
-+ Rect _limit;
-+ Position _currShift;
-+ Position _currOffset;
-+ Position _origin; // Base for all relative calculations
-+ float _margin;
-+ float _marginWt;
-+ float _len[4];
-+ uint16 _seqClass;
-+ uint16 _seqProxClass;
-+ uint16 _seqOrder;
-+
-+ //bool _scraping[4];
-+
-+}; // end of class ShiftCollider
-+
-+inline
-+ShiftCollider::ShiftCollider(GR_MAYBE_UNUSED json *dbgout)
-+: _target(0),
-+ _margin(0.0),
-+ _marginWt(0.0),
-+ _seqClass(0),
-+ _seqProxClass(0),
-+ _seqOrder(0)
-+{
-+#if !defined GRAPHITE2_NTRACING
-+ for (int i = 0; i < 4; ++i)
-+ _ranges[i].setdebug(dbgout);
-+#endif
-+}
-+
-+class KernCollider
-+{
-+public:
-+ KernCollider(json *dbg);
-+ ~KernCollider() throw() { };
-+ bool initSlot(Segment *seg, Slot *aSlot, const Rect &constraint, float margin,
-+ const Position &currShift, const Position &offsetPrev, int dir,
-+ float ymin, float ymax, json * const dbgout);
-+ bool mergeSlot(Segment *seg, Slot *slot, const Position &currShift, float currSpace, int dir, json * const dbgout);
-+ Position resolve(Segment *seg, Slot *slot, int dir, float margin, json * const dbgout);
-+ void shift(const Position &mv, int dir);
-+
-+ CLASS_NEW_DELETE;
-+
-+private:
-+ Slot * _target; // the glyph to fix
-+ Rect _limit;
-+ float _margin;
-+ Position _offsetPrev; // kern from a previous pass
-+ Position _currShift; // NOT USED??
-+ float _miny; // y-coordinates offset by global slot position
-+ float _maxy;
-+ Vector<float> _edges; // edges of horizontal slices
-+ float _sliceWidth; // width of each slice
-+ float _mingap;
-+ float _xbound; // max or min edge
-+
-+#if !defined GRAPHITE2_NTRACING
-+ // Debugging
-+ Segment * _seg;
-+ Vector<float> _nearEdges; // closest potential collision in each slice
-+ Vector<Slot*> _slotNear;
-+#endif
-+}; // end of class KernCollider
-+
-+
-+inline
-+float sqr(float x) {
-+ return x * x;
-+}
-+
-+inline
-+KernCollider::KernCollider(GR_MAYBE_UNUSED json *dbg)
-+: _target(0),
-+ _margin(0.0f),
-+ _miny(-1e38f),
-+ _maxy(1e38f),
-+ _sliceWidth(0.0f),
-+ _mingap(0.0f),
-+ _xbound(0.0)
-+{
-+#if !defined GRAPHITE2_NTRACING
-+ _seg = 0;
-+#endif
-+};
-+
-+}; // end of namespace graphite2
-+
-diff --git a/gfx/graphite2/src/inc/Compression.h b/gfx/graphite2/src/inc/Compression.h
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/inc/Compression.h
-@@ -0,0 +1,103 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2015, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+
-+#pragma once
-+
-+#include <cassert>
-+#include <cstddef>
-+#include <cstring>
-+
-+namespace
-+{
-+
-+#if defined(_MSC_VER)
-+typedef unsigned __int8 u8;
-+typedef unsigned __int16 u16;
-+typedef unsigned __int32 u32;
-+typedef unsigned __int64 u64;
-+#else
-+#include <stdint.h>
-+typedef uint8_t u8;
-+typedef uint16_t u16;
-+typedef uint32_t u32;
-+typedef uint64_t u64;
-+#endif
-+
-+ptrdiff_t const MINMATCH = 4;
-+
-+template<int S>
-+inline
-+void unaligned_copy(void * d, void const * s) {
-+ ::memcpy(d, s, S);
-+}
-+
-+inline
-+size_t align(size_t p) {
-+ return (p + sizeof(unsigned long)-1) & ~(sizeof(unsigned long)-1);
-+}
-+
-+inline
-+u8 * safe_copy(u8 * d, u8 const * s, size_t n) {
-+ while (n--) *d++ = *s++;
-+ return d;
-+}
-+
-+inline
-+u8 * overrun_copy(u8 * d, u8 const * s, size_t n) {
-+ size_t const WS = sizeof(unsigned long);
-+ u8 const * e = s + n;
-+ do
-+ {
-+ unaligned_copy<WS>(d, s);
-+ d += WS;
-+ s += WS;
-+ }
-+ while (s < e);
-+ d-=(s-e);
-+
-+ return d;
-+}
-+
-+
-+inline
-+u8 * fast_copy(u8 * d, u8 const * s, size_t n) {
-+ size_t const WS = sizeof(unsigned long);
-+ size_t wn = n/WS;
-+ while (wn--)
-+ {
-+ unaligned_copy<WS>(d, s);
-+ d += WS;
-+ s += WS;
-+ }
-+ n &= WS-1;
-+ return safe_copy(d, s, n);
-+}
-+
-+
-+} // end of anonymous namespace
-+
-+
-diff --git a/gfx/graphite2/src/inc/Decompressor.h b/gfx/graphite2/src/inc/Decompressor.h
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/inc/Decompressor.h
-@@ -0,0 +1,56 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2015, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+
-+#pragma once
-+
-+#include <cstddef>
-+
-+namespace lz4
-+{
-+
-+// decompress an LZ4 block
-+// Parameters:
-+// @in - Input buffer containing an LZ4 block.
-+// @in_size - Size of the input LZ4 block in bytes.
-+// @out - Output buffer to hold decompressed results.
-+// @out_size - The size of the buffer pointed to by @out.
-+// Invariants:
-+// @in - This buffer must be at least 1 machine word in length,
-+// regardless of the actual LZ4 block size.
-+// @in_size - This must be at least 4 and must also be <= to the
-+// allocated buffer @in.
-+// @out - This must be bigger than the input buffer and at least
-+// 13 bytes.
-+// @out_size - Must always be big enough to hold the expected size.
-+// Return:
-+// -1 - Decompression failed.
-+// size - Actual number of bytes decompressed.
-+int decompress(void const *in, size_t in_size, void *out, size_t out_size);
-+
-+} // end of namespace shrinker
-+
-+
-diff --git a/gfx/graphite2/src/inc/Error.h b/gfx/graphite2/src/inc/Error.h
---- a/gfx/graphite2/src/inc/Error.h
-+++ b/gfx/graphite2/src/inc/Error.h
-@@ -106,22 +106,30 @@ enum errors {
- E_BADRULECCODEPTR = 45, // The rule constraint code position does not align with where the forward reference says it should be
- E_BADCCODELEN = 46, // Bad rule/pass constraint code length
- E_BADACTIONCODEPTR = 47, // The action code position does not align with where the forward reference says it should be
- E_MUTABLECCODE = 48, // Constraint code edits slots. It shouldn't.
- E_BADSTATE = 49, // Bad state transition referencing an illegal state
- E_BADRULEMAPPING = 50, // The structure of the rule mapping is bad
- E_BADRANGE = 51, // Bad column range structure including a glyph in more than one column
- E_BADRULENUM = 52, // A reference to a rule is out of range (too high)
-+ E_BADACOLLISION = 53, // Bad Silf table collision attribute number (too high)
-+ E_BADEMPTYPASS = 54, // Can't have empty passes (no rules) except for collision passes
-+ E_BADSILFVERSION = 55, // The Silf table has a bad version (probably too high)
-+ E_BADCOLLISIONPASS = 56, // Collision flags set on a non positioning pass
-+ E_BADNUMCOLUMNS = 57, // Arbitrarily limit number of columns in fsm
- // Code errors
- E_CODEFAILURE = 60, // Base code error. The following subcodes must align with Machine::Code::status_t in Code.h
-- E_CODEALLOC = 61, // Out of memory
-- E_INVALIDOPCODE = 62, // Invalid op code
-- E_UNIMPOPCODE = 63, // Unimplemented op code encountered
-- E_OUTOFRANGECODE = 64, // Code argument out of range
-- E_BADJUMPCODE = 65, // Code jumps past end of op codes
-- E_CODEBADARGS = 66, // Code arguments exhausted
-- E_CODENORETURN = 67, // Missing return type op code at end of code
-- E_CODENESTEDCTXT = 68 // Nested context encountered in code
-+ E_CODEALLOC = 61, // Out of memory
-+ E_INVALIDOPCODE = 62, // Invalid op code
-+ E_UNIMPOPCODE = 63, // Unimplemented op code encountered
-+ E_OUTOFRANGECODE = 64, // Code argument out of range
-+ E_BADJUMPCODE = 65, // Code jumps past end of op codes
-+ E_CODEBADARGS = 66, // Code arguments exhausted
-+ E_CODENORETURN = 67, // Missing return type op code at end of code
-+ E_CODENESTEDCTXT = 68, // Nested context encountered in code
-+// Compression errors
-+ E_BADSCHEME = 69,
-+ E_SHRINKERFAILED = 70,
- };
-
- }
-
-diff --git a/gfx/graphite2/src/inc/Face.h b/gfx/graphite2/src/inc/Face.h
---- a/gfx/graphite2/src/inc/Face.h
-+++ b/gfx/graphite2/src/inc/Face.h
-@@ -21,33 +21,34 @@
-
- Alternatively, the contents of this file may be used under the terms of the
- Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- #pragma once
-
--#include <stdio.h>
-+#include <cstdio>
-
- #include "graphite2/Font.h"
-
- #include "inc/Main.h"
- #include "inc/FeatureMap.h"
- #include "inc/TtfUtil.h"
- #include "inc/Silf.h"
- #include "inc/Error.h"
-
- namespace graphite2 {
-
- class Cmap;
- class FileFace;
- class GlyphCache;
- class NameTable;
- class json;
-+class Font;
-
-
- using TtfUtil::Tag;
-
- // These are the actual tags, as distinct from the consecutive IDs in TtfUtil.h
-
- class Face
- {
-@@ -165,47 +166,51 @@ json * Face::logger() const throw()
-
-
-
- class Face::Table
- {
- const Face * _f;
- mutable const byte * _p;
- uint32 _sz;
-+ bool _compressed;
-+
-+ Error decompress();
-+
-+ void releaseBuffers();
-
- public:
- Table() throw();
-- Table(const Face & face, const Tag n) throw();
-+ Table(const Face & face, const Tag n, uint32 version=0xffffffff) throw();
- Table(const Table & rhs) throw();
- ~Table() throw();
-
- operator const byte * () const throw();
-
- Table & operator = (const Table & rhs) throw();
- size_t size() const throw();
- };
-
- inline
- Face::Table::Table() throw()
--: _f(0), _p(0), _sz(0)
-+: _f(0), _p(0), _sz(0), _compressed(false)
- {
- }
-
- inline
- Face::Table::Table(const Table & rhs) throw()
--: _f(rhs._f), _p(rhs._p), _sz(rhs._sz)
-+: _f(rhs._f), _p(rhs._p), _sz(rhs._sz), _compressed(rhs._compressed)
- {
- rhs._p = 0;
- }
-
- inline
- Face::Table::~Table() throw()
- {
-- if (_p && _f->m_ops.release_table)
-- (*_f->m_ops.release_table)(_f->m_appFaceHandle, _p);
-+ releaseBuffers();
- }
-
- inline
- Face::Table::operator const byte * () const throw()
- {
- return _p;
- }
-
-diff --git a/gfx/graphite2/src/inc/FeatureMap.h b/gfx/graphite2/src/inc/FeatureMap.h
---- a/gfx/graphite2/src/inc/FeatureMap.h
-+++ b/gfx/graphite2/src/inc/FeatureMap.h
-@@ -51,17 +51,17 @@ private:
- };
-
- class FeatureRef
- {
- typedef uint32 chunk_t;
- static const uint8 SIZEOF_CHUNK = sizeof(chunk_t)*8;
-
- public:
-- FeatureRef() : m_nameValues(0) {}
-+ FeatureRef();
- FeatureRef(const Face & face, unsigned short & bits_offset, uint32 max_val,
- uint32 name, uint16 uiName, uint16 flags,
- FeatureSetting *settings, uint16 num_set) throw();
- ~FeatureRef() throw();
-
- bool applyValToFeature(uint32 val, Features& pDest) const; //defined in GrFaceImp.h
- void maskFeature(Features & pDest) const {
- if (m_index < pDest.size()) //defensive
-@@ -94,16 +94,26 @@ private:
- byte m_bits, // how many bits to shift the value into place
- m_index; // index into the array to find the ulong to mask
-
- private: //unimplemented
- FeatureRef& operator=(const FeatureRef&);
- };
-
-
-+inline
-+FeatureRef::FeatureRef()
-+: m_pFace(0), m_nameValues(0),
-+ m_mask(0), m_max(0), m_id(0),
-+ m_nameid(0), m_flags(0), m_numSet(0),
-+ m_bits(0), m_index(0)
-+{
-+}
-+
-+
- class NameAndFeatureRef
- {
- public:
- NameAndFeatureRef(uint32 name = 0) : m_name(name) , m_pFRef(NULL){}
- NameAndFeatureRef(const FeatureRef* p/*not NULL*/) : m_name(p->getId()), m_pFRef(p) {}
-
- bool operator<(const NameAndFeatureRef& rhs) const //orders by m_name
- { return m_name<rhs.m_name; }
-@@ -112,35 +122,34 @@ class NameAndFeatureRef
-
- uint32 m_name;
- const FeatureRef* m_pFRef;
- };
-
- class FeatureMap
- {
- public:
-- FeatureMap() : m_numFeats(0), m_feats(NULL), m_pNamedFeats(NULL),
-- m_defaultFeatures(NULL) {}
-- ~FeatureMap() { delete [] m_feats; delete[] m_pNamedFeats; delete m_defaultFeatures; }
-+ FeatureMap() : m_numFeats(0), m_feats(NULL), m_pNamedFeats(NULL) {}
-+ ~FeatureMap() { delete [] m_feats; delete[] m_pNamedFeats; }
-
- bool readFeats(const Face & face);
- const FeatureRef *findFeatureRef(uint32 name) const;
- FeatureRef *feature(uint16 index) const { return m_feats + index; }
- //GrFeatureRef *featureRef(byte index) { return index < m_numFeats ? m_feats + index : NULL; }
- const FeatureRef *featureRef(byte index) const { return index < m_numFeats ? m_feats + index : NULL; }
- FeatureVal* cloneFeatures(uint32 langname/*0 means default*/) const; //call destroy_Features when done.
- uint16 numFeats() const { return m_numFeats; };
- CLASS_NEW_DELETE
- private:
- friend class SillMap;
- uint16 m_numFeats;
-
- FeatureRef *m_feats;
- NameAndFeatureRef* m_pNamedFeats; //owned
-- FeatureVal* m_defaultFeatures; //owned
-+ FeatureVal m_defaultFeatures; //owned
-
- private: //defensive on m_feats, m_pNamedFeats, and m_defaultFeatures
- FeatureMap(const FeatureMap&);
- FeatureMap& operator=(const FeatureMap&);
- };
-
-
- class SillMap
-diff --git a/gfx/graphite2/src/inc/FileFace.h b/gfx/graphite2/src/inc/FileFace.h
---- a/gfx/graphite2/src/inc/FileFace.h
-+++ b/gfx/graphite2/src/inc/FileFace.h
-@@ -27,17 +27,17 @@ of the License or (at your option) any l
- #pragma once
-
- //#include "inc/FeatureMap.h"
- //#include "inc/GlyphsCache.h"
- //#include "inc/Silf.h"
-
- #ifndef GRAPHITE2_NFILEFACE
-
--#include <stdio.h>
-+#include <cstdio>
- #include <cassert>
-
- #include "graphite2/Font.h"
-
- #include "inc/Main.h"
- #include "inc/TtfTypes.h"
- #include "inc/TtfUtil.h"
-
-diff --git a/gfx/graphite2/src/inc/GlyphCache.h b/gfx/graphite2/src/inc/GlyphCache.h
---- a/gfx/graphite2/src/inc/GlyphCache.h
-+++ b/gfx/graphite2/src/inc/GlyphCache.h
-@@ -24,24 +24,73 @@ Mozilla Public License (http://mozilla.o
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- #pragma once
-
-
- #include "graphite2/Font.h"
- #include "inc/Main.h"
-+#include "inc/Position.h"
-+#include "inc/GlyphFace.h"
-
- namespace graphite2 {
-
- class Face;
- class FeatureVal;
--class GlyphFace;
- class Segment;
-
-+
-+struct SlantBox
-+{
-+ static const SlantBox empty;
-+
-+// SlantBox(float psi = 0., float pdi = 0., float psa = 0., float pda = 0.) : si(psi), di(pdi), sa(psa), da(pda) {};
-+ float width() const { return sa - si; }
-+ float height() const { return da - di; }
-+ float si; // min
-+ float di; // min
-+ float sa; // max
-+ float da; // max
-+};
-+
-+
-+struct BBox
-+{
-+ BBox(float pxi = 0, float pyi = 0., float pxa = 0., float pya = 0.) : xi(pxi), yi(pyi), xa(pxa), ya(pya) {};
-+ float width() const { return xa - xi; }
-+ float height() const { return ya - yi; }
-+ float xi; // min
-+ float yi; // min
-+ float xa; // max
-+ float ya; // max
-+};
-+
-+
-+class GlyphBox
-+{
-+ GlyphBox(const GlyphBox &);
-+ GlyphBox & operator = (const GlyphBox &);
-+
-+public:
-+ GlyphBox(uint8 numsubs, unsigned short bitmap, Rect *slanted) : _num(numsubs), _bitmap(bitmap), _slant(*slanted) {};
-+
-+ void addSubBox(int subindex, int boundary, Rect *val) { _subs[subindex * 2 + boundary] = *val; }
-+ Rect &subVal(int subindex, int boundary) { return _subs[subindex * 2 + boundary]; }
-+ const Rect &slant() const { return _slant; }
-+ uint8 num() const { return _num; }
-+ const Rect *subs() const { return _subs; }
-+
-+private:
-+ uint8 _num;
-+ unsigned short _bitmap;
-+ Rect _slant;
-+ Rect _subs[1];
-+};
-+
- class GlyphCache
- {
- class Loader;
-
- GlyphCache(const GlyphCache&);
- GlyphCache& operator=(const GlyphCache&);
-
- public:
-@@ -49,22 +98,34 @@ public:
- ~GlyphCache();
-
- unsigned short numGlyphs() const throw();
- unsigned short numAttrs() const throw();
- unsigned short unitsPerEm() const throw();
-
- const GlyphFace *glyph(unsigned short glyphid) const; //result may be changed by subsequent call with a different glyphid
- const GlyphFace *glyphSafe(unsigned short glyphid) const;
-+ float getBoundingMetric(unsigned short glyphid, uint8 metric) const;
-+ uint8 numSubBounds(unsigned short glyphid) const;
-+ float getSubBoundingMetric(unsigned short glyphid, uint8 subindex, uint8 metric) const;
-+ const Rect & slant(unsigned short glyphid) const { return _boxes[glyphid] ? _boxes[glyphid]->slant() : _empty_slant_box; }
-+ const SlantBox & getBoundingSlantBox(unsigned short glyphid) const;
-+ const BBox & getBoundingBBox(unsigned short glyphid) const;
-+ const SlantBox & getSubBoundingSlantBox(unsigned short glyphid, uint8 subindex) const;
-+ const BBox & getSubBoundingBBox(unsigned short glyphid, uint8 subindex) const;
-+ bool check(unsigned short glyphid) const;
-+ bool hasBoxes() const { return _boxes != 0; }
-
- CLASS_NEW_DELETE;
-
- private:
-+ const Rect _empty_slant_box;
- const Loader * _glyph_loader;
- const GlyphFace * * _glyphs;
-+ GlyphBox * * _boxes;
- unsigned short _num_glyphs,
- _num_attrs,
- _upem;
- };
-
- inline
- unsigned short GlyphCache::numGlyphs() const throw()
- {
-@@ -79,14 +140,84 @@ unsigned short GlyphCache::numAttrs() co
-
- inline
- unsigned short GlyphCache::unitsPerEm() const throw()
- {
- return _upem;
- }
-
- inline
-+bool GlyphCache::check(unsigned short glyphid) const
-+{
-+ return _boxes && glyphid < _num_glyphs;
-+}
-+
-+inline
- const GlyphFace *GlyphCache::glyphSafe(unsigned short glyphid) const
- {
- return glyphid < _num_glyphs ? glyph(glyphid) : NULL;
- }
-
-+inline
-+float GlyphCache::getBoundingMetric(unsigned short glyphid, uint8 metric) const
-+{
-+ if (glyphid >= _num_glyphs) return 0.;
-+ switch (metric) {
-+ case 0: return (float)(glyph(glyphid)->theBBox().bl.x); // x_min
-+ case 1: return (float)(glyph(glyphid)->theBBox().bl.y); // y_min
-+ case 2: return (float)(glyph(glyphid)->theBBox().tr.x); // x_max
-+ case 3: return (float)(glyph(glyphid)->theBBox().tr.y); // y_max
-+ case 4: return (float)(_boxes[glyphid] ? _boxes[glyphid]->slant().bl.x : 0.f); // sum_min
-+ case 5: return (float)(_boxes[glyphid] ? _boxes[glyphid]->slant().bl.y : 0.f); // diff_min
-+ case 6: return (float)(_boxes[glyphid] ? _boxes[glyphid]->slant().tr.x : 0.f); // sum_max
-+ case 7: return (float)(_boxes[glyphid] ? _boxes[glyphid]->slant().tr.y : 0.f); // diff_max
-+ default: return 0.;
-+ }
-+}
-+
-+inline const SlantBox &GlyphCache::getBoundingSlantBox(unsigned short glyphid) const
-+{
-+ return _boxes[glyphid] ? *(SlantBox *)(&(_boxes[glyphid]->slant())) : SlantBox::empty;
-+}
-+
-+inline const BBox &GlyphCache::getBoundingBBox(unsigned short glyphid) const
-+{
-+ return *(BBox *)(&(glyph(glyphid)->theBBox()));
-+}
-+
-+inline
-+float GlyphCache::getSubBoundingMetric(unsigned short glyphid, uint8 subindex, uint8 metric) const
-+{
-+ GlyphBox *b = _boxes[glyphid];
-+ if (b == NULL || subindex >= b->num()) return 0;
-+
-+ switch (metric) {
-+ case 0: return b->subVal(subindex, 0).bl.x;
-+ case 1: return b->subVal(subindex, 0).bl.y;
-+ case 2: return b->subVal(subindex, 0).tr.x;
-+ case 3: return b->subVal(subindex, 0).tr.y;
-+ case 4: return b->subVal(subindex, 1).bl.x;
-+ case 5: return b->subVal(subindex, 1).bl.y;
-+ case 6: return b->subVal(subindex, 1).tr.x;
-+ case 7: return b->subVal(subindex, 1).tr.y;
-+ default: return 0.;
-+ }
-+}
-+
-+inline const SlantBox &GlyphCache::getSubBoundingSlantBox(unsigned short glyphid, uint8 subindex) const
-+{
-+ GlyphBox *b = _boxes[glyphid];
-+ return *(SlantBox *)(b->subs() + 2 * subindex + 1);
-+}
-+
-+inline const BBox &GlyphCache::getSubBoundingBBox(unsigned short glyphid, uint8 subindex) const
-+{
-+ GlyphBox *b = _boxes[glyphid];
-+ return *(BBox *)(b->subs() + 2 * subindex);
-+}
-+
-+inline
-+uint8 GlyphCache::numSubBounds(unsigned short glyphid) const
-+{
-+ return _boxes[glyphid] ? _boxes[glyphid]->num() : 0;
-+}
-+
- } // namespace graphite2
-diff --git a/gfx/graphite2/src/inc/Intervals.h b/gfx/graphite2/src/inc/Intervals.h
-new file mode 100644
---- /dev/null
-+++ b/gfx/graphite2/src/inc/Intervals.h
-@@ -0,0 +1,234 @@
-+/* GRAPHITE2 LICENSING
-+
-+ Copyright 2010, SIL International
-+ All rights reserved.
-+
-+ This library is free software; you can redistribute it and/or modify
-+ it under the terms of the GNU Lesser General Public License as published
-+ by the Free Software Foundation; either version 2.1 of License, or
-+ (at your option) any later version.
-+
-+ This program is distributed in the hope that it will be useful,
-+ but WITHOUT ANY WARRANTY; without even the implied warranty of
-+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-+ Lesser General Public License for more details.
-+
-+ You should also have received a copy of the GNU Lesser General Public
-+ License along with this library in the file named "LICENSE".
-+ If not, write to the Free Software Foundation, 51 Franklin Street,
-+ Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
-+ internet at http://www.fsf.org/licenses/lgpl.html.
-+
-+Alternatively, the contents of this file may be used under the terms of the
-+Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
-+License, as published by the Free Software Foundation, either version 2
-+of the License or (at your option) any later version.
-+*/
-+#pragma once
-+
-+#include <utility>
-+
-+#include "inc/Main.h"
-+#include "inc/List.h"
-+#include "inc/json.h"
-+#include "inc/Position.h"
-+
-+// An IntervalSet represents the possible movement of a given glyph in a given direction
-+// (horizontally, vertically, or diagonally).
-+// A vector is needed to represent disjoint ranges, eg, -300..-150, 20..200, 500..750.
-+// Each pair represents the min/max of a sub-range.
-+
-+namespace graphite2 {
-+
-+class Segment;
-+
-+enum zones_t {SD, XY};
-+
-+class Zones
-+{
-+ struct Exclusion
-+ {
-+ template<zones_t O>
-+ static Exclusion weighted(float xmin, float xmax, float f, float a0,
-+ float m, float xi, float ai, float c, bool nega);
-+
-+ float x, // x position
-+ xm, // xmax position
-+ c, // constant + sum(MiXi^2)
-+ sm, // sum(Mi)
-+ smx; // sum(MiXi)
-+ bool open;
-+
-+ Exclusion(float x, float w, float smi, float smxi, float c);
-+ Exclusion & operator += (Exclusion const & rhs);
-+ uint8 outcode(float p) const;
-+
-+ Exclusion split_at(float p);
-+ void left_trim(float p);
-+
-+ bool track_cost(float & cost, float & x, float origin) const;
-+
-+ private:
-+ float test_position(float origin) const;
-+ float cost(float x) const;
-+ };
-+
-+ typedef Vector<Exclusion> exclusions;
-+
-+ typedef exclusions::iterator iterator;
-+ typedef Exclusion * pointer;
-+ typedef Exclusion & reference;
-+ typedef std::reverse_iterator<iterator> reverse_iterator;
-+
-+public:
-+ typedef exclusions::const_iterator const_iterator;
-+ typedef Exclusion const * const_pointer;
-+ typedef Exclusion const & const_reference;
-+ typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
-+
-+#if !defined GRAPHITE2_NTRACING
-+ struct Debug
-+ {
-+ Exclusion _excl;
-+ bool _isdel;
-+ Vector<void *> _env;
-+
-+ Debug(Exclusion *e, bool isdel, json *dbg) : _excl(*e), _isdel(isdel), _env(dbg->getenvs()) { };
-+ };
-+
-+ typedef Vector<Debug> debugs;
-+ typedef debugs::const_iterator idebugs;
-+ void addDebug(Exclusion *e);
-+ void removeDebug(float pos, float posm);
-+ void setdebug(json *dbgout) { _dbg = dbgout; }
-+ idebugs dbgs_begin() const { return _dbgs.begin(); }
-+ idebugs dbgs_end() const { return _dbgs.end(); }
-+ void jsonDbgOut(Segment *seg) const;
-+ Position position() const { return Position(_pos, _posm); }
-+#endif
-+
-+ Zones();
-+ template<zones_t O>
-+ void initialise(float xmin, float xmax, float margin_len, float margin_weight, float ao);
-+
-+ void exclude(float xmin, float xmax);
-+ void exclude_with_margins(float xmin, float xmax, int axis);
-+
-+ template<zones_t O>
-+ void weighted(float xmin, float xmax, float f, float a0, float mi, float xi, float ai, float c, bool nega);
-+ void weightedAxis(int axis, float xmin, float xmax, float f, float a0, float mi, float xi, float ai, float c, bool nega);
-+
-+ float closest( float origin, float &cost) const;
-+
-+ const_iterator begin() const { return _exclusions.begin(); }
-+ const_iterator end() const { return _exclusions.end(); }
-+
-+private:
-+ exclusions _exclusions;
-+#if !defined GRAPHITE2_NTRACING
-+ json * _dbg;
-+ debugs _dbgs;
-+#endif
-+ float _margin_len,
-+ _margin_weight,
-+ _pos,
-+ _posm;
-+
-+ void insert(Exclusion e);
-+ void remove(float x, float xm);
-+ const_iterator find_exclusion_under(float x) const;
-+};
-+
-+
-+inline
-+Zones::Zones()
-+: _margin_len(0), _margin_weight(0), _pos(0), _posm(0)
-+{
-+#if !defined GRAPHITE2_NTRACING
-+ _dbg = 0;
-+#endif
-+ _exclusions.reserve(8);
-+}
-+
-+inline
-+Zones::Exclusion::Exclusion(float x_, float xm_, float smi, float smxi, float c_)
-+: x(x_), xm(xm_), c(c_), sm(smi), smx(smxi), open(false)
-+{ }
-+
-+template<zones_t O>
-+inline
-+void Zones::initialise(float xmin, float xmax, float margin_len,
-+ float margin_weight, float a0)
-+{
-+ _margin_len = margin_len;
-+ _margin_weight = margin_weight;
-+ _pos = xmin;
-+ _posm = xmax;
-+ _exclusions.clear();
-+ _exclusions.push_back(Exclusion::weighted<O>(xmin, xmax, 1, a0, 0, 0, 0, 0, false));
-+ _exclusions.front().open = true;
-+#if !defined GRAPHITE2_NTRACING
-+ _dbgs.clear();
-+#endif
-+}
-+
-+inline
-+void Zones::exclude(float xmin, float xmax) {
-+ remove(xmin, xmax);
-+}
-+
-+template<zones_t O>
-+inline
-+void Zones::weighted(float xmin, float xmax, float f, float a0,
-+ float m, float xi, float ai, float c, bool nega) {
-+ insert(Exclusion::weighted<O>(xmin, xmax, f, a0, m, xi, ai, c, nega));
-+}
-+
-+inline
-+void Zones::weightedAxis(int axis, float xmin, float xmax, float f, float a0,
-+ float m, float xi, float ai, float c, bool nega) {
-+ if (axis < 2)
-+ weighted<XY>(xmin, xmax, f, a0, m, xi, ai, c, nega);
-+ else
-+ weighted<SD>(xmin, xmax, f, a0, m, xi, ai, c, nega);
-+}
-+
-+#if !defined GRAPHITE2_NTRACING
-+inline
-+void Zones::addDebug(Exclusion *e) {
-+ if (_dbg)
-+ _dbgs.push_back(Debug(e, false, _dbg));
-+}
-+
-+inline
-+void Zones::removeDebug(float pos, float posm) {
-+ if (_dbg)
-+ {
-+ Exclusion e(pos, posm, 0, 0, 0);
-+ _dbgs.push_back(Debug(&e, true, _dbg));
-+ }
-+}
-+#endif
-+
-+template<>
-+inline
-+Zones::Exclusion Zones::Exclusion::weighted<XY>(float xmin, float xmax, float f, float a0,
-+ float m, float xi, GR_MAYBE_UNUSED float ai, float c, GR_MAYBE_UNUSED bool nega) {
-+ return Exclusion(xmin, xmax,
-+ m + f,
-+ m * xi,
-+ m * xi * xi + f * a0 * a0 + c);
-+}
-+
-+template<>
-+inline
-+Zones::Exclusion Zones::Exclusion::weighted<SD>(float xmin, float xmax, float f, float a0,
-+ float m, float xi, float ai,float c, bool nega) {
-+ float xia = nega ? xi - ai : xi + ai;
-+ return Exclusion(xmin, xmax,
-+ 0.25f * (m + 2.f * f),
-+ 0.25f * m * xia,
-+ 0.25f * (m * xia * xia + 2.f * f * a0 * a0) + c);
-+}
-+
-+} // end of namespace graphite2
-diff --git a/gfx/graphite2/src/inc/List.h b/gfx/graphite2/src/inc/List.h
---- a/gfx/graphite2/src/inc/List.h
-+++ b/gfx/graphite2/src/inc/List.h
-@@ -65,29 +65,30 @@ public:
- iterator end() { return m_last; }
- const_iterator end() const { return m_last; }
-
- bool empty() const { return m_first == m_last; }
- size_t size() const { return m_last - m_first; }
- size_t capacity() const{ return m_end - m_first; }
-
- void reserve(size_t n);
-+ void resize(size_t n, const T & v = T());
-
- reference front() { assert(size() > 0); return *begin(); }
- const_reference front() const { assert(size() > 0); return *begin(); }
- reference back() { assert(size() > 0); return *(end()-1); }
- const_reference back() const { assert(size() > 0); return *(end()-1); }
-
- Vector<T> & operator = (const Vector<T> & rhs) { assign(rhs.begin(), rhs.end()); return *this; }
- reference operator [] (size_t n) { assert(size() > n); return m_first[n]; }
- const_reference operator [] (size_t n) const { assert(size() > n); return m_first[n]; }
-
- void assign(size_t n, const T& u) { clear(); insert(begin(), n, u); }
- void assign(const_iterator first, const_iterator last) { clear(); insert(begin(), first, last); }
-- iterator insert(iterator p, const T & x) { p = _insert_default(p, 1); *p = x; return p; }
-+ iterator insert(iterator p, const T & x) { p = _insert_default(p, 1); new (p) T(x); return p; }
- void insert(iterator p, size_t n, const T & x);
- void insert(iterator p, const_iterator first, const_iterator last);
- void pop_back() { assert(size() > 0); --m_last; }
- void push_back(const T &v) { if (m_last == m_end) reserve(size()+1); new (m_last++) T(v); }
-
- void clear() { erase(begin(), end()); }
- iterator erase(iterator p) { return erase(p, p+1); }
- iterator erase(iterator first, iterator last);
-@@ -99,28 +100,37 @@ private:
- template <typename T>
- inline
- void Vector<T>::reserve(size_t n)
- {
- if (n > capacity())
- {
- const ptrdiff_t sz = size();
- m_first = static_cast<T*>(realloc(m_first, n*sizeof(T)));
-+ if (!m_first) std::abort();
- m_last = m_first + sz;
- m_end = m_first + n;
- }
- }
-
-+template <typename T>
-+inline
-+void Vector<T>::resize(size_t n, const T & v) {
-+ const ptrdiff_t d = n-size();
-+ if (d < 0) erase(end()+d, end());
-+ else if (d > 0) insert(end(), d, v);
-+}
-+
- template<typename T>
- inline
- typename Vector<T>::iterator Vector<T>::_insert_default(iterator p, size_t n)
- {
- assert(begin() <= p && p <= end());
- const ptrdiff_t i = p - begin();
-- reserve((size() + n + 7) >> 3 << 3);
-+ reserve(((size() + n + 7) >> 3) << 3);
- p = begin() + i;
- // Move tail if there is one
- if (p != end()) memmove(p + n, p, distance(p,end())*sizeof(T));
- m_last += n;
- return p;
- }
-
- template<typename T>
-diff --git a/gfx/graphite2/src/inc/Machine.h b/gfx/graphite2/src/inc/Machine.h
---- a/gfx/graphite2/src/inc/Machine.h
-+++ b/gfx/graphite2/src/inc/Machine.h
-@@ -105,17 +105,19 @@ enum opcode {
-
- PUSH_IGLYPH_ATTR, // not implemented
-
- POP_RET, RET_ZERO, RET_TRUE,
- IATTR_SET, IATTR_ADD, IATTR_SUB,
- PUSH_PROC_STATE, PUSH_VERSION,
- PUT_SUBS, PUT_SUBS2, PUT_SUBS3,
- PUT_GLYPH, PUSH_GLYPH_ATTR, PUSH_ATT_TO_GLYPH_ATTR,
-- MAX_OPCODE,
-+ BITOR, BITAND, BITNOT,
-+ BITSET, SET_FEAT,
-+ MAX_OPCODE,
- // private opcodes for internal use only, comes after all other on disk opcodes
- TEMP_COPY = MAX_OPCODE
- };
-
- struct opcode_t
- {
- instr impl[2];
- uint8 param_sz;
-@@ -143,17 +145,17 @@ public:
-
- Machine(SlotMap &) throw();
- static const opcode_t * getOpcodeTable() throw();
-
- CLASS_NEW_DELETE;
-
- SlotMap & slotMap() const throw();
- status_t status() const throw();
-- operator bool () const throw();
-+// operator bool () const throw();
-
- private:
- void check_final_stack(const stack_t * const sp);
- stack_t run(const instr * program, const byte * data,
- slotref * & map) HOT;
-
- SlotMap & _map;
- stack_t _stack[STACK_MAX + 2*STACK_GUARD];
-diff --git a/gfx/graphite2/src/inc/Main.h b/gfx/graphite2/src/inc/Main.h
---- a/gfx/graphite2/src/inc/Main.h
-+++ b/gfx/graphite2/src/inc/Main.h
-@@ -80,25 +80,25 @@ struct telemetry {};
- // typesafe wrapper around malloc for simple types
- // use free(pointer) to deallocate
-
- template <typename T> T * gralloc(size_t n)
- {
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::count_bytes(sizeof(T) * n);
- #endif
-- return reinterpret_cast<T*>(malloc(sizeof(T) * n));
-+ return static_cast<T*>(malloc(sizeof(T) * n));
- }
-
- template <typename T> T * grzeroalloc(size_t n)
- {
- #ifdef GRAPHITE2_TELEMETRY
- telemetry::count_bytes(sizeof(T) * n);
- #endif
-- return reinterpret_cast<T*>(calloc(n, sizeof(T)));
-+ return static_cast<T*>(calloc(n, sizeof(T)));
- }
-
- template <typename T>
- inline T min(const T a, const T b)
- {
- return a < b ? a : b;
- }
-
-@@ -115,13 +115,32 @@ inline T max(const T a, const T b)
- void * operator new (size_t, void * p) throw() { return p; } \
- void * operator new[] (size_t size) {return gralloc<byte>(size);} \
- void * operator new[] (size_t, void * p) throw() { return p; } \
- void operator delete (void * p) throw() { free(p);} \
- void operator delete (void *, void *) throw() {} \
- void operator delete[] (void * p)throw() { free(p); } \
- void operator delete[] (void *, void *) throw() {}
-
--#ifdef __GNUC__
-+#if defined(__GNUC__) || defined(__clang__)
- #define GR_MAYBE_UNUSED __attribute__((unused))
- #else
- #define GR_MAYBE_UNUSED
- #endif
-+
-+#if defined(__clang__) && __cplusplus >= 201103L
-+ /* clang's fallthrough annotations are only available starting in C++11. */
-+ #define GR_FALLTHROUGH [[clang::fallthrough]]
-+#elif defined(_MSC_VER)
-+ /*
-+ * MSVC's __fallthrough annotations are checked by /analyze (Code Analysis):
-+ * https://msdn.microsoft.com/en-us/library/ms235402%28VS.80%29.aspx
-+ */
-+ #include <sal.h>
-+ #define GR_FALLTHROUGH __fallthrough
-+#else
-+ #define GR_FALLTHROUGH /* fallthrough */
-+#endif
-+
-+#ifdef _MSC_VER
-+#pragma warning(disable: 4800)
-+#pragma warning(disable: 4355)
-+#endif
-diff --git a/gfx/graphite2/src/inc/Pass.h b/gfx/graphite2/src/inc/Pass.h
---- a/gfx/graphite2/src/inc/Pass.h
-+++ b/gfx/graphite2/src/inc/Pass.h
-@@ -34,65 +34,85 @@ namespace graphite2 {
- class Segment;
- class Face;
- class Silf;
- struct Rule;
- struct RuleEntry;
- struct State;
- class FiniteStateMachine;
- class Error;
-+class ShiftCollider;
-+class KernCollider;
-+class json;
-+
-+enum passtype;
-
- class Pass
- {
- public:
- Pass();
- ~Pass();
-
-- bool readPass(const byte * pPass, size_t pass_length, size_t subtable_base, Face & face, Error &e);
-- void runGraphite(vm::Machine & m, FiniteStateMachine & fsm) const;
-+ bool readPass(const byte * pPass, size_t pass_length, size_t subtable_base, Face & face,
-+ enum passtype pt, uint32 version, Error &e);
-+ bool runGraphite(vm::Machine & m, FiniteStateMachine & fsm, bool reverse) const;
- void init(Silf *silf) { m_silf = silf; }
-- byte spaceContextuals() const { return (m_flags & 0x0E) >> 1; }
-+ byte collisionLoops() const { return m_numCollRuns; }
-+ bool reverseDir() const { return m_isReverseDir; }
-
- CLASS_NEW_DELETE
- private:
- void findNDoRule(Slot* & iSlot, vm::Machine &, FiniteStateMachine& fsm) const;
- int doAction(const vm::Machine::Code* codeptr, Slot * & slot_out, vm::Machine &) const;
- bool testPassConstraint(vm::Machine & m) const;
- bool testConstraint(const Rule & r, vm::Machine &) const;
- bool readRules(const byte * rule_map, const size_t num_entries,
- const byte *precontext, const uint16 * sort_key,
- const uint16 * o_constraint, const byte *constraint_data,
- const uint16 * o_action, const byte * action_data,
-- Face &, Error &e);
-+ Face &, enum passtype pt, Error &e);
- bool readStates(const byte * starts, const byte * states, const byte * o_rule_map, Face &, Error &e);
- bool readRanges(const byte * ranges, size_t num_ranges, Error &e);
- uint16 glyphToCol(const uint16 gid) const;
- bool runFSM(FiniteStateMachine & fsm, Slot * slot) const;
- void dumpRuleEventConsidered(const FiniteStateMachine & fsm, const RuleEntry & re) const;
-- void dumpRuleEventOutput(const FiniteStateMachine & fsm, const Rule & r, Slot * os) const;
-+ void dumpRuleEventOutput(const FiniteStateMachine & fsm, vm::Machine & m, const Rule & r, Slot * os) const;
- void adjustSlot(int delta, Slot * & slot_out, SlotMap &) const;
-- const Silf* m_silf;
-- uint16 * m_cols;
-- Rule * m_rules; // rules
-- RuleEntry * m_ruleMap;
-- uint16 * m_startStates; // prectxt length
-- uint16 * m_transitions;
-- State * m_states;
--
-- byte m_flags;
-+ bool collisionShift(Segment *seg, int dir, json * const dbgout) const;
-+ bool collisionKern(Segment *seg, int dir, json * const dbgout) const;
-+ bool collisionFinish(Segment *seg, GR_MAYBE_UNUSED json * const dbgout) const;
-+ bool resolveCollisions(Segment *seg, Slot *slot, Slot *start, ShiftCollider &coll, bool isRev,
-+ int dir, bool &moved, bool &hasCol, json * const dbgout) const;
-+ float resolveKern(Segment *seg, Slot *slot, Slot *start, KernCollider &coll, int dir,
-+ float &ymin, float &ymax, json *const dbgout) const;
-+
-+ const Silf * m_silf;
-+ uint16 * m_cols;
-+ Rule * m_rules; // rules
-+ RuleEntry * m_ruleMap;
-+ uint16 * m_startStates; // prectxt length
-+ uint16 * m_transitions;
-+ State * m_states;
-+ vm::Machine::Code * m_codes;
-+ byte * m_progs;
-+
-+ byte m_numCollRuns;
-+ byte m_kernColls;
- byte m_iMaxLoop;
- uint16 m_numGlyphs;
- uint16 m_numRules;
- uint16 m_numStates;
- uint16 m_numTransition;
- uint16 m_numSuccess;
- uint16 m_successStart;
- uint16 m_numColumns;
- byte m_minPreCtxt;
- byte m_maxPreCtxt;
-+ byte m_colThreshold;
-+ bool m_isReverseDir;
- vm::Machine::Code m_cPConstraint;
-
- private: //defensive
- Pass(const Pass&);
- Pass& operator=(const Pass&);
- };
-
- } // namespace graphite2
-diff --git a/gfx/graphite2/src/inc/Position.h b/gfx/graphite2/src/inc/Position.h
---- a/gfx/graphite2/src/inc/Position.h
-+++ b/gfx/graphite2/src/inc/Position.h
-@@ -45,15 +45,24 @@ public:
-
- class Rect
- {
- public :
- Rect() {}
- Rect(const Position& botLeft, const Position& topRight): bl(botLeft), tr(topRight) {}
- Rect widen(const Rect& other) { return Rect(Position(bl.x > other.bl.x ? other.bl.x : bl.x, bl.y > other.bl.y ? other.bl.y : bl.y), Position(tr.x > other.tr.x ? tr.x : other.tr.x, tr.y > other.tr.y ? tr.y : other.tr.y)); }
- Rect operator + (const Position &a) const { return Rect(Position(bl.x + a.x, bl.y + a.y), Position(tr.x + a.x, tr.y + a.y)); }
-+ Rect operator - (const Position &a) const { return Rect(Position(bl.x - a.x, bl.y - a.y), Position(tr.x - a.x, tr.y - a.y)); }
- Rect operator * (float m) const { return Rect(Position(bl.x, bl.y) * m, Position(tr.x, tr.y) * m); }
-+ float width() const { return tr.x - bl.x; }
-+ float height() const { return tr.y - bl.y; }
-+
-+ bool hitTest(Rect &other);
-+
-+ // returns Position(overlapx, overlapy) where overlap<0 if overlapping else positive)
-+ Position overlap(Position &offset, Rect &other, Position &otherOffset);
-+ //Position constrainedAvoid(Position &offset, Rect &box, Rect &sdbox, Position &other, Rect &obox, Rect &osdbox);
-
- Position bl;
- Position tr;
- };
-
- } // namespace graphite2
-diff --git a/gfx/graphite2/src/inc/Rule.h b/gfx/graphite2/src/inc/Rule.h
---- a/gfx/graphite2/src/inc/Rule.h
-+++ b/gfx/graphite2/src/inc/Rule.h
-@@ -36,30 +36,36 @@ struct Rule {
- const vm::Machine::Code * constraint,
- * action;
- unsigned short sort;
- byte preContext;
- #ifndef NDEBUG
- uint16 rule_idx;
- #endif
-
-- Rule() : constraint(0), action(0), sort(0), preContext(0) {}
-- ~Rule();
-+ Rule();
-+ ~Rule() {}
-
- CLASS_NEW_DELETE;
-
- private:
- Rule(const Rule &);
- Rule & operator = (const Rule &);
- };
-
--inline Rule::~Rule()
-+inline
-+Rule::Rule()
-+: constraint(0),
-+ action(0),
-+ sort(0),
-+ preContext(0)
- {
-- delete constraint;
-- delete action;
-+#ifndef NDEBUG
-+ rule_idx = 0;
-+#endif
- }
-
-
- struct RuleEntry
- {
- const Rule * rule;
-
- inline
-@@ -91,40 +97,43 @@ bool State::empty() const
- return rules_end == rules;
- }
-
-
- class SlotMap
- {
- public:
- enum {MAX_SLOTS=64};
-- SlotMap(Segment & seg);
-+ SlotMap(Segment & seg, uint8 direction);
-
- Slot * * begin();
- Slot * * end();
- size_t size() const;
- unsigned short context() const;
- void reset(Slot &, unsigned short);
-
- Slot * const & operator[](int n) const;
- Slot * & operator [] (int);
- void pushSlot(Slot * const slot);
-- void collectGarbage();
-+ void collectGarbage(Slot *& aSlot);
-
- Slot * highwater() { return m_highwater; }
- void highwater(Slot *s) { m_highwater = s; m_highpassed = false; }
- bool highpassed() const { return m_highpassed; }
- void highpassed(bool v) { m_highpassed = v; }
-
-+ uint8 dir() const { return m_dir; }
-+
- Segment & segment;
- private:
- Slot * m_slot_map[MAX_SLOTS+1];
- unsigned short m_size;
- unsigned short m_precontext;
- Slot * m_highwater;
-+ uint8 m_dir;
- bool m_highpassed;
- };
-
-
- class FiniteStateMachine
- {
- public:
- enum {MAX_RULES=128};
-@@ -228,18 +237,18 @@ void FiniteStateMachine::Rules::accumula
- return;
- }
- }
- while (rre != rrend && out != lrend) { *out++ = *rre++; }
- m_end = out;
- }
-
- inline
--SlotMap::SlotMap(Segment & seg)
--: segment(seg), m_size(0), m_precontext(0), m_highwater(0), m_highpassed(false)
-+SlotMap::SlotMap(Segment & seg, uint8 direction)
-+: segment(seg), m_size(0), m_precontext(0), m_highwater(0), m_dir(direction), m_highpassed(false)
- {
- m_slot_map[0] = 0;
- }
-
- inline
- Slot * * SlotMap::begin()
- {
- return &m_slot_map[1]; // allow map to go 1 before slot_map when inserting
-diff --git a/gfx/graphite2/src/inc/SegCache.h b/gfx/graphite2/src/inc/SegCache.h
---- a/gfx/graphite2/src/inc/SegCache.h
-+++ b/gfx/graphite2/src/inc/SegCache.h
-@@ -258,17 +258,17 @@ public:
-
- CLASS_NEW_DELETE
- private:
- void freeLevel(SegCacheStore * store, SegCachePrefixArray prefixes, size_t level);
- void purgeLevel(SegCacheStore * store, SegCachePrefixArray prefixes, size_t level,
- unsigned long long minAccessCount, unsigned long long oldAccessTime);
-
- uint16 m_prefixLength;
-- uint16 m_maxCachedSegLength;
-+// uint16 m_maxCachedSegLength;
- size_t m_segmentCount;
- SegCachePrefixArray m_prefixes;
- Features m_features;
- mutable unsigned long long m_totalAccessCount;
- mutable unsigned long long m_totalMisses;
- float m_purgeFactor;
- };
-
-diff --git a/gfx/graphite2/src/inc/Segment.h b/gfx/graphite2/src/inc/Segment.h
---- a/gfx/graphite2/src/inc/Segment.h
-+++ b/gfx/graphite2/src/inc/Segment.h
-@@ -30,29 +30,28 @@ of the License or (at your option) any l
-
- #include <cassert>
-
- #include "inc/CharInfo.h"
- #include "inc/Face.h"
- #include "inc/FeatureVal.h"
- #include "inc/GlyphCache.h"
- #include "inc/GlyphFace.h"
--//#include "inc/Silf.h"
- #include "inc/Slot.h"
- #include "inc/Position.h"
- #include "inc/List.h"
--#include "inc/Bidi.h"
-+#include "inc/Collider.h"
-
- #define MAX_SEG_GROWTH_FACTOR 256
-
- namespace graphite2 {
-
- typedef Vector<Features> FeatureList;
- typedef Vector<Slot *> SlotRope;
--typedef Vector<int16 *> AttributeRope;
-+typedef Vector<int16 *> AttributeRope;
- typedef Vector<SlotJustify *> JustifyRope;
-
- #ifndef GRAPHITE2_NSEGCACHE
- class SegmentScopeState;
- #endif
- class Font;
- class Segment;
- class Silf;
-@@ -81,119 +80,151 @@ private:
-
- class Segment
- {
- // Prevent copying of any kind.
- Segment(const Segment&);
- Segment& operator=(const Segment&);
-
- public:
-+
-+ enum {
-+ SEG_INITCOLLISIONS = 1,
-+ SEG_HASCOLLISIONS = 2
-+ };
-+
- unsigned int slotCount() const { return m_numGlyphs; } //one slot per glyph
- void extendLength(int num) { m_numGlyphs += num; }
- Position advance() const { return m_advance; }
- bool runGraphite() { if (m_silf) return m_face->runGraphite(this, m_silf); else return true;};
- void chooseSilf(uint32 script) { m_silf = m_face->chooseSilf(script); }
- const Silf *silf() const { return m_silf; }
- unsigned int charInfoCount() const { return m_numCharinfo; }
- const CharInfo *charinfo(unsigned int index) const { return index < m_numCharinfo ? m_charinfo + index : NULL; }
- CharInfo *charinfo(unsigned int index) { return index < m_numCharinfo ? m_charinfo + index : NULL; }
-- int8 dir() const { return m_dir; }
-
- Segment(unsigned int numchars, const Face* face, uint32 script, int dir);
- ~Segment();
- #ifndef GRAPHITE2_NSEGCACHE
- SegmentScopeState setScope(Slot * firstSlot, Slot * lastSlot, size_t subLength);
- void removeScope(SegmentScopeState & state);
- void append(const Segment &other);
- void splice(size_t offset, size_t length, Slot * const startSlot,
- Slot * endSlot, const Slot * srcSlot,
- const size_t numGlyphs);
- #endif
-+ uint8 flags() const { return m_flags; }
-+ void flags(uint8 f) { m_flags = f; }
- Slot *first() { return m_first; }
- void first(Slot *p) { m_first = p; }
- Slot *last() { return m_last; }
- void last(Slot *p) { m_last = p; }
- void appendSlot(int i, int cid, int gid, int fid, size_t coffset);
- Slot *newSlot();
- void freeSlot(Slot *);
- SlotJustify *newJustify();
- void freeJustify(SlotJustify *aJustify);
-- Position positionSlots(const Font *font, Slot *first=0, Slot *last=0);
-+ Position positionSlots(const Font *font=0, Slot *first=0, Slot *last=0, bool isRtl = false, bool isFinal = true);
- void associateChars(int offset, int num);
- void linkClusters(Slot *first, Slot *last);
- uint16 getClassGlyph(uint16 cid, uint16 offset) const { return m_silf->getClassGlyph(cid, offset); }
- uint16 findClassIndex(uint16 cid, uint16 gid) const { return m_silf->findClassIndex(cid, gid); }
- int addFeatures(const Features& feats) { m_feats.push_back(feats); return m_feats.size() - 1; }
- uint32 getFeature(int index, uint8 findex) const { const FeatureRef* pFR=m_face->theSill().theFeatureMap().featureRef(findex); if (!pFR) return 0; else return pFR->getFeatureVal(m_feats[index]); }
-+ void setFeature(int index, uint8 findex, uint32 val) {
-+ const FeatureRef* pFR=m_face->theSill().theFeatureMap().featureRef(findex);
-+ if (pFR)
-+ {
-+ if (val > pFR->maxVal()) val = pFR->maxVal();
-+ pFR->applyValToFeature(val, m_feats[index]);
-+ } }
-+ int8 dir() const { return m_dir; }
- void dir(int8 val) { m_dir = val; }
-+ bool currdir() const { return ((m_dir >> 6) ^ m_dir) & 1; }
- unsigned int passBits() const { return m_passBits; }
- void mergePassBits(const unsigned int val) { m_passBits &= val; }
- int16 glyphAttr(uint16 gid, uint16 gattr) const { const GlyphFace * p = m_face->glyphs().glyphSafe(gid); return p ? p->attrs()[gattr] : 0; }
-- int32 getGlyphMetric(Slot *iSlot, uint8 metric, uint8 attrLevel) const;
-+ int32 getGlyphMetric(Slot *iSlot, uint8 metric, uint8 attrLevel, bool rtl) const;
- float glyphAdvance(uint16 gid) const { return m_face->glyphs().glyph(gid)->theAdvance().x; }
- const Rect &theGlyphBBoxTemporary(uint16 gid) const { return m_face->glyphs().glyph(gid)->theBBox(); } //warning value may become invalid when another glyph is accessed
- Slot *findRoot(Slot *is) const { return is->attachedTo() ? findRoot(is->attachedTo()) : is; }
- int numAttrs() const { return m_silf->numUser(); }
- int defaultOriginal() const { return m_defaultOriginal; }
- const Face * getFace() const { return m_face; }
- const Features & getFeatures(unsigned int /*charIndex*/) { assert(m_feats.size() == 1); return m_feats[0]; }
-- void bidiPass(uint8 aBidi, int paradir, uint8 aMirror);
-+ void bidiPass(int paradir, uint8 aMirror);
-+ int8 getSlotBidiClass(Slot *s) const;
-+ void doMirror(uint16 aMirror);
- Slot *addLineEnd(Slot *nSlot);
- void delLineEnd(Slot *s);
- bool hasJustification() const { return m_justifies.size() != 0; }
-+ void reverseSlots();
-
- bool isWhitespace(const int cid) const;
--
-+ bool hasCollisionInfo() const { return (m_flags & SEG_HASCOLLISIONS); }
-+ SlotCollision *collisionInfo(const Slot *s) const { return m_collisions ? m_collisions + s->index() : 0; }
- CLASS_NEW_DELETE
-
- public: //only used by: GrSegment* makeAndInitialize(const GrFont *font, const GrFace *face, uint32 script, const FeaturesHandle& pFeats/*must not be IsNull*/, encform enc, const void* pStart, size_t nChars, int dir);
- bool read_text(const Face *face, const Features* pFeats/*must not be NULL*/, gr_encform enc, const void*pStart, size_t nChars);
-- void prepare_pos(const Font *font);
-- void finalise(const Font *font);
-+ void finalise(const Font *font, bool reverse=false);
- float justify(Slot *pSlot, const Font *font, float width, enum justFlags flags, Slot *pFirst, Slot *pLast);
-+ bool initCollisions();
-
- private:
- Position m_advance; // whole segment advance
- SlotRope m_slots; // Vector of slot buffers
- AttributeRope m_userAttrs; // Vector of userAttrs buffers
- JustifyRope m_justifies; // Slot justification info buffers
- FeatureList m_feats; // feature settings referenced by charinfos in this segment
- Slot * m_freeSlots; // linked list of free slots
- SlotJustify * m_freeJustifies; // Slot justification blocks free list
- CharInfo * m_charinfo; // character info, one per input character
-+ SlotCollision * m_collisions;
- const Face * m_face; // GrFace
- const Silf * m_silf;
- Slot * m_first; // first slot in segment
- Slot * m_last; // last slot in segment
- unsigned int m_bufSize, // how big a buffer to create when need more slots
- m_numGlyphs,
- m_numCharinfo, // size of the array and number of input characters
- m_passBits; // if bit set then skip pass
- int m_defaultOriginal; // number of whitespace chars in the string
- int8 m_dir;
-+ uint8 m_flags; // General purpose flags
- };
-
--
-+inline
-+int8 Segment::getSlotBidiClass(Slot *s) const
-+{
-+ int8 res = s->getBidiClass();
-+ if (res != -1) return res;
-+ res = int8(glyphAttr(s->gid(), m_silf->aBidi()));
-+ s->setBidiClass(res);
-+ return res;
-+}
-
- inline
--void Segment::finalise(const Font *font)
-+void Segment::finalise(const Font *font, bool reverse)
- {
- if (!m_first) return;
-
-- m_advance = positionSlots(font);
-- associateChars(0, m_numCharinfo);
-+ m_advance = positionSlots(font, m_first, m_last, m_silf->dir(), true);
-+ //associateChars(0, m_numCharinfo);
-+ if (reverse && currdir() != (m_dir & 1))
-+ reverseSlots();
- linkClusters(m_first, m_last);
- }
-
- inline
--int32 Segment::getGlyphMetric(Slot *iSlot, uint8 metric, uint8 attrLevel) const {
-+int32 Segment::getGlyphMetric(Slot *iSlot, uint8 metric, uint8 attrLevel, bool rtl) const {
- if (attrLevel > 0)
- {
- Slot *is = findRoot(iSlot);
-- return is->clusterMetric(this, metric, attrLevel);
-+ return is->clusterMetric(this, metric, attrLevel, rtl);
- }
- else
- return m_face->getGlyphMetric(iSlot->gid(), metric);
- }
-
- inline
- bool Segment::isWhitespace(const int cid) const
- {
-@@ -206,68 +237,12 @@ bool Segment::isWhitespace(const int cid
- + (cid >= 0x2000) * (cid <= 0x200A)
- + (cid == 0x2028)
- + (cid == 0x2029)
- + (cid == 0x202F)
- + (cid == 0x205F)
- + (cid == 0x3000)) != 0;
- }
-
--//inline
--//bool Segment::isWhitespace(const int cid) const
--//{
--// switch (cid >> 8)
--// {
--// case 0x00:
--// switch (cid)
--// {
--// case 0x09:
--// case 0x0A:
--// case 0x0B:
--// case 0x0C:
--// case 0x0D:
--// case 0x20:
--// return true;
--// default:
--// break;
--// }
--// break;
--// case 0x16:
--// return cid == 0x1680;
--// break;
--// case 0x18:
--// return cid == 0x180E;
--// break;
--// case 0x20:
--// switch (cid)
--// {
--// case 0x00:
--// case 0x01:
--// case 0x02:
--// case 0x03:
--// case 0x04:
--// case 0x05:
--// case 0x06:
--// case 0x07:
--// case 0x08:
--// case 0x09:
--// case 0x0A:
--// case 0x28:
--// case 0x29:
--// case 0x2F:
--// case 0x5F:
--// return true
--// default:
--// break;
--// }
--// break;
--// case 0x30:
--// return cid == 0x3000;
--// break;
--// }
--//
--// return false;
--//}
--
- } // namespace graphite2
-
- struct gr_segment : public graphite2::Segment {};
-
-diff --git a/gfx/graphite2/src/inc/Silf.h b/gfx/graphite2/src/inc/Silf.h
---- a/gfx/graphite2/src/inc/Silf.h
-+++ b/gfx/graphite2/src/inc/Silf.h
-@@ -80,24 +80,26 @@ public:
- uint16 getClassGlyph(uint16 cid, unsigned int index) const;
- uint16 findPseudo(uint32 uid) const;
- uint8 numUser() const { return m_aUser; }
- uint8 aPseudo() const { return m_aPseudo; }
- uint8 aBreak() const { return m_aBreak; }
- uint8 aMirror() const {return m_aMirror; }
- uint8 aPassBits() const { return m_aPassBits; }
- uint8 aBidi() const { return m_aBidi; }
-+ uint8 aCollision() const { return m_aCollision; }
- uint8 substitutionPass() const { return m_sPass; }
- uint8 positionPass() const { return m_pPass; }
- uint8 justificationPass() const { return m_jPass; }
- uint8 bidiPass() const { return m_bPass; }
- uint8 numPasses() const { return m_numPasses; }
- uint8 maxCompPerLig() const { return m_iMaxComp; }
- uint16 numClasses() const { return m_nClass; }
- byte flags() const { return m_flags; }
-+ byte dir() const { return m_dir; }
- uint8 numJustLevels() const { return m_numJusts; }
- Justinfo *justAttrs() const { return m_justs; }
- uint16 endLineGlyphid() const { return m_gEndLine; }
- const gr_faceinfo *silfInfo() const { return &m_silfinfo; }
-
- CLASS_NEW_DELETE;
-
- private:
-@@ -107,23 +109,20 @@ private:
- Pass * m_passes;
- Pseudo * m_pseudos;
- uint32 * m_classOffsets;
- uint16 * m_classData;
- Justinfo * m_justs;
- uint8 m_numPasses;
- uint8 m_numJusts;
- uint8 m_sPass, m_pPass, m_jPass, m_bPass,
-- m_flags;
-+ m_flags, m_dir;
-
-- uint8 m_aPseudo, m_aBreak, m_aUser, m_aBidi, m_aMirror, m_aPassBits,
-- m_iMaxComp;
-- uint16 m_aLig,
-- m_numPseudo,
-- m_nClass,
-- m_nLinear,
-- m_gEndLine;
-+ uint8 m_aPseudo, m_aBreak, m_aUser, m_aBidi, m_aMirror, m_aPassBits,
-+ m_iMaxComp, m_aCollision;
-+ uint16 m_aLig, m_numPseudo, m_nClass, m_nLinear,
-+ m_gEndLine;
- gr_faceinfo m_silfinfo;
-
- void releaseBuffers() throw();
- };
-
- } // namespace graphite2
-diff --git a/gfx/graphite2/src/inc/Slot.h b/gfx/graphite2/src/inc/Slot.h
---- a/gfx/graphite2/src/inc/Slot.h
-+++ b/gfx/graphite2/src/inc/Slot.h
-@@ -27,25 +27,23 @@ of the License or (at your option) any l
- #pragma once
-
- #include "graphite2/Types.h"
- #include "graphite2/Segment.h"
- #include "inc/Main.h"
- #include "inc/Font.h"
- #include "inc/Position.h"
-
--
--
- namespace graphite2 {
-
- typedef gr_attrCode attrCode;
-
- class GlyphFace;
-+class SegCacheEntry;
- class Segment;
--class SegCacheEntry;
-
- struct SlotJustify
- {
- static const int NUMJUSTPARAMS = 5;
-
- SlotJustify(const SlotJustify &);
- SlotJustify & operator = (const SlotJustify &);
-
-@@ -70,73 +68,79 @@ class Slot
- };
-
- public:
- struct iterator;
-
- unsigned short gid() const { return m_glyphid; }
- Position origin() const { return m_position; }
- float advance() const { return m_advance.x; }
-+ void advance(Position &val) { m_advance = val; }
- Position advancePos() const { return m_advance; }
- int before() const { return m_before; }
- int after() const { return m_after; }
- uint32 index() const { return m_index; }
- void index(uint32 val) { m_index = val; }
-
-- Slot();
-+ Slot(int16 *m_userAttr = NULL);
- void set(const Slot & slot, int charOffset, size_t numUserAttr, size_t justLevels, size_t numChars);
- Slot *next() const { return m_next; }
- void next(Slot *s) { m_next = s; }
- Slot *prev() const { return m_prev; }
- void prev(Slot *s) { m_prev = s; }
- uint16 glyph() const { return m_realglyphid ? m_realglyphid : m_glyphid; }
- void setGlyph(Segment *seg, uint16 glyphid, const GlyphFace * theGlyph = NULL);
- void setRealGid(uint16 realGid) { m_realglyphid = realGid; }
- void adjKern(const Position &pos) { m_shift = m_shift + pos; m_advance = m_advance + pos; }
- void origin(const Position &pos) { m_position = pos + m_shift; }
- void originate(int ind) { m_original = ind; }
- int original() const { return m_original; }
- void before(int ind) { m_before = ind; }
- void after(int ind) { m_after = ind; }
- bool isBase() const { return (!m_parent); }
- void update(int numSlots, int numCharInfo, Position &relpos);
-- Position finalise(const Segment* seg, const Font* font, Position & base, Rect & bbox, uint8 attrLevel, float & clusterMin);
-+ Position finalise(const Segment* seg, const Font* font, Position & base, Rect & bbox, uint8 attrLevel, float & clusterMin, bool rtl, bool isFinal);
- bool isDeleted() const { return (m_flags & DELETED) ? true : false; }
- void markDeleted(bool state) { if (state) m_flags |= DELETED; else m_flags &= ~DELETED; }
- bool isCopied() const { return (m_flags & COPIED) ? true : false; }
- void markCopied(bool state) { if (state) m_flags |= COPIED; else m_flags &= ~COPIED; }
- bool isPositioned() const { return (m_flags & POSITIONED) ? true : false; }
- void markPositioned(bool state) { if (state) m_flags |= POSITIONED; else m_flags &= ~POSITIONED; }
- bool isInsertBefore() const { return !(m_flags & INSERTED); }
- uint8 getBidiLevel() const { return m_bidiLevel; }
- void setBidiLevel(uint8 level) { m_bidiLevel = level; }
-+ int8 getBidiClass(const Segment *seg);
- int8 getBidiClass() const { return m_bidiCls; }
- void setBidiClass(int8 cls) { m_bidiCls = cls; }
- int16 *userAttrs() const { return m_userAttr; }
- void userAttrs(int16 *p) { m_userAttr = p; }
- void markInsertBefore(bool state) { if (!state) m_flags |= INSERTED; else m_flags &= ~INSERTED; }
- void setAttr(Segment* seg, attrCode ind, uint8 subindex, int16 val, const SlotMap & map);
- int getAttr(const Segment *seg, attrCode ind, uint8 subindex) const;
- int getJustify(const Segment *seg, uint8 level, uint8 subindex) const;
- void setJustify(Segment *seg, uint8 level, uint8 subindex, int16 value);
- bool isLocalJustify() const { return m_justs != NULL; };
- void attachTo(Slot *ap) { m_parent = ap; }
- Slot *attachedTo() const { return m_parent; }
- Position attachOffset() const { return m_attach - m_with; }
- Slot* firstChild() const { return m_child; }
-+ void firstChild(Slot *ap) { m_child = ap; }
- bool child(Slot *ap);
- Slot* nextSibling() const { return m_sibling; }
-+ void nextSibling(Slot *ap) { m_sibling = ap; }
- bool sibling(Slot *ap);
- bool removeChild(Slot *ap);
- bool removeSibling(Slot *ap);
-- int32 clusterMetric(const Segment* seg, uint8 metric, uint8 attrLevel);
-+ int32 clusterMetric(const Segment* seg, uint8 metric, uint8 attrLevel, bool rtl);
- void positionShift(Position a) { m_position += a; }
- void floodShift(Position adj);
- float just() const { return m_just; }
- void just(float j) { m_just = j; }
-+ Slot *nextInCluster(const Slot *s) const;
-+ bool isChildOf(const Slot *base) const;
-
- CLASS_NEW_DELETE
-
- private:
- Slot *m_next; // linked list of slots
- Slot *m_prev;
- unsigned short m_glyphid; // glyph id
- uint16 m_realglyphid;
-diff --git a/gfx/graphite2/src/inc/Sparse.h b/gfx/graphite2/src/inc/Sparse.h
---- a/gfx/graphite2/src/inc/Sparse.h
-+++ b/gfx/graphite2/src/inc/Sparse.h
-@@ -51,17 +51,17 @@ private:
- static const unsigned char SIZEOF_CHUNK = (sizeof(mask_t) - sizeof(key_type))*8;
-
- struct chunk
- {
- mask_t mask:SIZEOF_CHUNK;
- key_type offset;
- };
-
-- static chunk empty_chunk;
-+ static const chunk empty_chunk;
- sparse(const sparse &);
- sparse & operator = (const sparse &);
-
- public:
- template<typename I>
- sparse(I first, const I last);
- sparse() throw();
- ~sparse() throw();
-@@ -83,17 +83,17 @@ private:
- } m_array;
- key_type m_nchunks;
- };
-
-
- inline
- sparse::sparse() throw() : m_nchunks(0)
- {
-- m_array.map = &empty_chunk;
-+ m_array.map = const_cast<graphite2::sparse::chunk *>(&empty_chunk);
- }
-
-
- template <typename I>
- sparse::sparse(I attr, const I last)
- : m_nchunks(0)
- {
- m_array.map = 0;
-@@ -108,30 +108,31 @@ sparse::sparse(I attr, const I last)
- if (v.first <= lastkey) { m_nchunks = 0; return; }
-
- lastkey = v.first;
- const key_type k = v.first / SIZEOF_CHUNK;
- if (k >= m_nchunks) m_nchunks = k+1;
- }
- if (m_nchunks == 0)
- {
-- m_array.map=&empty_chunk;
-+ m_array.map=const_cast<graphite2::sparse::chunk *>(&empty_chunk);
- return;
- }
-
- m_array.values = grzeroalloc<mapped_type>((m_nchunks*sizeof(chunk) + sizeof(mapped_type)-1)
- / sizeof(mapped_type)
- + n_values);
-
- if (m_array.values == 0)
- {
- free(m_array.values); m_array.map=0;
- return;
- }
-
-+ // coverity[forward_null : FALSE] Since m_array is union and m_array.values is not NULL
- chunk * ci = m_array.map;
- ci->offset = (m_nchunks*sizeof(chunk) + sizeof(mapped_type)-1)/sizeof(mapped_type);
- mapped_type * vi = m_array.values + ci->offset;
- for (; attr != last; ++attr, ++vi)
- {
- const typename std::iterator_traits<I>::value_type v = *attr;
- if (v.second == 0) { --vi; continue; }
-
-diff --git a/gfx/graphite2/src/inc/TtfUtil.h b/gfx/graphite2/src/inc/TtfUtil.h
---- a/gfx/graphite2/src/inc/TtfUtil.h
-+++ b/gfx/graphite2/src/inc/TtfUtil.h
-@@ -132,21 +132,21 @@ public:
- int GetLangsForNames(const void * pName, int nPlatformId, int nEncodingId,
- int *nameIdList, int cNameIds, short *langIdList);
- void SwapWString(void * pWStr, size_t nSize = 0); // throw (std::invalid_argument);
- #endif
-
- ////////////////////////////////// cmap lookup tools
- const void * FindCmapSubtable(const void * pCmap, int nPlatformId = 3,
- int nEncodingId = 1, size_t length = 0);
-- bool CheckCmapSubtable4(const void * pCmap31);
-+ bool CheckCmapSubtable4(const void * pCmap31, size_t table_len /*, unsigned int maxgid*/);
- gid16 CmapSubtable4Lookup(const void * pCmapSubtabel4, unsigned int nUnicodeId, int rangeKey = 0);
- unsigned int CmapSubtable4NextCodepoint(const void *pCmap31, unsigned int nUnicodeId,
- int * pRangeKey = 0);
-- bool CheckCmapSubtable12(const void *pCmap310);
-+ bool CheckCmapSubtable12(const void *pCmap310, size_t table_len /*, unsigned int maxgid*/);
- gid16 CmapSubtable12Lookup(const void * pCmap310, unsigned int uUnicodeId, int rangeKey = 0);
- unsigned int CmapSubtable12NextCodepoint(const void *pCmap310, unsigned int nUnicodeId,
- int * pRangeKey = 0);
-
- ///////////////////////////////// horizontal metric data for a glyph
- bool HorMetrics(gid16 nGlyphId, const void * pHmtx, size_t lHmtxSize,
- const void * pHhea, int & nLsb, unsigned int & nAdvWid);
-
-diff --git a/gfx/graphite2/src/inc/UtfCodec.h b/gfx/graphite2/src/inc/UtfCodec.h
---- a/gfx/graphite2/src/inc/UtfCodec.h
-+++ b/gfx/graphite2/src/inc/UtfCodec.h
-@@ -126,19 +126,22 @@ public:
- static uchar_t get(const codeunit_t * cp, int8 & l) throw()
- {
- const int8 seq_sz = sz_lut[*cp >> 4];
- uchar_t u = *cp & mask_lut[seq_sz];
- l = 1;
- bool toolong = false;
-
- switch(seq_sz) {
-- case 4: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong = (u < 0x10); // no break
-- case 3: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong |= (u < 0x20); // no break
-- case 2: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong |= (u < 0x80); // no break
-+ case 4: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong = (u < 0x10); GR_FALLTHROUGH;
-+ // no break
-+ case 3: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong |= (u < 0x20); GR_FALLTHROUGH;
-+ // no break
-+ case 2: u <<= 6; u |= *++cp & 0x3F; if (*cp >> 6 != 2) break; ++l; toolong |= (u < 0x80); GR_FALLTHROUGH;
-+ // no break
- case 1: break;
- case 0: l = -1; return 0xFFFD;
- }
-
- if (l != seq_sz || toolong)
- {
- l = -l;
- return 0xFFFD;
-diff --git a/gfx/graphite2/src/inc/bits.h b/gfx/graphite2/src/inc/bits.h
---- a/gfx/graphite2/src/inc/bits.h
-+++ b/gfx/graphite2/src/inc/bits.h
-@@ -24,25 +24,73 @@ Mozilla Public License (http://mozilla.o
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- #pragma once
-
- namespace graphite2
- {
-
-+
-+#if defined GRAPHITE2_BUILTINS && (defined __GNUC__ || defined __clang__)
-+
- template<typename T>
- inline unsigned int bit_set_count(T v)
- {
-- v = v - ((v >> 1) & T(~T(0)/3)); // temp
-- v = (v & T(~T(0)/15*3)) + ((v >> 2) & T(~T(0)/15*3)); // temp
-- v = (v + (v >> 4)) & T(~T(0)/255*15); // temp
-- return (T)(v * T(~T(0)/255)) >> (sizeof(T)-1)*8; // count
-+ return __builtin_popcount(v);
- }
-
-+template<>
-+inline unsigned int bit_set_count(int16 v)
-+{
-+ return __builtin_popcount(static_cast<uint16>(v));
-+}
-+
-+template<>
-+inline unsigned int bit_set_count(int8 v)
-+{
-+ return __builtin_popcount(static_cast<uint8>(v));
-+}
-+
-+template<>
-+inline unsigned int bit_set_count(unsigned long v)
-+{
-+ return __builtin_popcountl(v);
-+}
-+
-+template<>
-+inline unsigned int bit_set_count(signed long v)
-+{
-+ return __builtin_popcountl(v);
-+}
-+
-+template<>
-+inline unsigned int bit_set_count(unsigned long long v)
-+{
-+ return __builtin_popcountll(v);
-+}
-+
-+template<>
-+inline unsigned int bit_set_count(signed long long v)
-+{
-+ return __builtin_popcountll(v);
-+}
-+#else
-+
-+template<typename T>
-+inline unsigned int bit_set_count(T v)
-+{
-+ v = v - ((v >> 1) & T(~(0UL)/3)); // temp
-+ v = (v & T(~(0UL)/15*3)) + ((v >> 2) & T(~(0UL)/15*3)); // temp
-+ v = (v + (v >> 4)) & T(~(0UL)/255*15); // temp
-+ return (T)(v * T(~(0UL)/255)) >> (sizeof(T)-1)*8; // count
-+}
-+
-+#endif
-+
-
- template<int S>
- inline unsigned long _mask_over_val(unsigned long v)
- {
- v = _mask_over_val<S/2>(v);
- v |= v >> S*4;
- return v;
- }
-@@ -82,9 +130,17 @@ inline T has_zero(const T x)
-
- template<typename T>
- inline T zero_bytes(const T x, unsigned char n)
- {
- const T t = T(~T(0)/255*n);
- return T((has_zero(x^t) >> 7)*n);
- }
-
-+#if 0
-+inline float float_round(float x, uint32 m)
-+{
-+ *reinterpret_cast<unsigned int *>(&x) &= m;
-+ return *reinterpret_cast<float *>(&x);
- }
-+#endif
-+
-+}
-diff --git a/gfx/graphite2/src/inc/debug.h b/gfx/graphite2/src/inc/debug.h
---- a/gfx/graphite2/src/inc/debug.h
-+++ b/gfx/graphite2/src/inc/debug.h
-@@ -49,31 +49,39 @@ struct objectid
- {
- char name[16];
- objectid(const dslot &) throw();
- objectid(const Segment * const p) throw();
- };
-
-
- json & operator << (json & j, const Position &) throw();
-+json & operator << (json & j, const Rect &) throw();
- json & operator << (json & j, const CharInfo &) throw();
- json & operator << (json & j, const dslot &) throw();
- json & operator << (json & j, const objectid &) throw();
- json & operator << (json & j, const telemetry &) throw();
-
-
-
- inline
- json & operator << (json & j, const Position & p) throw()
- {
- return j << json::flat << json::array << p.x << p.y << json::close;
- }
-
-
- inline
-+json & operator << (json & j, const Rect & p) throw()
-+{
-+ return j << json::flat << json::array << p.bl.x << p.bl.y << p.tr.x << p.tr.y << json::close;
-+}
-+
-+
-+inline
- json & operator << (json & j, const objectid & sid) throw()
- {
- return j << sid.name;
- }
-
-
- } // namespace graphite2
-
-diff --git a/gfx/graphite2/src/inc/json.h b/gfx/graphite2/src/inc/json.h
---- a/gfx/graphite2/src/inc/json.h
-+++ b/gfx/graphite2/src/inc/json.h
-@@ -24,19 +24,21 @@ Mozilla Public License (http://mozilla.o
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- // JSON pretty printer for graphite font debug output logging.
- // Created on: 15 Dec 2011
- // Author: Tim Eves
-
- #pragma once
-+
- #include "inc/Main.h"
- #include <cassert>
--#include <stdio.h>
-+#include <cstdio>
-+#include "inc/List.h"
-
- namespace graphite2 {
-
- class json
- {
- // Prevent copying
- json(const json &);
- json & operator = (const json &);
-@@ -44,31 +46,36 @@ class json
- typedef void (*_context_t)(json &);
- class _null_t {};
-
- FILE * const _stream;
- char _contexts[128], // context stack
- * _context, // current context (top of stack)
- * _flatten; // if !0 points to context above which
- // pretty printed output should occur.
-+ Vector<void *> _env;
-
- void context(const char current) throw();
- void indent(const int d=0) throw();
- void push_context(const char, const char) throw();
- void pop_context() throw();
-
- public:
- class closer;
-
- typedef const char * string;
- typedef double number;
- typedef long signed int integer;
- typedef bool boolean;
- static const _null_t null;
-
-+ void setenv(unsigned int index, void *val) { _env.reserve(index + 1); if (index >= _env.size()) _env.insert(_env.end(), _env.size() - index + 1, 0); _env[index] = val; }
-+ void *getenv(unsigned int index) const { return _env[index]; }
-+ const Vector<void *> &getenvs() const { return _env; }
-+
- static void flat(json &) throw();
- static void close(json &) throw();
- static void object(json &) throw();
- static void array(json &) throw();
- static void item(json &) throw();
-
- json(FILE * stream) throw();
- ~json() throw ();
-diff --git a/gfx/graphite2/src/inc/opcode_table.h b/gfx/graphite2/src/inc/opcode_table.h
---- a/gfx/graphite2/src/inc/opcode_table.h
-+++ b/gfx/graphite2/src/inc/opcode_table.h
-@@ -38,17 +38,17 @@ of the License or (at your option) any l
- // sattrnum - 0 .. 29 (gr_slatJWidth) , 55 (gr_slatUserDefn)
- // attrid - 0 .. silf.numUser() where sattrnum == 55; 0..silf.m_iMaxComp where sattrnum == 15 otherwise 0
- // gattrnum - 0 .. face->getGlyphFaceCache->numAttrs()
- // gmetric - 0 .. 11 (kgmetDescent)
- // featidx - 0 .. face.numFeatures()
- // level - any byte
- static const opcode_t opcode_table[] =
- {
-- {{do2(nop)}, 0, "NOP"},
-+ {{do2(nop)}, 0, "NOP"},
-
- {{do2(push_byte)}, 1, "PUSH_BYTE"}, // number
- {{do2(push_byte_u)}, 1, "PUSH_BYTE_U"}, // number
- {{do2(push_short)}, 2, "PUSH_SHORT"}, // number number
- {{do2(push_short_u)}, 2, "PUSH_SHORT_U"}, // number number
- {{do2(push_long)}, 4, "PUSH_LONG"}, // number number number number
-
- {{do2(add)}, 0, "ADD"},
-@@ -109,12 +109,17 @@ static const opcode_t opcode_table[] =
- {{do2(push_proc_state)}, 1, "PUSH_PROC_STATE"}, // dummy
- {{do2(push_version)}, 0, "PUSH_VERSION"},
- {{do_(put_subs), NILOP}, 5, "PUT_SUBS"}, // slot input_class input_class output_class output_class
- {{NILOP,NILOP}, 0, "PUT_SUBS2"},
- {{NILOP,NILOP}, 0, "PUT_SUBS3"},
- {{do_(put_glyph), NILOP}, 2, "PUT_GLYPH"}, // output_class output_class
- {{do2(push_glyph_attr)}, 3, "PUSH_GLYPH_ATTR"}, // gattrnum gattrnum slot
- {{do2(push_att_to_glyph_attr)}, 3, "PUSH_ATT_TO_GLYPH_ATTR"}, // gattrnum gattrnum slot
-+ {{do2(bor)}, 0, "BITOR"},
-+ {{do2(band)}, 0, "BITAND"},
-+ {{do2(bnot)}, 0, "BITNOT"}, // 0x40
-+ {{do2(setbits)}, 4, "BITSET"},
-+ {{do2(set_feat)}, 2, "SET_FEAT"},
- // private opcodes for internal use only, comes after all other on disk opcodes.
- {{do_(temp_copy), NILOP}, 0, "TEMP_COPY"}
- };
-
-diff --git a/gfx/graphite2/src/inc/opcodes.h b/gfx/graphite2/src/inc/opcodes.h
---- a/gfx/graphite2/src/inc/opcodes.h
-+++ b/gfx/graphite2/src/inc/opcodes.h
-@@ -56,16 +56,17 @@ of the License or (at your option) any l
- // pushed.
- // seg = A reference to the Segment this code is running over.
- // is = The current slot index
- // isb = The original base slot index at the start of this rule
- // isf = The first positioned slot
- // isl = The last positioned slot
- // ip = The current instruction pointer
- // endPos = Position of advance of last cluster
-+// dir = writing system directionality of the font
-
-
- // #define NOT_IMPLEMENTED assert(false)
- #define NOT_IMPLEMENTED
-
- #define binop(op) const int32 a = pop(); *sp = int32(*sp) op a
- #define use_params(n) dp += n
-
-@@ -236,30 +237,34 @@ STARTOP(put_subs_8bit_obs)
- index = seg.findClassIndex(input_class, slot->gid());
- is->setGlyph(&seg, seg.getClassGlyph(output_class, index));
- }
- ENDOP
-
- STARTOP(put_copy)
- declare_params(1);
- const int slot_ref = int8(*param);
-- if (is && (slot_ref ||is != *map))
-+ if (is)
- {
-- int16 *tempUserAttrs = is->userAttrs();
- slotref ref = slotat(slot_ref);
-- if (ref)
-+ if (ref && ref != is)
- {
-- memcpy(tempUserAttrs, ref->userAttrs(), seg.numAttrs() * sizeof(uint16));
-+ int16 *tempUserAttrs = is->userAttrs();
-+ if (is->attachedTo() || is->firstChild()) DIE
- Slot *prev = is->prev();
- Slot *next = is->next();
-- memcpy(is, slotat(slot_ref), sizeof(Slot));
-+ memcpy(tempUserAttrs, ref->userAttrs(), seg.numAttrs() * sizeof(uint16));
-+ memcpy(is, ref, sizeof(Slot));
-+ is->firstChild(NULL);
-+ is->nextSibling(NULL);
- is->userAttrs(tempUserAttrs);
- is->next(next);
- is->prev(prev);
-- is->sibling(NULL);
-+ if (is->attachedTo())
-+ is->attachedTo()->child(is);
- }
- is->markCopied(false);
- is->markDeleted(false);
- }
- ENDOP
-
- STARTOP(insert)
- Slot *newSlot = seg.newSlot();
-@@ -304,24 +309,26 @@ STARTOP(insert)
- {
- newSlot->originate(newSlot->prev()->original());
- newSlot->after(newSlot->prev()->after());
- }
- else
- {
- newSlot->originate(seg.defaultOriginal());
- }
-+ if (is == smap.highwater())
-+ smap.highpassed(false);
- is = newSlot;
- seg.extendLength(1);
- if (map != &smap[-1])
- --map;
- ENDOP
-
- STARTOP(delete_)
-- if (!is) DIE
-+ if (!is || is->isDeleted()) DIE
- is->markDeleted(true);
- if (is->prev())
- is->prev()->next(is->next());
- else
- seg.first(is->next());
-
- if (is->next())
- is->next()->prev(is->prev());
-@@ -380,30 +387,30 @@ STARTOP(attr_set)
- ENDOP
-
- STARTOP(attr_add)
- declare_params(1);
- const attrCode slat = attrCode(uint8(*param));
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, 0);
- is->setAttr(&seg, slat, 0, val + res, smap);
- ENDOP
-
- STARTOP(attr_sub)
- declare_params(1);
- const attrCode slat = attrCode(uint8(*param));
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, 0);
- is->setAttr(&seg, slat, 0, res - val, smap);
- ENDOP
-
- STARTOP(attr_set_slot)
- declare_params(1);
-@@ -422,17 +429,17 @@ STARTOP(iattr_set_slot)
- ENDOP
-
- STARTOP(push_slot_attr)
- declare_params(2);
- const attrCode slat = attrCode(uint8(param[0]));
- const int slot_ref = int8(param[1]);
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- slotref slot = slotat(slot_ref);
- if (slot)
- {
- int res = slot->getAttr(&seg, slat, 0);
- push(res);
- }
-@@ -449,17 +456,17 @@ ENDOP
-
- STARTOP(push_glyph_metric)
- declare_params(3);
- const unsigned int glyph_attr = uint8(param[0]);
- const int slot_ref = int8(param[1]);
- const signed int attr_level = uint8(param[2]);
- slotref slot = slotat(slot_ref);
- if (slot)
-- push(seg.getGlyphMetric(slot, glyph_attr, attr_level));
-+ push(seg.getGlyphMetric(slot, glyph_attr, attr_level, dir));
- ENDOP
-
- STARTOP(push_feat)
- declare_params(2);
- const unsigned int feat = uint8(param[0]);
- const int slot_ref = int8(param[1]);
- slotref slot = slotat(slot_ref);
- if (slot)
-@@ -487,28 +494,28 @@ STARTOP(push_att_to_glyph_metric)
- const unsigned int glyph_attr = uint8(param[0]);
- const int slot_ref = int8(param[1]);
- const signed int attr_level = uint8(param[2]);
- slotref slot = slotat(slot_ref);
- if (slot)
- {
- slotref att = slot->attachedTo();
- if (att) slot = att;
-- push(int32(seg.getGlyphMetric(slot, glyph_attr, attr_level)));
-+ push(int32(seg.getGlyphMetric(slot, glyph_attr, attr_level, dir)));
- }
- ENDOP
-
- STARTOP(push_islot_attr)
- declare_params(3);
- const attrCode slat = attrCode(uint8(param[0]));
- const int slot_ref = int8(param[1]),
- idx = uint8(param[2]);
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- slotref slot = slotat(slot_ref);
- if (slot)
- {
- int res = slot->getAttr(&seg, slat, idx);
- push(res);
- }
-@@ -543,31 +550,31 @@ ENDOP
-
- STARTOP(iattr_add)
- declare_params(2);
- const attrCode slat = attrCode(uint8(param[0]));
- const size_t idx = uint8(param[1]);
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, idx);
- is->setAttr(&seg, slat, idx, val + res, smap);
- ENDOP
-
- STARTOP(iattr_sub)
- declare_params(2);
- const attrCode slat = attrCode(uint8(param[0]));
- const size_t idx = uint8(param[1]);
- const int val = int(pop());
- if ((slat == gr_slatPosX || slat == gr_slatPosY) && (flags & POSITIONED) == 0)
- {
-- seg.positionSlots(0, *smap.begin(), *(smap.end()-1));
-+ seg.positionSlots(0, *smap.begin(), *(smap.end()-1), dir);
- flags |= POSITIONED;
- }
- int res = is->getAttr(&seg, slat, idx);
- is->setAttr(&seg, slat, idx, res - val, smap);
- ENDOP
-
- STARTOP(push_proc_state)
- use_params(1);
-@@ -631,16 +638,50 @@ STARTOP(push_att_to_glyph_attr)
- slotref att = slot->attachedTo();
- if (att) slot = att;
- push(int32(seg.glyphAttr(slot->gid(), glyph_attr)));
- }
- ENDOP
-
- STARTOP(temp_copy)
- slotref newSlot = seg.newSlot();
-- if (!newSlot) DIE;
-+ if (!newSlot || !is) DIE;
- int16 *tempUserAttrs = newSlot->userAttrs();
- memcpy(newSlot, is, sizeof(Slot));
- memcpy(tempUserAttrs, is->userAttrs(), seg.numAttrs() * sizeof(uint16));
- newSlot->userAttrs(tempUserAttrs);
- newSlot->markCopied(true);
- *map = newSlot;
- ENDOP
-+
-+STARTOP(band)
-+ binop(&);
-+ENDOP
-+
-+STARTOP(bor)
-+ binop(|);
-+ENDOP
-+
-+STARTOP(bnot)
-+ *sp = ~*sp;
-+ENDOP
-+
-+STARTOP(setbits)
-+ declare_params(4);
-+ const uint16 m = uint16(param[0]) << 8
-+ | uint8(param[1]);
-+ const uint16 v = uint16(param[2]) << 8
-+ | uint8(param[3]);
-+ *sp = ((*sp) & ~m) | v;
-+ENDOP
-+
-+STARTOP(set_feat)
-+ declare_params(2);
-+ const unsigned int feat = uint8(param[0]);
-+ const int slot_ref = int8(param[1]);
-+ slotref slot = slotat(slot_ref);
-+ if (slot)
-+ {
-+ uint8 fid = seg.charinfo(slot->original())->fid();
-+ seg.setFeature(fid, feat, pop());
-+ }
-+ENDOP
-+
-diff --git a/gfx/graphite2/src/json.cpp b/gfx/graphite2/src/json.cpp
---- a/gfx/graphite2/src/json.cpp
-+++ b/gfx/graphite2/src/json.cpp
-@@ -24,17 +24,18 @@ Mozilla Public License (http://mozilla.o
- License, as published by the Free Software Foundation, either version 2
- of the License or (at your option) any later version.
- */
- // JSON debug logging
- // Author: Tim Eves
-
- #if !defined GRAPHITE2_NTRACING
-
--#include <stdio.h>
-+#include <cstdio>
-+#include <limits>
- #include "inc/json.h"
-
- using namespace graphite2;
-
- namespace
- {
- enum
- {
-@@ -111,16 +112,29 @@ json & json::operator << (json::string s
- const char ctxt = _context[-1] == obj ? *_context == member ? seq : member : seq;
- context(ctxt);
- fprintf(_stream, "\"%s\"", s);
- if (ctxt == member) fputc(' ', _stream);
-
- return *this;
- }
-
--json & json::operator << (json::number f) throw() { context(seq); fprintf(_stream, "%g", f); return *this; }
-+json & json::operator << (json::number f) throw()
-+{
-+ context(seq);
-+ if (std::numeric_limits<json::number>::infinity() == f)
-+ fputs("Infinity", _stream);
-+ else if (-std::numeric_limits<json::number>::infinity() == f)
-+ fputs("-Infinity", _stream);
-+ else if (std::numeric_limits<json::number>::quiet_NaN() == f ||
-+ std::numeric_limits<json::number>::signaling_NaN() == f)
-+ fputs("NaN", _stream);
-+ else
-+ fprintf(_stream, "%g", f);
-+ return *this;
-+}
- json & json::operator << (json::integer d) throw() { context(seq); fprintf(_stream, "%ld", d); return *this; }
- json & json::operator << (long unsigned d) throw() { context(seq); fprintf(_stream, "%ld", d); return *this; }
- json & json::operator << (json::boolean b) throw() { context(seq); fputs(b ? "true" : "false", _stream); return *this; }
- json & json::operator << (json::_null_t) throw() { context(seq); fputs("null",_stream); return *this; }
-
- #endif
-
-diff --git a/gfx/graphite2/src/moz.build b/gfx/graphite2/src/moz.build
---- a/gfx/graphite2/src/moz.build
-+++ b/gfx/graphite2/src/moz.build
-@@ -18,37 +18,40 @@ if CONFIG['GNU_CC']:
- ]
- else:
- UNIFIED_SOURCES += [
- 'call_machine.cpp'
- ]
-
- # This should contain all of the _SOURCES from files.mk, except *_machine.cpp
- UNIFIED_SOURCES += [
-- 'Bidi.cpp',
- 'CachedFace.cpp',
- 'CmapCache.cpp',
- 'Code.cpp',
-+ 'Collider.cpp',
-+ 'Decompressor.cpp',
- 'Face.cpp',
- 'FeatureMap.cpp',
- 'FileFace.cpp',
- 'Font.cpp',
- 'GlyphCache.cpp',
- 'GlyphFace.cpp',
- 'gr_char_info.cpp',
- 'gr_face.cpp',
- 'gr_features.cpp',
- 'gr_font.cpp',
- 'gr_logging.cpp',
- 'gr_segment.cpp',
- 'gr_slot.cpp',
-+ 'Intervals.cpp',
- 'json.cpp',
- 'Justifier.cpp',
- 'NameTable.cpp',
- 'Pass.cpp',
-+ 'Position.cpp',
- 'SegCache.cpp',
- 'SegCacheEntry.cpp',
- 'SegCacheStore.cpp',
- 'Segment.cpp',
- 'Silf.cpp',
- 'Slot.cpp',
- 'Sparse.cpp',
- 'TtfUtil.cpp',
-