プログラミング

UnityのCompute Shaderに線分と三角形の衝突判定をさせる

投稿日:2019年5月11日 更新日:

はじめてCompute Shader使ってみた。でもあまり性能は出ないのでただの失敗の記録です。

まず元のアルゴリズムはコレ。
線分と三角形の当たり判定 – 富士見研究所

で、これをまず三角形の表裏問わず判定するように改造して、
さらにGPUにやらせるにあたって、きっとif文とかが苦手だと思うのでif文を取っ払ってこう書く。 resultが正の値ならHitしてます。

#pragma kernel LineTriangleHit

struct LTHit
{
    float3 lineStart;
    float3 lineEnd;
    float3 triP;
    float3 triQ;
    float3 triR;
    int    result;
};

RWStructuredBuffer<LTHit> LTHits;

[numthreads(8,1,1)]
void LineTriangleHit (uint3 id : SV_DispatchThreadID)
{
    float3 s = LTHits[id.x].lineStart;
    float3 e = LTHits[id.x].lineEnd;
    float3 p = LTHits[id.x].triP;
    float3 q = LTHits[id.x].triQ;
    float3 r = LTHits[id.x].triR;
    float3 l = e - s;
    float3 n = cross(q - p, r - p);
    float  dists = dot(s - p, n);
    float  diste = dot(e - p, n);
    
    float3 crossPoint = dists / (dists - diste) * l + s;
    
    int r0 = -sign(dists * diste);
    int r1 = sign(dot(n, cross(q - p, p - crossPoint)));
    int r2 = sign(dot(n, cross(r - q, q - crossPoint)));
    int r3 = sign(dot(n, cross(p - r, r - crossPoint)));
    
    LTHits[id.x].result = r0 | ((r1 | r2 | r3) ^ (r1 & r2 & r3));
}

一番のポイントは最後のビット演算なんですが、
r1~r3がそれぞれ線分と三角形の交点から各頂点へのベクトルと各辺ベクトルの外積になっていて(正確にはその符号部分)、
交点が三角形の内側にある場合、この外積はすべて同じ方向を向く。
ベクトルが同じ方向を向いてる=符号が同じ、ということなので、3つの変数の符号が一致しているか見るんですよ。

で、変数の符号の一致を確認する。
たとえば2変数なら “if (x*y>0)”みたいにする。
if文が嫌いなのでビット演算にやらせると、 “result = x XOR y”みたいになる。resultが正なら符号が一致している。
これを3変数に拡張すると、”(x OR y OR z) XOR (x AND y AND z)”で取れるのでそのように書く、というわけです。

とりあえずアイデアとしてはそんな感じで、この処理の速さはともかく、データを流し込む処理に時間食ってぜんぜん性能出ないので封印しようかな。って。

-プログラミング

執筆者:

関連記事

GeoJSONで市町村境界をマージして都道府県境界にしたい(その2)

GeoJSONのPolygonをマージしたい第2回です。 前回、純粋な多角形の統合ではなくて、領域が被らない多角形の統合になるのでグラフ問題として解くことができるという説明をしました。 今回はどうやっ …

no image

Google Apps Script (GAS)でBlueskyのbotを作る

Blueskyで何かやりたくなったので、PRTimesのプレスリリースを自動投稿するbotを作ってみました。 Google Apps Scriptを使うと、無料でbotが作れるので便利です。 せっかく …

GeoJSONで市町村境界をマージして都道府県境界にしたい(実践編)

前々回と前回で問題を整理して、ようやく実践編です。 まず実物のリンク貼りましょう。GitHubに上げました。 今回はGo言語で書いてますが、ポイントがいくつかあります。 ちなみに言語としてGoを選択し …

SHOWROOMのガチイベをGoogleスプレッドシートでグラフにしよう

まえがき SHOWROOMのガチイベ、と言えば配信者が獲得ポイント数でランキングされ、最終的にファンの札束で殴り合って勝者を競う札束タワーバトルなわけです。 数日~数週間にわたって配信者同士でランキン …

no image

例え話をしないC言語のポインタの説明

もくじ まえおき 巷では「プログラマーになりたい人に初学者にとって、ポインタという考え方がわけわかめ」という話がよくあります。 そこでいろいろな人が「ポインタは住所だ」とか「変数がハコで」とか手を変え …