You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

91 lines
3.8 KiB

  1. commit a64e5574e40e3e0819c82e35a7e3d2fa65febc73
  2. Author: Willy Tarreau <w@1wt.eu>
  3. Date: Fri Jan 11 19:38:25 2019 +0100
  4. BUG/MAJOR: cache: fix confusion between zero and uninitialized cache key
  5. The cache uses the first 32 bits of the uri's hash as the key to reference
  6. the object in the cache. It makes a special case of the value zero to mean
  7. that the object is not in the cache anymore. The problem is that when an
  8. object hashes as zero, it's still inserted but the eb32_delete() call is
  9. skipped, resulting in the object still being chained in the memory area
  10. while the block has been reclaimed and used for something else. Then when
  11. objects which were chained below it (techically any object since zero is
  12. at the root) are deleted, the walk through the upper object may encounter
  13. corrupted values where valid pointers were expected.
  14. But while this should only happen statically once on 4 billion, the problem
  15. gets worse when the cache-use conditions don't match the cache-store ones,
  16. because cache-store runs with an uninitialized key, which can create objects
  17. that will never be found by the lookup code, or worse, entries with a zero
  18. key preventing eviction of the tree node and resulting in a crash. It's easy
  19. to accidently end up on such a config because the request rules generally
  20. can't be used to decide on the response :
  21. http-request cache-use cache if { path_beg /images }
  22. http-response cache-store cache
  23. In this test, mixing traffic with /images/$RANDOM and /foo/$RANDOM will
  24. result in random keys being inserted, some of them possibly being zero,
  25. and crashes will quickly happen.
  26. The fix consists in 1) always initializing the transaction's cache_hash
  27. to zero, and 2) never storing a response for which the hash has not been
  28. calculated, as indicated by the value zero.
  29. It is worth noting that objects hashing as value zero will never be cached,
  30. but given that there's only one chance among 4 billion that this happens,
  31. this is totally harmless.
  32. This fix must be backported to 1.9 and 1.8.
  33. (cherry picked from commit c9036c00044a8d81561113886ecec9a9ce71bd3b)
  34. Signed-off-by: Willy Tarreau <w@1wt.eu>
  35. (cherry picked from commit 5a6279fcc16da479304bcabc1705e8653f274337)
  36. Signed-off-by: William Lallemand <wlallemand@haproxy.org>
  37. diff --git a/src/cache.c b/src/cache.c
  38. index 667cede3..3d8ed241 100644
  39. --- a/src/cache.c
  40. +++ b/src/cache.c
  41. @@ -400,7 +400,7 @@ enum act_return http_action_store_cache(struct act_rule *rule, struct proxy *px,
  42. struct cache *cache = (struct cache *)rule->arg.act.p[0];
  43. struct shared_context *shctx = shctx_ptr(cache);
  44. struct cache_entry *object;
  45. -
  46. + unsigned int key = *(unsigned int *)txn->cache_hash;
  47. /* Don't cache if the response came from a cache */
  48. if ((obj_type(s->target) == OBJ_TYPE_APPLET) &&
  49. @@ -420,6 +420,10 @@ enum act_return http_action_store_cache(struct act_rule *rule, struct proxy *px,
  50. if (txn->meth != HTTP_METH_GET)
  51. goto out;
  52. + /* cache key was not computed */
  53. + if (!key)
  54. + goto out;
  55. +
  56. /* cache only 200 status code */
  57. if (txn->status != 200)
  58. goto out;
  59. @@ -478,7 +482,7 @@ enum act_return http_action_store_cache(struct act_rule *rule, struct proxy *px,
  60. cache_ctx->first_block = first;
  61. - object->eb.key = (*(unsigned int *)&txn->cache_hash);
  62. + object->eb.key = key;
  63. memcpy(object->hash, txn->cache_hash, sizeof(object->hash));
  64. /* Insert the node later on caching success */
  65. diff --git a/src/proto_http.c b/src/proto_http.c
  66. index 7e4a8351..29a1083a 100644
  67. --- a/src/proto_http.c
  68. +++ b/src/proto_http.c
  69. @@ -8210,6 +8210,7 @@ void http_init_txn(struct stream *s)
  70. txn->flags = 0;
  71. txn->status = -1;
  72. + *(unsigned int *)txn->cache_hash = 0;
  73. txn->cookie_first_date = 0;
  74. txn->cookie_last_date = 0;