En supposant que votre tableau Topic_Tags est unique, cela répond à votre exact question - mais peut ne pas être généralisable à votre problème réel :
SELECT
TopicId
FROM Topic_Tags
JOIN Tags ON
Topic_Tags.TagId = Tags.TagId
WHERE
Tags.Name IN ('A', 'B', 'C', 'D', 'E', 'F')
GROUP BY
TopicId
HAVING
COUNT(*) = 3
AND MAX(Tags.Name) = 'C'
Une solution plus générale serait :
SELECT
*
FROM (
SELECT
TopicId
FROM Topic_Tags
JOIN Tags ON
Topic_Tags.TagId = Tags.TagId
WHERE
Tags.Name IN ('A', 'B', 'C')
GROUP BY
TopicId
HAVING
COUNT(*) = 3
) as GoodTags
LEFT JOIN (
SELECT
TopicId
FROM Topic_Tags
JOIN Tags ON
Topic_Tags.TagId = Tags.TagId
WHERE
Tags.Name = 'D'
OR Tags.Name = 'E'
OR Tags.Name = 'F'
) as BadTags ON
GoodTags.TopicId = BadTags.TopicId
WHERE
BadTags.TopicId IS NULL