mirror of
				https://github.com/esphome/esphome.git
				synced 2025-11-02 16:11:53 +00:00 
			
		
		
		
	Compare commits
	
		
			330 Commits
		
	
	
		
			2024.10.2
			...
			2024.12.0b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					1656ced351 | ||
| 
						 | 
					5dcaf1241f | ||
| 
						 | 
					7aa54b6879 | ||
| 
						 | 
					444e162c92 | ||
| 
						 | 
					bb27eaaf1e | ||
| 
						 | 
					517f659da8 | ||
| 
						 | 
					5a92e24662 | ||
| 
						 | 
					437b236a4d | ||
| 
						 | 
					14eac3dbce | ||
| 
						 | 
					132a096ae7 | ||
| 
						 | 
					440080a753 | ||
| 
						 | 
					f15e3cfb9b | ||
| 
						 | 
					9d000e9abf | ||
| 
						 | 
					97fd7493b5 | ||
| 
						 | 
					4c87658503 | ||
| 
						 | 
					c80e035bd5 | ||
| 
						 | 
					c8ec0bb7ea | ||
| 
						 | 
					86ae1c5931 | ||
| 
						 | 
					d0958f7cf2 | ||
| 
						 | 
					982ce1db72 | ||
| 
						 | 
					f042c6e643 | ||
| 
						 | 
					5fcd26bfe9 | ||
| 
						 | 
					5717d557f5 | ||
| 
						 | 
					3bac45e737 | ||
| 
						 | 
					e623989878 | ||
| 
						 | 
					39cbc6b183 | ||
| 
						 | 
					749a5e3348 | ||
| 
						 | 
					b0e3ac01e8 | ||
| 
						 | 
					58123845ff | ||
| 
						 | 
					bfd75d736c | ||
| 
						 | 
					4e3195b474 | ||
| 
						 | 
					d3a71a1d45 | ||
| 
						 | 
					555bdac604 | ||
| 
						 | 
					acc8d24a32 | ||
| 
						 | 
					f3cc1e541a | ||
| 
						 | 
					ece72c6b18 | ||
| 
						 | 
					4e839d42d0 | ||
| 
						 | 
					d429aa8bb8 | ||
| 
						 | 
					472402745d | ||
| 
						 | 
					016fac2496 | ||
| 
						 | 
					79478cdb8a | ||
| 
						 | 
					dbed74b50d | ||
| 
						 | 
					d00ec7e544 | ||
| 
						 | 
					a37ff2dbd9 | ||
| 
						 | 
					00ddb0a427 | ||
| 
						 | 
					c95887a14a | ||
| 
						 | 
					dc5942a59b | ||
| 
						 | 
					584dbf2668 | ||
| 
						 | 
					9c8976be13 | ||
| 
						 | 
					e08a9cc3a3 | ||
| 
						 | 
					b79a3d6727 | ||
| 
						 | 
					fb96e3588d | ||
| 
						 | 
					edd847ea40 | ||
| 
						 | 
					83d6834e27 | ||
| 
						 | 
					8f69d07061 | ||
| 
						 | 
					30477c764d | ||
| 
						 | 
					217a80a178 | ||
| 
						 | 
					5486b40aab | ||
| 
						 | 
					beb8ab50e2 | ||
| 
						 | 
					7cdf5b55ef | ||
| 
						 | 
					c9b0490305 | ||
| 
						 | 
					d305870284 | ||
| 
						 | 
					ff5004d7db | ||
| 
						 | 
					7aa3a1a1cc | ||
| 
						 | 
					e124151e5c | ||
| 
						 | 
					e229ed0da3 | ||
| 
						 | 
					12cdeca48a | ||
| 
						 | 
					a825ef59d4 | ||
| 
						 | 
					65a5216d17 | ||
| 
						 | 
					567256bd62 | ||
| 
						 | 
					4da57c35d0 | ||
| 
						 | 
					f2e8e655ba | ||
| 
						 | 
					8439232b11 | ||
| 
						 | 
					e6c730ab10 | ||
| 
						 | 
					e49df765d2 | ||
| 
						 | 
					e6da55b925 | ||
| 
						 | 
					c894645747 | ||
| 
						 | 
					2539cba610 | ||
| 
						 | 
					5ddbe5cdba | ||
| 
						 | 
					4c7552eca4 | ||
| 
						 | 
					72bf0086e4 | ||
| 
						 | 
					1b91e0027b | ||
| 
						 | 
					e9851e7eb2 | ||
| 
						 | 
					80fedbc1a5 | ||
| 
						 | 
					a4a71797d9 | ||
| 
						 | 
					4a97064b2c | ||
| 
						 | 
					a3ef2ed7fd | ||
| 
						 | 
					3a8b41daa3 | ||
| 
						 | 
					921be1a17c | ||
| 
						 | 
					e3d673d16c | ||
| 
						 | 
					39f3f795e2 | ||
| 
						 | 
					53691d28a8 | ||
| 
						 | 
					3730b0310b | ||
| 
						 | 
					2b9013699d | ||
| 
						 | 
					be78827274 | ||
| 
						 | 
					cd1ee96606 | ||
| 
						 | 
					2fa8d907b3 | ||
| 
						 | 
					4c383906c4 | ||
| 
						 | 
					bdc6302ea1 | ||
| 
						 | 
					31c13e4c16 | ||
| 
						 | 
					6b59f55a50 | ||
| 
						 | 
					e6bd2238ce | ||
| 
						 | 
					2d4688a206 | ||
| 
						 | 
					536bcab5de | ||
| 
						 | 
					1c2d2bce5a | ||
| 
						 | 
					2eac8b6c46 | ||
| 
						 | 
					6e50e2aa65 | ||
| 
						 | 
					841d278224 | ||
| 
						 | 
					11076e4614 | ||
| 
						 | 
					72df3d1606 | ||
| 
						 | 
					ae6736311a | ||
| 
						 | 
					c0dcecc465 | ||
| 
						 | 
					d9d368d38e | ||
| 
						 | 
					a70cee1dc1 | ||
| 
						 | 
					f4766ab74f | ||
| 
						 | 
					4fbf41472a | ||
| 
						 | 
					6ee02c47c2 | ||
| 
						 | 
					140d77061b | ||
| 
						 | 
					d6f4f05090 | ||
| 
						 | 
					bdb91112ea | ||
| 
						 | 
					b027b6a711 | ||
| 
						 | 
					89ecfc2004 | ||
| 
						 | 
					cf835d1580 | ||
| 
						 | 
					17a09cd221 | ||
| 
						 | 
					1bd2d41ffd | ||
| 
						 | 
					aa6cea6f7e | ||
| 
						 | 
					ebf895990b | ||
| 
						 | 
					46a435f5f2 | ||
| 
						 | 
					6c548a1596 | ||
| 
						 | 
					7f75f2135d | ||
| 
						 | 
					c49f7293fe | ||
| 
						 | 
					71496574e9 | ||
| 
						 | 
					b95b4a0694 | ||
| 
						 | 
					59653ec785 | ||
| 
						 | 
					e02f3cdac7 | ||
| 
						 | 
					d4d630823c | ||
| 
						 | 
					9fc1377b44 | ||
| 
						 | 
					e3e3d92347 | ||
| 
						 | 
					13077095c2 | ||
| 
						 | 
					4001d82ca2 | ||
| 
						 | 
					4936ca1700 | ||
| 
						 | 
					2ecd5cff07 | ||
| 
						 | 
					dea297c8d7 | ||
| 
						 | 
					ef7c5c6055 | ||
| 
						 | 
					ee3cfb2b76 | ||
| 
						 | 
					2cc2a2153b | ||
| 
						 | 
					e51f3d9498 | ||
| 
						 | 
					1c1f3f7c55 | ||
| 
						 | 
					ea424b0699 | ||
| 
						 | 
					489d0d20d2 | ||
| 
						 | 
					f04e3de7b8 | ||
| 
						 | 
					a0693060e4 | ||
| 
						 | 
					888b237964 | ||
| 
						 | 
					122ff731ef | ||
| 
						 | 
					3232866dc3 | ||
| 
						 | 
					ccf2854b61 | ||
| 
						 | 
					03ae6b2c1b | ||
| 
						 | 
					6bcbbcce02 | ||
| 
						 | 
					fbb9967117 | ||
| 
						 | 
					6d4f787f67 | ||
| 
						 | 
					5e27a8df1f | ||
| 
						 | 
					846b091aac | ||
| 
						 | 
					372d68a177 | ||
| 
						 | 
					4fc19902ab | ||
| 
						 | 
					9a7d5dcad8 | ||
| 
						 | 
					ef78c404dd | ||
| 
						 | 
					c857f98557 | ||
| 
						 | 
					01a24de3a8 | ||
| 
						 | 
					ae46dcef7e | ||
| 
						 | 
					872b8ee753 | ||
| 
						 | 
					eb8a2326ad | ||
| 
						 | 
					cf63d627fe | ||
| 
						 | 
					49e9c43339 | ||
| 
						 | 
					f1dc9537ff | ||
| 
						 | 
					1ad535d030 | ||
| 
						 | 
					1ed27b7cc0 | ||
| 
						 | 
					585586780b | ||
| 
						 | 
					50aeefc662 | ||
| 
						 | 
					6e41c22e9d | ||
| 
						 | 
					e81191ebd2 | ||
| 
						 | 
					b29c119408 | ||
| 
						 | 
					e819185de1 | ||
| 
						 | 
					00465f4a6f | ||
| 
						 | 
					f4dc11477f | ||
| 
						 | 
					754352b4d7 | ||
| 
						 | 
					67a4e56fcf | ||
| 
						 | 
					9bc7b74d01 | ||
| 
						 | 
					15bfc4c91f | ||
| 
						 | 
					a0159a2746 | ||
| 
						 | 
					44545a18a0 | ||
| 
						 | 
					0b51ec2c88 | ||
| 
						 | 
					5e62c489b0 | ||
| 
						 | 
					d015088855 | ||
| 
						 | 
					39c889e662 | ||
| 
						 | 
					c7c8711c9c | ||
| 
						 | 
					0a92405f2d | ||
| 
						 | 
					b4b6b75e84 | ||
| 
						 | 
					a2cab960a9 | ||
| 
						 | 
					1f7f03f563 | ||
| 
						 | 
					80226694d5 | ||
| 
						 | 
					053465d3f6 | ||
| 
						 | 
					7d75c9157b | ||
| 
						 | 
					b367c01b4b | ||
| 
						 | 
					e6a1254e65 | ||
| 
						 | 
					1e80c4807e | ||
| 
						 | 
					928b39f495 | ||
| 
						 | 
					58d028ac13 | ||
| 
						 | 
					a2dccc4730 | ||
| 
						 | 
					ffee2f0e88 | ||
| 
						 | 
					d885d65c9b | ||
| 
						 | 
					c35240ca32 | ||
| 
						 | 
					7c00c5db70 | ||
| 
						 | 
					335faf858b | ||
| 
						 | 
					1829e68730 | ||
| 
						 | 
					b8eadb2ba5 | ||
| 
						 | 
					551ea37882 | ||
| 
						 | 
					3a25eaca3f | ||
| 
						 | 
					e85cbf26f8 | ||
| 
						 | 
					2ec17eed58 | ||
| 
						 | 
					2f77d31690 | ||
| 
						 | 
					3f123d7542 | ||
| 
						 | 
					d189cc1fbe | ||
| 
						 | 
					c0658ffe2c | ||
| 
						 | 
					248b0bc378 | ||
| 
						 | 
					80b4c26481 | ||
| 
						 | 
					5bb4d042e4 | ||
| 
						 | 
					dcc537d0d4 | ||
| 
						 | 
					2dca3d79e4 | ||
| 
						 | 
					01497c891d | ||
| 
						 | 
					77bb46ff3b | ||
| 
						 | 
					cefbfb75bd | ||
| 
						 | 
					749b942132 | ||
| 
						 | 
					a043022444 | ||
| 
						 | 
					8b7e061f3a | ||
| 
						 | 
					74ea1b60e3 | ||
| 
						 | 
					5a2fed3569 | ||
| 
						 | 
					e85157db4b | ||
| 
						 | 
					d3563e4e97 | ||
| 
						 | 
					765579dabb | ||
| 
						 | 
					6afd004ec5 | ||
| 
						 | 
					ee3ee3a63b | ||
| 
						 | 
					aae2ee2ecb | ||
| 
						 | 
					bac6880a1e | ||
| 
						 | 
					0982ab58ac | ||
| 
						 | 
					38dd566e0c | ||
| 
						 | 
					71e1e3b5f8 | ||
| 
						 | 
					abbd7faa64 | ||
| 
						 | 
					aa0e155e22 | ||
| 
						 | 
					0dab280440 | ||
| 
						 | 
					90b076eccd | ||
| 
						 | 
					444c0fc67f | ||
| 
						 | 
					302ba2874e | ||
| 
						 | 
					df750d0d11 | ||
| 
						 | 
					63e4d4b493 | ||
| 
						 | 
					88627095fb | ||
| 
						 | 
					858d97ccef | ||
| 
						 | 
					22f30d42a6 | ||
| 
						 | 
					1e2497748d | ||
| 
						 | 
					34de2bbe99 | ||
| 
						 | 
					21cb941bbe | ||
| 
						 | 
					33fdbbe30c | ||
| 
						 | 
					09f9d91577 | ||
| 
						 | 
					34a8eaddb2 | ||
| 
						 | 
					7dbda12008 | ||
| 
						 | 
					4101d5dad1 | ||
| 
						 | 
					c20e1975d1 | ||
| 
						 | 
					4fa3c6915c | ||
| 
						 | 
					ca5c73d170 | ||
| 
						 | 
					5b5c2fe71b | ||
| 
						 | 
					9acc21e81a | ||
| 
						 | 
					bff0e81ed3 | ||
| 
						 | 
					2feffddc55 | ||
| 
						 | 
					4289e00ad0 | ||
| 
						 | 
					574ee404d2 | ||
| 
						 | 
					fdebf04196 | ||
| 
						 | 
					dd8d25e43f | ||
| 
						 | 
					68844c4869 | ||
| 
						 | 
					7c0543862a | ||
| 
						 | 
					a932ca2f64 | ||
| 
						 | 
					2597975ae0 | ||
| 
						 | 
					6330177d24 | ||
| 
						 | 
					3ac730fb2f | ||
| 
						 | 
					ff48f53989 | ||
| 
						 | 
					8bb4316956 | ||
| 
						 | 
					40cdb778f5 | ||
| 
						 | 
					dc42427c60 | ||
| 
						 | 
					40ad6befa8 | ||
| 
						 | 
					612e2c1644 | ||
| 
						 | 
					c8d0cde329 | ||
| 
						 | 
					5e8794175d | ||
| 
						 | 
					657527655d | ||
| 
						 | 
					f7543a7b8d | ||
| 
						 | 
					43a020641b | ||
| 
						 | 
					c019ff34bc | ||
| 
						 | 
					ef6ccddc0d | ||
| 
						 | 
					8bbe4efded | ||
| 
						 | 
					f490585f66 | ||
| 
						 | 
					fcfc76b01b | ||
| 
						 | 
					5ad68e926d | ||
| 
						 | 
					56fa6fef85 | ||
| 
						 | 
					c9e5919739 | ||
| 
						 | 
					0451b31f9e | ||
| 
						 | 
					1c845e0ff8 | ||
| 
						 | 
					22478ffb0f | ||
| 
						 | 
					c38cc128db | ||
| 
						 | 
					fa01149771 | ||
| 
						 | 
					254522dd93 | ||
| 
						 | 
					6a86d92781 | ||
| 
						 | 
					b274d6901a | ||
| 
						 | 
					3ef31e55ca | ||
| 
						 | 
					fb002ac3b0 | ||
| 
						 | 
					de943908bd | ||
| 
						 | 
					b0a25872da | ||
| 
						 | 
					403d450f47 | ||
| 
						 | 
					2cca26ada4 | ||
| 
						 | 
					312799babf | ||
| 
						 | 
					5bc5a9dcb6 | ||
| 
						 | 
					39e922580a | ||
| 
						 | 
					77d0bfc4bb | ||
| 
						 | 
					654cee6f83 | ||
| 
						 | 
					cf14c02b8a | ||
| 
						 | 
					42f6095960 | ||
| 
						 | 
					f224984858 | ||
| 
						 | 
					efe4c5e3bc | ||
| 
						 | 
					cedb671f07 | ||
| 
						 | 
					c18bd3ac81 | ||
| 
						 | 
					b08432bd0d | ||
| 
						 | 
					4bac9707fe | ||
| 
						 | 
					7e16cda949 | ||
| 
						 | 
					1c05f5af03 | 
							
								
								
									
										36
									
								
								.clang-tidy
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								.clang-tidy
									
									
									
									
									
								
							@@ -7,28 +7,39 @@ Checks: >-
 | 
			
		||||
  -boost-*,
 | 
			
		||||
  -bugprone-easily-swappable-parameters,
 | 
			
		||||
  -bugprone-implicit-widening-of-multiplication-result,
 | 
			
		||||
  -bugprone-multi-level-implicit-pointer-conversion,
 | 
			
		||||
  -bugprone-narrowing-conversions,
 | 
			
		||||
  -bugprone-signed-char-misuse,
 | 
			
		||||
  -bugprone-switch-missing-default-case,
 | 
			
		||||
  -cert-dcl50-cpp,
 | 
			
		||||
  -cert-err33-c,
 | 
			
		||||
  -cert-err58-cpp,
 | 
			
		||||
  -cert-oop57-cpp,
 | 
			
		||||
  -cert-str34-c,
 | 
			
		||||
  -clang-analyzer-optin.core.EnumCastOutOfRange,
 | 
			
		||||
  -clang-analyzer-optin.cplusplus.UninitializedObject,
 | 
			
		||||
  -clang-analyzer-osx.*,
 | 
			
		||||
  -clang-diagnostic-delete-abstract-non-virtual-dtor,
 | 
			
		||||
  -clang-diagnostic-delete-non-abstract-non-virtual-dtor,
 | 
			
		||||
  -clang-diagnostic-deprecated-declarations,
 | 
			
		||||
  -clang-diagnostic-ignored-optimization-argument,
 | 
			
		||||
  -clang-diagnostic-missing-field-initializers,
 | 
			
		||||
  -clang-diagnostic-shadow-field,
 | 
			
		||||
  -clang-diagnostic-unused-const-variable,
 | 
			
		||||
  -clang-diagnostic-unused-parameter,
 | 
			
		||||
  -clang-diagnostic-vla-cxx-extension,
 | 
			
		||||
  -concurrency-*,
 | 
			
		||||
  -cppcoreguidelines-avoid-c-arrays,
 | 
			
		||||
  -cppcoreguidelines-avoid-const-or-ref-data-members,
 | 
			
		||||
  -cppcoreguidelines-avoid-do-while,
 | 
			
		||||
  -cppcoreguidelines-avoid-magic-numbers,
 | 
			
		||||
  -cppcoreguidelines-init-variables,
 | 
			
		||||
  -cppcoreguidelines-macro-to-enum,
 | 
			
		||||
  -cppcoreguidelines-macro-usage,
 | 
			
		||||
  -cppcoreguidelines-missing-std-forward,
 | 
			
		||||
  -cppcoreguidelines-narrowing-conversions,
 | 
			
		||||
  -cppcoreguidelines-non-private-member-variables-in-classes,
 | 
			
		||||
  -cppcoreguidelines-owning-memory,
 | 
			
		||||
  -cppcoreguidelines-prefer-member-initializer,
 | 
			
		||||
  -cppcoreguidelines-pro-bounds-array-to-pointer-decay,
 | 
			
		||||
  -cppcoreguidelines-pro-bounds-constant-array-index,
 | 
			
		||||
@@ -40,7 +51,9 @@ Checks: >-
 | 
			
		||||
  -cppcoreguidelines-pro-type-static-cast-downcast,
 | 
			
		||||
  -cppcoreguidelines-pro-type-union-access,
 | 
			
		||||
  -cppcoreguidelines-pro-type-vararg,
 | 
			
		||||
  -cppcoreguidelines-rvalue-reference-param-not-moved,
 | 
			
		||||
  -cppcoreguidelines-special-member-functions,
 | 
			
		||||
  -cppcoreguidelines-use-default-member-init,
 | 
			
		||||
  -cppcoreguidelines-virtual-class-destructor,
 | 
			
		||||
  -fuchsia-multiple-inheritance,
 | 
			
		||||
  -fuchsia-overloaded-operator,
 | 
			
		||||
@@ -60,20 +73,32 @@ Checks: >-
 | 
			
		||||
  -llvm-include-order,
 | 
			
		||||
  -llvm-qualified-auto,
 | 
			
		||||
  -llvmlibc-*,
 | 
			
		||||
  -misc-non-private-member-variables-in-classes,
 | 
			
		||||
  -misc-const-correctness,
 | 
			
		||||
  -misc-include-cleaner,
 | 
			
		||||
  -misc-no-recursion,
 | 
			
		||||
  -misc-non-private-member-variables-in-classes,
 | 
			
		||||
  -misc-unused-parameters,
 | 
			
		||||
  -modernize-avoid-c-arrays,
 | 
			
		||||
  -misc-use-anonymous-namespace,
 | 
			
		||||
  -modernize-avoid-bind,
 | 
			
		||||
  -modernize-avoid-c-arrays,
 | 
			
		||||
  -modernize-concat-nested-namespaces,
 | 
			
		||||
  -modernize-macro-to-enum,
 | 
			
		||||
  -modernize-return-braced-init-list,
 | 
			
		||||
  -modernize-type-traits,
 | 
			
		||||
  -modernize-use-auto,
 | 
			
		||||
  -modernize-use-constraints,
 | 
			
		||||
  -modernize-use-default-member-init,
 | 
			
		||||
  -modernize-use-equals-default,
 | 
			
		||||
  -modernize-use-trailing-return-type,
 | 
			
		||||
  -modernize-use-nodiscard,
 | 
			
		||||
  -modernize-use-nullptr,
 | 
			
		||||
  -modernize-use-nodiscard,
 | 
			
		||||
  -modernize-use-nullptr,
 | 
			
		||||
  -modernize-use-trailing-return-type,
 | 
			
		||||
  -mpi-*,
 | 
			
		||||
  -objc-*,
 | 
			
		||||
  -performance-enum-size,
 | 
			
		||||
  -readability-avoid-nested-conditional-operator,
 | 
			
		||||
  -readability-container-contains,
 | 
			
		||||
  -readability-container-data-pointer,
 | 
			
		||||
  -readability-convert-member-functions-to-static,
 | 
			
		||||
  -readability-else-after-return,
 | 
			
		||||
@@ -82,11 +107,14 @@ Checks: >-
 | 
			
		||||
  -readability-isolate-declaration,
 | 
			
		||||
  -readability-magic-numbers,
 | 
			
		||||
  -readability-make-member-function-const,
 | 
			
		||||
  -readability-named-parameter,
 | 
			
		||||
  -readability-redundant-casting,
 | 
			
		||||
  -readability-redundant-inline-specifier,
 | 
			
		||||
  -readability-redundant-member-init,
 | 
			
		||||
  -readability-redundant-string-init,
 | 
			
		||||
  -readability-uppercase-literal-suffix,
 | 
			
		||||
  -readability-use-anyofallof,
 | 
			
		||||
WarningsAsErrors: '*'
 | 
			
		||||
AnalyzeTemporaryDtors: false
 | 
			
		||||
FormatStyle:     google
 | 
			
		||||
CheckOptions:
 | 
			
		||||
  - key:             google-readability-function-size.StatementThreshold
 | 
			
		||||
 
 | 
			
		||||
@@ -75,6 +75,9 @@ target/
 | 
			
		||||
# pyenv
 | 
			
		||||
.python-version
 | 
			
		||||
 | 
			
		||||
# asdf
 | 
			
		||||
.tool-versions
 | 
			
		||||
 | 
			
		||||
# celery beat schedule file
 | 
			
		||||
celerybeat-schedule
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										15
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							@@ -7,11 +7,16 @@
 | 
			
		||||
- [ ] Bugfix (non-breaking change which fixes an issue)
 | 
			
		||||
- [ ] New feature (non-breaking change which adds functionality)
 | 
			
		||||
- [ ] Breaking change (fix or feature that would cause existing functionality to not work as expected)
 | 
			
		||||
- [ ] Code quality improvements to existing code or addition of tests
 | 
			
		||||
- [ ] Other
 | 
			
		||||
 | 
			
		||||
**Related issue or feature (if applicable):** fixes <link to issue>
 | 
			
		||||
**Related issue or feature (if applicable):**
 | 
			
		||||
 | 
			
		||||
**Pull request in [esphome-docs](https://github.com/esphome/esphome-docs) with documentation (if applicable):** esphome/esphome-docs#<esphome-docs PR number goes here>
 | 
			
		||||
- fixes <link to issue>
 | 
			
		||||
 | 
			
		||||
**Pull request in [esphome-docs](https://github.com/esphome/esphome-docs) with documentation (if applicable):**
 | 
			
		||||
 | 
			
		||||
- esphome/esphome-docs#<esphome-docs PR number goes here>
 | 
			
		||||
 | 
			
		||||
## Test Environment
 | 
			
		||||
 | 
			
		||||
@@ -23,12 +28,6 @@
 | 
			
		||||
- [ ] RTL87xx
 | 
			
		||||
 | 
			
		||||
## Example entry for `config.yaml`:
 | 
			
		||||
<!--
 | 
			
		||||
  Supplying a configuration snippet, makes it easier for a maintainer to test
 | 
			
		||||
  your PR. Furthermore, for new integrations, it gives an impression of how
 | 
			
		||||
  the configuration would look like.
 | 
			
		||||
  Note: Remove this section if this PR does not have an example entry.
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
```yaml
 | 
			
		||||
# Example config.yaml
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								.github/actions/build-image/action.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/actions/build-image/action.yaml
									
									
									
									
										vendored
									
									
								
							@@ -46,7 +46,7 @@ runs:
 | 
			
		||||
 | 
			
		||||
    - name: Build and push to ghcr by digest
 | 
			
		||||
      id: build-ghcr
 | 
			
		||||
      uses: docker/build-push-action@v6.9.0
 | 
			
		||||
      uses: docker/build-push-action@v6.10.0
 | 
			
		||||
      env:
 | 
			
		||||
        DOCKER_BUILD_SUMMARY: false
 | 
			
		||||
        DOCKER_BUILD_RECORD_UPLOAD: false
 | 
			
		||||
@@ -72,7 +72,7 @@ runs:
 | 
			
		||||
 | 
			
		||||
    - name: Build and push to dockerhub by digest
 | 
			
		||||
      id: build-dockerhub
 | 
			
		||||
      uses: docker/build-push-action@v6.9.0
 | 
			
		||||
      uses: docker/build-push-action@v6.10.0
 | 
			
		||||
      env:
 | 
			
		||||
        DOCKER_BUILD_SUMMARY: false
 | 
			
		||||
        DOCKER_BUILD_RECORD_UPLOAD: false
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								.github/actions/restore-python/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/actions/restore-python/action.yml
									
									
									
									
										vendored
									
									
								
							@@ -17,12 +17,12 @@ runs:
 | 
			
		||||
  steps:
 | 
			
		||||
    - name: Set up Python ${{ inputs.python-version }}
 | 
			
		||||
      id: python
 | 
			
		||||
      uses: actions/setup-python@v5.2.0
 | 
			
		||||
      uses: actions/setup-python@v5.3.0
 | 
			
		||||
      with:
 | 
			
		||||
        python-version: ${{ inputs.python-version }}
 | 
			
		||||
    - name: Restore Python virtual environment
 | 
			
		||||
      id: cache-venv
 | 
			
		||||
      uses: actions/cache/restore@v4.1.1
 | 
			
		||||
      uses: actions/cache/restore@v4.2.0
 | 
			
		||||
      with:
 | 
			
		||||
        path: venv
 | 
			
		||||
        # yamllint disable-line rule:line-length
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								.github/workflows/ci-api-proto.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci-api-proto.yml
									
									
									
									
										vendored
									
									
								
							@@ -23,7 +23,7 @@ jobs:
 | 
			
		||||
      - name: Checkout
 | 
			
		||||
        uses: actions/checkout@v4.1.7
 | 
			
		||||
      - name: Set up Python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: "3.11"
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								.github/workflows/ci-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci-docker.yml
									
									
									
									
										vendored
									
									
								
							@@ -42,7 +42,7 @@ jobs:
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4.1.7
 | 
			
		||||
      - name: Set up Python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: "3.9"
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										41
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							@@ -30,7 +30,7 @@ concurrency:
 | 
			
		||||
jobs:
 | 
			
		||||
  common:
 | 
			
		||||
    name: Create common environment
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    outputs:
 | 
			
		||||
      cache-key: ${{ steps.cache-key.outputs.key }}
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -41,12 +41,12 @@ jobs:
 | 
			
		||||
        run: echo key="${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}" >> $GITHUB_OUTPUT
 | 
			
		||||
      - name: Set up Python ${{ env.DEFAULT_PYTHON }}
 | 
			
		||||
        id: python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: ${{ env.DEFAULT_PYTHON }}
 | 
			
		||||
      - name: Restore Python virtual environment
 | 
			
		||||
        id: cache-venv
 | 
			
		||||
        uses: actions/cache@v4.1.1
 | 
			
		||||
        uses: actions/cache@v4.2.0
 | 
			
		||||
        with:
 | 
			
		||||
          path: venv
 | 
			
		||||
          # yamllint disable-line rule:line-length
 | 
			
		||||
@@ -62,7 +62,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  black:
 | 
			
		||||
    name: Check black
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -83,7 +83,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  flake8:
 | 
			
		||||
    name: Check flake8
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -104,7 +104,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  pylint:
 | 
			
		||||
    name: Check pylint
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -125,7 +125,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  pyupgrade:
 | 
			
		||||
    name: Check pyupgrade
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -146,7 +146,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  ci-custom:
 | 
			
		||||
    name: Run script/ci-custom
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -219,13 +219,13 @@ jobs:
 | 
			
		||||
          . venv/bin/activate
 | 
			
		||||
          pytest -vv --cov-report=xml --tb=native tests
 | 
			
		||||
      - name: Upload coverage to Codecov
 | 
			
		||||
        uses: codecov/codecov-action@v4
 | 
			
		||||
        uses: codecov/codecov-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
			
		||||
 | 
			
		||||
  clang-format:
 | 
			
		||||
    name: Check clang-format
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
@@ -251,7 +251,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  clang-tidy:
 | 
			
		||||
    name: ${{ matrix.name }}
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
      - black
 | 
			
		||||
@@ -302,23 +302,18 @@ jobs:
 | 
			
		||||
 | 
			
		||||
      - name: Cache platformio
 | 
			
		||||
        if: github.ref == 'refs/heads/dev'
 | 
			
		||||
        uses: actions/cache@v4.1.1
 | 
			
		||||
        uses: actions/cache@v4.2.0
 | 
			
		||||
        with:
 | 
			
		||||
          path: ~/.platformio
 | 
			
		||||
          key: platformio-${{ matrix.pio_cache_key }}
 | 
			
		||||
 | 
			
		||||
      - name: Cache platformio
 | 
			
		||||
        if: github.ref != 'refs/heads/dev'
 | 
			
		||||
        uses: actions/cache/restore@v4.1.1
 | 
			
		||||
        uses: actions/cache/restore@v4.2.0
 | 
			
		||||
        with:
 | 
			
		||||
          path: ~/.platformio
 | 
			
		||||
          key: platformio-${{ matrix.pio_cache_key }}
 | 
			
		||||
 | 
			
		||||
      - name: Install clang-tidy
 | 
			
		||||
        run: |
 | 
			
		||||
          sudo apt-get update
 | 
			
		||||
          sudo apt-get install clang-tidy-14
 | 
			
		||||
 | 
			
		||||
      - name: Register problem matchers
 | 
			
		||||
        run: |
 | 
			
		||||
          echo "::add-matcher::.github/workflows/matchers/gcc.json"
 | 
			
		||||
@@ -345,7 +340,7 @@ jobs:
 | 
			
		||||
        if: always()
 | 
			
		||||
 | 
			
		||||
  list-components:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    if: github.event_name == 'pull_request'
 | 
			
		||||
@@ -387,7 +382,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  test-build-components:
 | 
			
		||||
    name: Component test ${{ matrix.file }}
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
      - list-components
 | 
			
		||||
@@ -421,7 +416,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  test-build-components-splitter:
 | 
			
		||||
    name: Split components for testing into 20 groups maximum
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
      - list-components
 | 
			
		||||
@@ -439,7 +434,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  test-build-components-split:
 | 
			
		||||
    name: Test split components
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
      - list-components
 | 
			
		||||
@@ -483,7 +478,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
  ci-status:
 | 
			
		||||
    name: CI Status
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
      - black
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										8
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							@@ -53,7 +53,7 @@ jobs:
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4.1.7
 | 
			
		||||
      - name: Set up Python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: "3.x"
 | 
			
		||||
      - name: Set up python environment
 | 
			
		||||
@@ -65,7 +65,7 @@ jobs:
 | 
			
		||||
          pip3 install build
 | 
			
		||||
          python3 -m build
 | 
			
		||||
      - name: Publish
 | 
			
		||||
        uses: pypa/gh-action-pypi-publish@v1.10.3
 | 
			
		||||
        uses: pypa/gh-action-pypi-publish@v1.12.3
 | 
			
		||||
 | 
			
		||||
  deploy-docker:
 | 
			
		||||
    name: Build ESPHome ${{ matrix.platform }}
 | 
			
		||||
@@ -85,7 +85,7 @@ jobs:
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4.1.7
 | 
			
		||||
      - name: Set up Python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: "3.9"
 | 
			
		||||
 | 
			
		||||
@@ -141,7 +141,7 @@ jobs:
 | 
			
		||||
          echo name=$(cat /tmp/platform) >> $GITHUB_OUTPUT
 | 
			
		||||
 | 
			
		||||
      - name: Upload digests
 | 
			
		||||
        uses: actions/upload-artifact@v4.4.2
 | 
			
		||||
        uses: actions/upload-artifact@v4.4.3
 | 
			
		||||
        with:
 | 
			
		||||
          name: digests-${{ steps.sanitize.outputs.name }}
 | 
			
		||||
          path: /tmp/digests
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								.github/workflows/sync-device-classes.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/sync-device-classes.yml
									
									
									
									
										vendored
									
									
								
							@@ -22,7 +22,7 @@ jobs:
 | 
			
		||||
          path: lib/home-assistant
 | 
			
		||||
 | 
			
		||||
      - name: Setup Python
 | 
			
		||||
        uses: actions/setup-python@v5.2.0
 | 
			
		||||
        uses: actions/setup-python@v5.3.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: 3.12
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -75,6 +75,9 @@ cov.xml
 | 
			
		||||
# pyenv
 | 
			
		||||
.python-version
 | 
			
		||||
 | 
			
		||||
# asdf
 | 
			
		||||
.tool-versions
 | 
			
		||||
 | 
			
		||||
# Environments
 | 
			
		||||
.env
 | 
			
		||||
.venv
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										17
									
								
								CODEOWNERS
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								CODEOWNERS
									
									
									
									
									
								
							@@ -48,7 +48,9 @@ esphome/components/at581x/* @X-Ryl669
 | 
			
		||||
esphome/components/atc_mithermometer/* @ahpohl
 | 
			
		||||
esphome/components/atm90e26/* @danieltwagner
 | 
			
		||||
esphome/components/atm90e32/* @circuitsetup @descipher
 | 
			
		||||
esphome/components/audio/* @kahrendt
 | 
			
		||||
esphome/components/audio_dac/* @kbx81
 | 
			
		||||
esphome/components/axs15231/* @clydebarrow
 | 
			
		||||
esphome/components/b_parasite/* @rbaron
 | 
			
		||||
esphome/components/ballu/* @bazuchan
 | 
			
		||||
esphome/components/bang_bang/* @OttoWinter
 | 
			
		||||
@@ -83,6 +85,7 @@ esphome/components/bmp581/* @kahrendt
 | 
			
		||||
esphome/components/bp1658cj/* @Cossid
 | 
			
		||||
esphome/components/bp5758d/* @Cossid
 | 
			
		||||
esphome/components/button/* @esphome/core
 | 
			
		||||
esphome/components/bytebuffer/* @clydebarrow
 | 
			
		||||
esphome/components/canbus/* @danielschramm @mvturnho
 | 
			
		||||
esphome/components/cap1188/* @mreditor97
 | 
			
		||||
esphome/components/captive_portal/* @OttoWinter
 | 
			
		||||
@@ -128,6 +131,7 @@ esphome/components/ens160_base/* @latonita @vincentscode
 | 
			
		||||
esphome/components/ens160_i2c/* @latonita
 | 
			
		||||
esphome/components/ens160_spi/* @latonita
 | 
			
		||||
esphome/components/ens210/* @itn3rd77
 | 
			
		||||
esphome/components/es8311/* @kahrendt @kroimon
 | 
			
		||||
esphome/components/esp32/* @esphome/core
 | 
			
		||||
esphome/components/esp32_ble/* @Rapsssito @jesserockz
 | 
			
		||||
esphome/components/esp32_ble_client/* @jesserockz
 | 
			
		||||
@@ -175,6 +179,7 @@ esphome/components/haier/text_sensor/* @paveldn
 | 
			
		||||
esphome/components/havells_solar/* @sourabhjaiswal
 | 
			
		||||
esphome/components/hbridge/fan/* @WeekendWarrior
 | 
			
		||||
esphome/components/hbridge/light/* @DotNetDann
 | 
			
		||||
esphome/components/hbridge/switch/* @dwmw2
 | 
			
		||||
esphome/components/he60r/* @clydebarrow
 | 
			
		||||
esphome/components/heatpumpir/* @rob-deutsch
 | 
			
		||||
esphome/components/hitachi_ac424/* @sourabhjaiswal
 | 
			
		||||
@@ -196,10 +201,11 @@ esphome/components/htu31d/* @betterengineering
 | 
			
		||||
esphome/components/hydreon_rgxx/* @functionpointer
 | 
			
		||||
esphome/components/hyt271/* @Philippe12
 | 
			
		||||
esphome/components/i2c/* @esphome/core
 | 
			
		||||
esphome/components/i2c_device/* @gabest11
 | 
			
		||||
esphome/components/i2s_audio/* @jesserockz
 | 
			
		||||
esphome/components/i2s_audio/media_player/* @jesserockz
 | 
			
		||||
esphome/components/i2s_audio/microphone/* @jesserockz
 | 
			
		||||
esphome/components/i2s_audio/speaker/* @jesserockz
 | 
			
		||||
esphome/components/i2s_audio/speaker/* @jesserockz @kahrendt
 | 
			
		||||
esphome/components/iaqcore/* @yozik04
 | 
			
		||||
esphome/components/ili9xxx/* @clydebarrow @nielsnl68
 | 
			
		||||
esphome/components/improv_base/* @esphome/core
 | 
			
		||||
@@ -237,6 +243,7 @@ esphome/components/ltr_als_ps/* @latonita
 | 
			
		||||
esphome/components/lvgl/* @clydebarrow
 | 
			
		||||
esphome/components/m5stack_8angle/* @rnauber
 | 
			
		||||
esphome/components/matrix_keypad/* @ssieb
 | 
			
		||||
esphome/components/max17043/* @blacknell
 | 
			
		||||
esphome/components/max31865/* @DAVe3283
 | 
			
		||||
esphome/components/max44009/* @berfenger
 | 
			
		||||
esphome/components/max6956/* @looping40
 | 
			
		||||
@@ -324,7 +331,7 @@ esphome/components/pvvx_mithermometer/* @pasiz
 | 
			
		||||
esphome/components/pylontech/* @functionpointer
 | 
			
		||||
esphome/components/qmp6988/* @andrewpc
 | 
			
		||||
esphome/components/qr_code/* @wjtje
 | 
			
		||||
esphome/components/qspi_amoled/* @clydebarrow
 | 
			
		||||
esphome/components/qspi_dbi/* @clydebarrow
 | 
			
		||||
esphome/components/qwiic_pir/* @kahrendt
 | 
			
		||||
esphome/components/radon_eye_ble/* @jeffeb3
 | 
			
		||||
esphome/components/radon_eye_rd200/* @jeffeb3
 | 
			
		||||
@@ -348,6 +355,8 @@ esphome/components/sdl/* @clydebarrow
 | 
			
		||||
esphome/components/sdm_meter/* @jesserockz @polyfaces
 | 
			
		||||
esphome/components/sdp3x/* @Azimath
 | 
			
		||||
esphome/components/seeed_mr24hpc1/* @limengdu
 | 
			
		||||
esphome/components/seeed_mr60bha2/* @limengdu
 | 
			
		||||
esphome/components/seeed_mr60fda2/* @limengdu
 | 
			
		||||
esphome/components/selec_meter/* @sourabhjaiswal
 | 
			
		||||
esphome/components/select/* @esphome/core
 | 
			
		||||
esphome/components/sen0321/* @notjj
 | 
			
		||||
@@ -373,7 +382,7 @@ esphome/components/smt100/* @piechade
 | 
			
		||||
esphome/components/sn74hc165/* @jesserockz
 | 
			
		||||
esphome/components/socket/* @esphome/core
 | 
			
		||||
esphome/components/sonoff_d1/* @anatoly-savchenkov
 | 
			
		||||
esphome/components/speaker/* @jesserockz
 | 
			
		||||
esphome/components/speaker/* @jesserockz @kahrendt
 | 
			
		||||
esphome/components/spi/* @clydebarrow @esphome/core
 | 
			
		||||
esphome/components/spi_device/* @clydebarrow
 | 
			
		||||
esphome/components/spi_led_strip/* @clydebarrow
 | 
			
		||||
@@ -402,7 +411,9 @@ esphome/components/substitutions/* @esphome/core
 | 
			
		||||
esphome/components/sun/* @OttoWinter
 | 
			
		||||
esphome/components/sun_gtil2/* @Mat931
 | 
			
		||||
esphome/components/switch/* @esphome/core
 | 
			
		||||
esphome/components/switch/binary_sensor/* @ssieb
 | 
			
		||||
esphome/components/t6615/* @tylermenezes
 | 
			
		||||
esphome/components/tc74/* @sethgirvan
 | 
			
		||||
esphome/components/tca9548a/* @andreashergert1984
 | 
			
		||||
esphome/components/tca9555/* @mobrembski
 | 
			
		||||
esphome/components/tcl112/* @glmnet
 | 
			
		||||
 
 | 
			
		||||
@@ -32,33 +32,14 @@ RUN \
 | 
			
		||||
        python3-setuptools=66.1.1-1 \
 | 
			
		||||
        python3-venv=3.11.2-1+b1 \
 | 
			
		||||
        python3-wheel=0.38.4-2 \
 | 
			
		||||
        iputils-ping=3:20221126-1 \
 | 
			
		||||
        iputils-ping=3:20221126-1+deb12u1 \
 | 
			
		||||
        git=1:2.39.5-0+deb12u1 \
 | 
			
		||||
        curl=7.88.1-10+deb12u7 \
 | 
			
		||||
        curl=7.88.1-10+deb12u8 \
 | 
			
		||||
        openssh-client=1:9.2p1-2+deb12u3 \
 | 
			
		||||
        python3-cffi=1.15.1-5 \
 | 
			
		||||
        libcairo2=1.16.0-7 \
 | 
			
		||||
        libmagic1=1:5.44-3 \
 | 
			
		||||
        patch=2.7.6-7 \
 | 
			
		||||
    && ( \
 | 
			
		||||
        ( \
 | 
			
		||||
            [ "$TARGETARCH$TARGETVARIANT" = "armv7" ] && \
 | 
			
		||||
                apt-get install -y --no-install-recommends \
 | 
			
		||||
                build-essential=12.9 \
 | 
			
		||||
                python3-dev=3.11.2-1+b1 \
 | 
			
		||||
                zlib1g-dev=1:1.2.13.dfsg-1 \
 | 
			
		||||
                libjpeg-dev=1:2.1.5-2 \
 | 
			
		||||
                libfreetype-dev=2.12.1+dfsg-5+deb12u3 \
 | 
			
		||||
                libssl-dev=3.0.14-1~deb12u2 \
 | 
			
		||||
                libffi-dev=3.4.4-1 \
 | 
			
		||||
                libopenjp2-7=2.5.0-2 \
 | 
			
		||||
                libtiff6=4.5.0-6+deb12u1 \
 | 
			
		||||
                cargo=0.66.0+ds1-1 \
 | 
			
		||||
                pkg-config=1.8.1-1 \
 | 
			
		||||
                gcc-arm-linux-gnueabihf=4:12.2.0-3 \
 | 
			
		||||
        ) \
 | 
			
		||||
        || [ "$TARGETARCH$TARGETVARIANT" != "armv7" ] \
 | 
			
		||||
    ) \
 | 
			
		||||
    && rm -rf \
 | 
			
		||||
        /tmp/* \
 | 
			
		||||
        /var/{cache,log}/* \
 | 
			
		||||
@@ -86,7 +67,7 @@ RUN \
 | 
			
		||||
    pip3 install \
 | 
			
		||||
    --break-system-packages --no-cache-dir \
 | 
			
		||||
    # Keep platformio version in sync with requirements.txt
 | 
			
		||||
    platformio==6.1.15 \
 | 
			
		||||
    platformio==6.1.16 \
 | 
			
		||||
    # Change some platformio settings
 | 
			
		||||
    && platformio settings set enable_telemetry No \
 | 
			
		||||
    && platformio settings set check_platformio_interval 1000000 \
 | 
			
		||||
@@ -97,15 +78,50 @@ RUN \
 | 
			
		||||
# tmpfs is for https://github.com/rust-lang/cargo/issues/8719
 | 
			
		||||
 | 
			
		||||
COPY requirements.txt requirements_optional.txt /
 | 
			
		||||
RUN --mount=type=tmpfs,target=/root/.cargo if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
 | 
			
		||||
        curl -L https://www.piwheels.org/cp311/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl -o /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl \
 | 
			
		||||
        && pip3 install --break-system-packages --no-cache-dir /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl \
 | 
			
		||||
        && rm /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl \
 | 
			
		||||
        && export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
 | 
			
		||||
    fi; \
 | 
			
		||||
    CARGO_REGISTRIES_CRATES_IO_PROTOCOL=sparse CARGO_HOME=/root/.cargo \
 | 
			
		||||
    pip3 install \
 | 
			
		||||
    --break-system-packages --no-cache-dir -r /requirements.txt -r /requirements_optional.txt
 | 
			
		||||
RUN --mount=type=tmpfs,target=/root/.cargo <<END-OF-RUN
 | 
			
		||||
# Fail on any non-zero status
 | 
			
		||||
set -e
 | 
			
		||||
 | 
			
		||||
if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]
 | 
			
		||||
then
 | 
			
		||||
    curl -L https://www.piwheels.org/cp311/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl -o /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl
 | 
			
		||||
    pip3 install --break-system-packages --no-cache-dir /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl
 | 
			
		||||
    rm /tmp/cryptography-43.0.0-cp37-abi3-linux_armv7l.whl
 | 
			
		||||
    export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple";
 | 
			
		||||
fi
 | 
			
		||||
 | 
			
		||||
# install build tools in case wheels are not available
 | 
			
		||||
BUILD_DEPS="
 | 
			
		||||
    build-essential=12.9
 | 
			
		||||
    python3-dev=3.11.2-1+b1
 | 
			
		||||
    zlib1g-dev=1:1.2.13.dfsg-1
 | 
			
		||||
    libjpeg-dev=1:2.1.5-2
 | 
			
		||||
    libfreetype-dev=2.12.1+dfsg-5+deb12u3
 | 
			
		||||
    libssl-dev=3.0.15-1~deb12u1
 | 
			
		||||
    libffi-dev=3.4.4-1
 | 
			
		||||
    cargo=0.66.0+ds1-1
 | 
			
		||||
    pkg-config=1.8.1-1
 | 
			
		||||
"
 | 
			
		||||
LIB_DEPS="
 | 
			
		||||
    libtiff6=4.5.0-6+deb12u1
 | 
			
		||||
    libopenjp2-7=2.5.0-2
 | 
			
		||||
"
 | 
			
		||||
if [ "$TARGETARCH$TARGETVARIANT" = "arm64" ] || [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]
 | 
			
		||||
then
 | 
			
		||||
    apt-get update
 | 
			
		||||
    apt-get install -y --no-install-recommends $BUILD_DEPS $LIB_DEPS
 | 
			
		||||
fi
 | 
			
		||||
 | 
			
		||||
CARGO_REGISTRIES_CRATES_IO_PROTOCOL=sparse CARGO_HOME=/root/.cargo
 | 
			
		||||
pip3 install --break-system-packages --no-cache-dir -r /requirements.txt -r /requirements_optional.txt
 | 
			
		||||
 | 
			
		||||
if [ "$TARGETARCH$TARGETVARIANT" = "arm64" ] || [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]
 | 
			
		||||
then
 | 
			
		||||
    apt-get remove -y --purge --auto-remove $BUILD_DEPS
 | 
			
		||||
    rm -rf /tmp/* /var/{cache,log}/* /var/lib/apt/lists/*
 | 
			
		||||
fi
 | 
			
		||||
END-OF-RUN
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
COPY script/platformio_install_deps.py platformio.ini /
 | 
			
		||||
RUN /platformio_install_deps.py /platformio.ini --libraries
 | 
			
		||||
@@ -147,6 +163,18 @@ ENTRYPOINT ["/entrypoint.sh"]
 | 
			
		||||
CMD ["dashboard", "/config"]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ARG BUILD_VERSION=dev
 | 
			
		||||
 | 
			
		||||
# Labels
 | 
			
		||||
LABEL \
 | 
			
		||||
    org.opencontainers.image.authors="The ESPHome Authors" \
 | 
			
		||||
    org.opencontainers.image.title="ESPHome" \
 | 
			
		||||
    org.opencontainers.image.description="ESPHome is a system to configure your microcontrollers by simple yet powerful configuration files and control them remotely through Home Automation systems" \
 | 
			
		||||
    org.opencontainers.image.url="https://esphome.io/" \
 | 
			
		||||
    org.opencontainers.image.documentation="https://esphome.io/" \
 | 
			
		||||
    org.opencontainers.image.source="https://github.com/esphome/esphome" \
 | 
			
		||||
    org.opencontainers.image.licenses="ESPHome" \
 | 
			
		||||
    org.opencontainers.image.version=${BUILD_VERSION}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# ======================= hassio-type image =======================
 | 
			
		||||
@@ -178,7 +206,7 @@ RUN if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
 | 
			
		||||
# Labels
 | 
			
		||||
LABEL \
 | 
			
		||||
    io.hass.name="ESPHome" \
 | 
			
		||||
    io.hass.description="Manage and program ESP8266/ESP32 microcontrollers through YAML configuration files" \
 | 
			
		||||
    io.hass.description="ESPHome is a system to configure your microcontrollers by simple yet powerful configuration files and control them remotely through Home Automation systems" \
 | 
			
		||||
    io.hass.type="addon" \
 | 
			
		||||
    io.hass.version="${BUILD_VERSION}"
 | 
			
		||||
    # io.hass.arch is inherited from addon-debian-base
 | 
			
		||||
@@ -193,17 +221,22 @@ ENV \
 | 
			
		||||
  PLATFORMIO_CORE_DIR=/esphome/.temp/platformio
 | 
			
		||||
 | 
			
		||||
RUN \
 | 
			
		||||
    apt-get update \
 | 
			
		||||
    curl -L https://apt.llvm.org/llvm-snapshot.gpg.key -o /etc/apt/trusted.gpg.d/apt.llvm.org.asc \
 | 
			
		||||
    && echo "deb http://apt.llvm.org/bookworm/ llvm-toolchain-bookworm-18 main" > /etc/apt/sources.list.d/llvm.sources.list \
 | 
			
		||||
    && apt-get update \
 | 
			
		||||
    # Use pinned versions so that we get updates with build caching
 | 
			
		||||
    && apt-get install -y --no-install-recommends \
 | 
			
		||||
        clang-format-13=1:13.0.1-11+b2 \
 | 
			
		||||
        clang-tidy-14=1:14.0.6-12 \
 | 
			
		||||
        patch=2.7.6-7 \
 | 
			
		||||
        software-properties-common=0.99.30-4.1~deb12u1 \
 | 
			
		||||
        nano=7.2-1+deb12u1 \
 | 
			
		||||
        build-essential=12.9 \
 | 
			
		||||
        python3-dev=3.11.2-1+b1 \
 | 
			
		||||
    && rm -rf \
 | 
			
		||||
    && if [ "$TARGETARCH$TARGETVARIANT" != "armv7" ]; then \
 | 
			
		||||
    # move this up after armv7 is retired
 | 
			
		||||
    apt-get install -y --no-install-recommends clang-tidy-18=1:18.1.8~++20240731024826+3b5b5c1ec4a3-1~exp1~20240731144843.145 ; \
 | 
			
		||||
    fi; \
 | 
			
		||||
    rm -rf \
 | 
			
		||||
        /tmp/* \
 | 
			
		||||
        /var/{cache,log}/* \
 | 
			
		||||
        /var/lib/apt/lists/*
 | 
			
		||||
 
 | 
			
		||||
@@ -20,6 +20,8 @@ from esphome.const import (
 | 
			
		||||
    CONF_DEASSERT_RTS_DTR,
 | 
			
		||||
    CONF_DISABLED,
 | 
			
		||||
    CONF_ESPHOME,
 | 
			
		||||
    CONF_LEVEL,
 | 
			
		||||
    CONF_LOG_TOPIC,
 | 
			
		||||
    CONF_LOGGER,
 | 
			
		||||
    CONF_MDNS,
 | 
			
		||||
    CONF_MQTT,
 | 
			
		||||
@@ -30,6 +32,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_PLATFORMIO_OPTIONS,
 | 
			
		||||
    CONF_PORT,
 | 
			
		||||
    CONF_SUBSTITUTIONS,
 | 
			
		||||
    CONF_TOPIC,
 | 
			
		||||
    PLATFORM_BK72XX,
 | 
			
		||||
    PLATFORM_ESP32,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
@@ -38,7 +41,7 @@ from esphome.const import (
 | 
			
		||||
    SECRETS_FILES,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, EsphomeError, coroutine
 | 
			
		||||
from esphome.helpers import indent, is_ip_address, get_bool_env
 | 
			
		||||
from esphome.helpers import get_bool_env, indent, is_ip_address
 | 
			
		||||
from esphome.log import Fore, color, setup_log
 | 
			
		||||
from esphome.util import (
 | 
			
		||||
    get_serial_ports,
 | 
			
		||||
@@ -95,8 +98,12 @@ def choose_upload_log_host(
 | 
			
		||||
        options.append((f"Over The Air ({CORE.address})", CORE.address))
 | 
			
		||||
        if default == "OTA":
 | 
			
		||||
            return CORE.address
 | 
			
		||||
    if show_mqtt and CONF_MQTT in CORE.config:
 | 
			
		||||
        options.append((f"MQTT ({CORE.config['mqtt'][CONF_BROKER]})", "MQTT"))
 | 
			
		||||
    if (
 | 
			
		||||
        show_mqtt
 | 
			
		||||
        and (mqtt_config := CORE.config.get(CONF_MQTT))
 | 
			
		||||
        and mqtt_logging_enabled(mqtt_config)
 | 
			
		||||
    ):
 | 
			
		||||
        options.append((f"MQTT ({mqtt_config[CONF_BROKER]})", "MQTT"))
 | 
			
		||||
        if default == "OTA":
 | 
			
		||||
            return "MQTT"
 | 
			
		||||
    if default is not None:
 | 
			
		||||
@@ -106,6 +113,17 @@ def choose_upload_log_host(
 | 
			
		||||
    return choose_prompt(options, purpose=purpose)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def mqtt_logging_enabled(mqtt_config):
 | 
			
		||||
    log_topic = mqtt_config[CONF_LOG_TOPIC]
 | 
			
		||||
    if log_topic is None:
 | 
			
		||||
        return False
 | 
			
		||||
    if CONF_TOPIC not in log_topic:
 | 
			
		||||
        return False
 | 
			
		||||
    if log_topic.get(CONF_LEVEL, None) == "NONE":
 | 
			
		||||
        return False
 | 
			
		||||
    return True
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_port_type(port):
 | 
			
		||||
    if port.startswith("/") or port.startswith("COM"):
 | 
			
		||||
        return "SERIAL"
 | 
			
		||||
@@ -345,7 +363,7 @@ def upload_program(config, args, host):
 | 
			
		||||
 | 
			
		||||
    from esphome import espota2
 | 
			
		||||
 | 
			
		||||
    remote_port = ota_conf[CONF_PORT]
 | 
			
		||||
    remote_port = int(ota_conf[CONF_PORT])
 | 
			
		||||
    password = ota_conf.get(CONF_PASSWORD, "")
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
@@ -378,7 +396,7 @@ def show_logs(config, args, port):
 | 
			
		||||
 | 
			
		||||
            port = mqtt.get_esphome_device_ip(
 | 
			
		||||
                config, args.username, args.password, args.client_id
 | 
			
		||||
            )
 | 
			
		||||
            )[0]
 | 
			
		||||
 | 
			
		||||
        from esphome.components.api.client import run_logs
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,8 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ALL,
 | 
			
		||||
    CONF_ANY,
 | 
			
		||||
    CONF_AUTOMATION_ID,
 | 
			
		||||
    CONF_CONDITION,
 | 
			
		||||
    CONF_COUNT,
 | 
			
		||||
@@ -73,6 +75,13 @@ def validate_potentially_and_condition(value):
 | 
			
		||||
    return validate_condition(value)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_potentially_or_condition(value):
 | 
			
		||||
    if isinstance(value, list):
 | 
			
		||||
        with cv.remove_prepend_path(["or"]):
 | 
			
		||||
            return validate_condition({"or": value})
 | 
			
		||||
    return validate_condition(value)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
DelayAction = cg.esphome_ns.class_("DelayAction", Action, cg.Component)
 | 
			
		||||
LambdaAction = cg.esphome_ns.class_("LambdaAction", Action)
 | 
			
		||||
IfAction = cg.esphome_ns.class_("IfAction", Action)
 | 
			
		||||
@@ -166,6 +175,18 @@ async def or_condition_to_code(config, condition_id, template_arg, args):
 | 
			
		||||
    return cg.new_Pvariable(condition_id, template_arg, conditions)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register_condition("all", AndCondition, validate_condition_list)
 | 
			
		||||
async def all_condition_to_code(config, condition_id, template_arg, args):
 | 
			
		||||
    conditions = await build_condition_list(config, template_arg, args)
 | 
			
		||||
    return cg.new_Pvariable(condition_id, template_arg, conditions)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register_condition("any", OrCondition, validate_condition_list)
 | 
			
		||||
async def any_condition_to_code(config, condition_id, template_arg, args):
 | 
			
		||||
    conditions = await build_condition_list(config, template_arg, args)
 | 
			
		||||
    return cg.new_Pvariable(condition_id, template_arg, conditions)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@register_condition("not", NotCondition, validate_potentially_and_condition)
 | 
			
		||||
async def not_condition_to_code(config, condition_id, template_arg, args):
 | 
			
		||||
    condition = await build_condition(config, template_arg, args)
 | 
			
		||||
@@ -223,15 +244,21 @@ async def delay_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
    IfAction,
 | 
			
		||||
    cv.All(
 | 
			
		||||
        {
 | 
			
		||||
            cv.Required(CONF_CONDITION): validate_potentially_and_condition,
 | 
			
		||||
            cv.Exclusive(
 | 
			
		||||
                CONF_CONDITION, CONF_CONDITION
 | 
			
		||||
            ): validate_potentially_and_condition,
 | 
			
		||||
            cv.Exclusive(CONF_ANY, CONF_CONDITION): validate_potentially_or_condition,
 | 
			
		||||
            cv.Exclusive(CONF_ALL, CONF_CONDITION): validate_potentially_and_condition,
 | 
			
		||||
            cv.Optional(CONF_THEN): validate_action_list,
 | 
			
		||||
            cv.Optional(CONF_ELSE): validate_action_list,
 | 
			
		||||
        },
 | 
			
		||||
        cv.has_at_least_one_key(CONF_THEN, CONF_ELSE),
 | 
			
		||||
        cv.has_at_least_one_key(CONF_CONDITION, CONF_ANY, CONF_ALL),
 | 
			
		||||
    ),
 | 
			
		||||
)
 | 
			
		||||
async def if_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
    conditions = await build_condition(config[CONF_CONDITION], template_arg, args)
 | 
			
		||||
    cond_conf = next(el for el in config if el in (CONF_ANY, CONF_ALL, CONF_CONDITION))
 | 
			
		||||
    conditions = await build_condition(config[cond_conf], template_arg, args)
 | 
			
		||||
    var = cg.new_Pvariable(action_id, template_arg, conditions)
 | 
			
		||||
    if CONF_THEN in config:
 | 
			
		||||
        actions = await build_action_list(config[CONF_THEN], template_arg, args)
 | 
			
		||||
 
 | 
			
		||||
@@ -3,13 +3,12 @@
 | 
			
		||||
#include "esphome/components/sensor/sensor.h"
 | 
			
		||||
#include "esphome/components/voltage_sampler/voltage_sampler.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/defines.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
#include <esp_adc_cal.h>
 | 
			
		||||
#include "driver/adc.h"
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace adc {
 | 
			
		||||
@@ -43,7 +42,7 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
 | 
			
		||||
    this->channel1_ = ADC1_CHANNEL_MAX;
 | 
			
		||||
  }
 | 
			
		||||
  void set_autorange(bool autorange) { this->autorange_ = autorange; }
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
 | 
			
		||||
  /// Update ADC values
 | 
			
		||||
  void update() override;
 | 
			
		||||
@@ -59,11 +58,11 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
  std::string unique_id() override;
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_ESP8266
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
  void set_is_temperature() { this->is_temperature_ = true; }
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_RP2040
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  InternalGPIOPin *pin_;
 | 
			
		||||
@@ -72,7 +71,7 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
  bool is_temperature_{false};
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_RP2040
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
  adc_atten_t attenuation_{ADC_ATTEN_DB_0};
 | 
			
		||||
@@ -83,8 +82,8 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
 | 
			
		||||
  esp_adc_cal_characteristics_t cal_characteristics_[SOC_ADC_ATTEN_NUM] = {};
 | 
			
		||||
#else
 | 
			
		||||
  esp_adc_cal_characteristics_t cal_characteristics_[ADC_ATTEN_MAX] = {};
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // ESP_IDF_VERSION_MAJOR
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace adc
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										24
									
								
								esphome/components/adc/adc_sensor_common.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								esphome/components/adc/adc_sensor_common.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
#include "adc_sensor.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace adc {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "adc.common";
 | 
			
		||||
 | 
			
		||||
void ADCSensor::update() {
 | 
			
		||||
  float value_v = this->sample();
 | 
			
		||||
  ESP_LOGV(TAG, "'%s': Got voltage=%.4fV", this->get_name().c_str(), value_v);
 | 
			
		||||
  this->publish_state(value_v);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ADCSensor::set_sample_count(uint8_t sample_count) {
 | 
			
		||||
  if (sample_count != 0) {
 | 
			
		||||
    this->sample_count_ = sample_count;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float ADCSensor::get_setup_priority() const { return setup_priority::DATA; }
 | 
			
		||||
 | 
			
		||||
}  // namespace adc
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -1,30 +1,13 @@
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
 | 
			
		||||
#include "adc_sensor.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
#include <Esp.h>
 | 
			
		||||
ADC_MODE(ADC_VCC)
 | 
			
		||||
#else
 | 
			
		||||
#include <Arduino.h>
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
#include "pico/cyw43_arch.h"
 | 
			
		||||
#endif
 | 
			
		||||
#include <hardware/adc.h>
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace adc {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "adc";
 | 
			
		||||
static const char *const TAG = "adc.esp32";
 | 
			
		||||
 | 
			
		||||
// 13-bit for S2, 12-bit for all other ESP32 variants
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
static const adc_bits_width_t ADC_WIDTH_MAX_SOC_BITS = static_cast<adc_bits_width_t>(ADC_WIDTH_MAX - 1);
 | 
			
		||||
 | 
			
		||||
#ifndef SOC_ADC_RTC_MAX_BITWIDTH
 | 
			
		||||
@@ -32,24 +15,15 @@ static const adc_bits_width_t ADC_WIDTH_MAX_SOC_BITS = static_cast<adc_bits_widt
 | 
			
		||||
static const int32_t SOC_ADC_RTC_MAX_BITWIDTH = 13;
 | 
			
		||||
#else
 | 
			
		||||
static const int32_t SOC_ADC_RTC_MAX_BITWIDTH = 12;
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32S2
 | 
			
		||||
#endif  // SOC_ADC_RTC_MAX_BITWIDTH
 | 
			
		||||
 | 
			
		||||
static const int ADC_MAX = (1 << SOC_ADC_RTC_MAX_BITWIDTH) - 1;    // 4095 (12 bit) or 8191 (13 bit)
 | 
			
		||||
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;  // 2048 (12 bit) or 4096 (13 bit)
 | 
			
		||||
#endif
 | 
			
		||||
static const int ADC_MAX = (1 << SOC_ADC_RTC_MAX_BITWIDTH) - 1;
 | 
			
		||||
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
extern "C"
 | 
			
		||||
#endif
 | 
			
		||||
    void
 | 
			
		||||
    ADCSensor::setup() {
 | 
			
		||||
void ADCSensor::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
 | 
			
		||||
#if !defined(USE_ADC_SENSOR_VCC) && !defined(USE_RP2040)
 | 
			
		||||
  this->pin_->setup();
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
  if (this->channel1_ != ADC1_CHANNEL_MAX) {
 | 
			
		||||
    adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
 | 
			
		||||
    if (!this->autorange_) {
 | 
			
		||||
@@ -61,7 +35,6 @@ extern "C"
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // load characteristics for each attenuation
 | 
			
		||||
  for (int32_t i = 0; i <= ADC_ATTEN_DB_12_COMPAT; i++) {
 | 
			
		||||
    auto adc_unit = this->channel1_ != ADC1_CHANNEL_MAX ? ADC_UNIT_1 : ADC_UNIT_2;
 | 
			
		||||
    auto cal_value = esp_adc_cal_characterize(adc_unit, (adc_atten_t) i, ADC_WIDTH_MAX_SOC_BITS,
 | 
			
		||||
@@ -79,31 +52,10 @@ extern "C"
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
  static bool initialized = false;
 | 
			
		||||
  if (!initialized) {
 | 
			
		||||
    adc_init();
 | 
			
		||||
    initialized = true;
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "ADC '%s' setup finished!", this->get_name().c_str());
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ADCSensor::dump_config() {
 | 
			
		||||
  LOG_SENSOR("", "ADC Sensor", this);
 | 
			
		||||
#if defined(USE_ESP8266) || defined(USE_LIBRETINY)
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Pin: VCC");
 | 
			
		||||
#else
 | 
			
		||||
  LOG_PIN("  Pin: ", this->pin_);
 | 
			
		||||
#endif
 | 
			
		||||
#endif  // USE_ESP8266 || USE_LIBRETINY
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
  LOG_PIN("  Pin: ", this->pin_);
 | 
			
		||||
  if (this->autorange_) {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Attenuation: auto");
 | 
			
		||||
@@ -125,55 +77,10 @@ void ADCSensor::dump_config() {
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
  if (this->is_temperature_) {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Pin: Temperature");
 | 
			
		||||
  } else {
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Pin: VCC");
 | 
			
		||||
#else
 | 
			
		||||
    LOG_PIN("  Pin: ", this->pin_);
 | 
			
		||||
#endif  // USE_ADC_SENSOR_VCC
 | 
			
		||||
  }
 | 
			
		||||
#endif  // USE_RP2040
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Samples: %i", this->sample_count_);
 | 
			
		||||
  LOG_UPDATE_INTERVAL(this);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float ADCSensor::get_setup_priority() const { return setup_priority::DATA; }
 | 
			
		||||
void ADCSensor::update() {
 | 
			
		||||
  float value_v = this->sample();
 | 
			
		||||
  ESP_LOGV(TAG, "'%s': Got voltage=%.4fV", this->get_name().c_str(), value_v);
 | 
			
		||||
  this->publish_state(value_v);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ADCSensor::set_sample_count(uint8_t sample_count) {
 | 
			
		||||
  if (sample_count != 0) {
 | 
			
		||||
    this->sample_count_ = sample_count;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  uint32_t raw = 0;
 | 
			
		||||
  for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
    raw += ESP.getVcc();  // NOLINT(readability-static-accessed-through-instance)
 | 
			
		||||
#else
 | 
			
		||||
    raw += analogRead(this->pin_->get_pin());  // NOLINT
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
  raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
  if (this->output_raw_) {
 | 
			
		||||
    return raw;
 | 
			
		||||
  }
 | 
			
		||||
  return raw / 1024.0f;
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  if (!this->autorange_) {
 | 
			
		||||
    uint32_t sum = 0;
 | 
			
		||||
@@ -240,93 +147,17 @@ float ADCSensor::sample() {
 | 
			
		||||
  uint32_t mv2 = esp_adc_cal_raw_to_voltage(raw2, &this->cal_characteristics_[(int32_t) ADC_ATTEN_DB_2_5]);
 | 
			
		||||
  uint32_t mv0 = esp_adc_cal_raw_to_voltage(raw0, &this->cal_characteristics_[(int32_t) ADC_ATTEN_DB_0]);
 | 
			
		||||
 | 
			
		||||
  // Contribution of each value, in range 0-2048 (12 bit ADC) or 0-4096 (13 bit ADC)
 | 
			
		||||
  uint32_t c12 = std::min(raw12, ADC_HALF);
 | 
			
		||||
  uint32_t c6 = ADC_HALF - std::abs(raw6 - ADC_HALF);
 | 
			
		||||
  uint32_t c2 = ADC_HALF - std::abs(raw2 - ADC_HALF);
 | 
			
		||||
  uint32_t c0 = std::min(ADC_MAX - raw0, ADC_HALF);
 | 
			
		||||
  // max theoretical csum value is 4096*4 = 16384
 | 
			
		||||
  uint32_t csum = c12 + c6 + c2 + c0;
 | 
			
		||||
 | 
			
		||||
  // each mv is max 3900; so max value is 3900*4096*4, fits in unsigned32
 | 
			
		||||
  uint32_t mv_scaled = (mv12 * c12) + (mv6 * c6) + (mv2 * c2) + (mv0 * c0);
 | 
			
		||||
  return mv_scaled / (float) (csum * 1000U);
 | 
			
		||||
}
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  if (this->is_temperature_) {
 | 
			
		||||
    adc_set_temp_sensor_enabled(true);
 | 
			
		||||
    delay(1);
 | 
			
		||||
    adc_select_input(4);
 | 
			
		||||
    uint32_t raw = 0;
 | 
			
		||||
    for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
      raw += adc_read();
 | 
			
		||||
    }
 | 
			
		||||
    raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
    adc_set_temp_sensor_enabled(false);
 | 
			
		||||
    if (this->output_raw_) {
 | 
			
		||||
      return raw;
 | 
			
		||||
    }
 | 
			
		||||
    return raw * 3.3f / 4096.0f;
 | 
			
		||||
  } else {
 | 
			
		||||
    uint8_t pin = this->pin_->get_pin();
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
    if (pin == PICO_VSYS_PIN) {
 | 
			
		||||
      // Measuring VSYS on Raspberry Pico W needs to be wrapped with
 | 
			
		||||
      // `cyw43_thread_enter()`/`cyw43_thread_exit()` as discussed in
 | 
			
		||||
      // https://github.com/raspberrypi/pico-sdk/issues/1222, since Wifi chip and
 | 
			
		||||
      // VSYS ADC both share GPIO29
 | 
			
		||||
      cyw43_thread_enter();
 | 
			
		||||
    }
 | 
			
		||||
#endif  // CYW43_USES_VSYS_PIN
 | 
			
		||||
 | 
			
		||||
    adc_gpio_init(pin);
 | 
			
		||||
    adc_select_input(pin - 26);
 | 
			
		||||
 | 
			
		||||
    uint32_t raw = 0;
 | 
			
		||||
    for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
      raw += adc_read();
 | 
			
		||||
    }
 | 
			
		||||
    raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
    if (pin == PICO_VSYS_PIN) {
 | 
			
		||||
      cyw43_thread_exit();
 | 
			
		||||
    }
 | 
			
		||||
#endif  // CYW43_USES_VSYS_PIN
 | 
			
		||||
 | 
			
		||||
    if (this->output_raw_) {
 | 
			
		||||
      return raw;
 | 
			
		||||
    }
 | 
			
		||||
    float coeff = pin == PICO_VSYS_PIN ? 3.0 : 1.0;
 | 
			
		||||
    return raw * 3.3f / 4096.0f * coeff;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LIBRETINY
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  uint32_t raw = 0;
 | 
			
		||||
  if (this->output_raw_) {
 | 
			
		||||
    for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
      raw += analogRead(this->pin_->get_pin());  // NOLINT
 | 
			
		||||
    }
 | 
			
		||||
    raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
    return raw;
 | 
			
		||||
  }
 | 
			
		||||
  for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
    raw += analogReadVoltage(this->pin_->get_pin());  // NOLINT
 | 
			
		||||
  }
 | 
			
		||||
  raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
  return raw / 1000.0f;
 | 
			
		||||
}
 | 
			
		||||
#endif  // USE_LIBRETINY
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
std::string ADCSensor::unique_id() { return get_mac_address() + "-adc"; }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
}  // namespace adc
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
							
								
								
									
										58
									
								
								esphome/components/adc/adc_sensor_esp8266.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								esphome/components/adc/adc_sensor_esp8266.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
 | 
			
		||||
#include "adc_sensor.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
#include <Esp.h>
 | 
			
		||||
ADC_MODE(ADC_VCC)
 | 
			
		||||
#else
 | 
			
		||||
#include <Arduino.h>
 | 
			
		||||
#endif  // USE_ADC_SENSOR_VCC
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace adc {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "adc.esp8266";
 | 
			
		||||
 | 
			
		||||
void ADCSensor::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
 | 
			
		||||
#ifndef USE_ADC_SENSOR_VCC
 | 
			
		||||
  this->pin_->setup();
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ADCSensor::dump_config() {
 | 
			
		||||
  LOG_SENSOR("", "ADC Sensor", this);
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Pin: VCC");
 | 
			
		||||
#else
 | 
			
		||||
  LOG_PIN("  Pin: ", this->pin_);
 | 
			
		||||
#endif  // USE_ADC_SENSOR_VCC
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Samples: %i", this->sample_count_);
 | 
			
		||||
  LOG_UPDATE_INTERVAL(this);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  uint32_t raw = 0;
 | 
			
		||||
  for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
    raw += ESP.getVcc();  // NOLINT(readability-static-accessed-through-instance)
 | 
			
		||||
#else
 | 
			
		||||
    raw += analogRead(this->pin_->get_pin());  // NOLINT
 | 
			
		||||
#endif  // USE_ADC_SENSOR_VCC
 | 
			
		||||
  }
 | 
			
		||||
  raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
  if (this->output_raw_) {
 | 
			
		||||
    return raw;
 | 
			
		||||
  }
 | 
			
		||||
  return raw / 1024.0f;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::string ADCSensor::unique_id() { return get_mac_address() + "-adc"; }
 | 
			
		||||
 | 
			
		||||
}  // namespace adc
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP8266
 | 
			
		||||
							
								
								
									
										93
									
								
								esphome/components/adc/adc_sensor_rp2040.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								esphome/components/adc/adc_sensor_rp2040.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,93 @@
 | 
			
		||||
#ifdef USE_RP2040
 | 
			
		||||
 | 
			
		||||
#include "adc_sensor.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
#include "pico/cyw43_arch.h"
 | 
			
		||||
#endif  // CYW43_USES_VSYS_PIN
 | 
			
		||||
#include <hardware/adc.h>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace adc {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "adc.rp2040";
 | 
			
		||||
 | 
			
		||||
void ADCSensor::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
 | 
			
		||||
  static bool initialized = false;
 | 
			
		||||
  if (!initialized) {
 | 
			
		||||
    adc_init();
 | 
			
		||||
    initialized = true;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ADCSensor::dump_config() {
 | 
			
		||||
  LOG_SENSOR("", "ADC Sensor", this);
 | 
			
		||||
  if (this->is_temperature_) {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Pin: Temperature");
 | 
			
		||||
  } else {
 | 
			
		||||
#ifdef USE_ADC_SENSOR_VCC
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Pin: VCC");
 | 
			
		||||
#else
 | 
			
		||||
    LOG_PIN("  Pin: ", this->pin_);
 | 
			
		||||
#endif  // USE_ADC_SENSOR_VCC
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Samples: %i", this->sample_count_);
 | 
			
		||||
  LOG_UPDATE_INTERVAL(this);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float ADCSensor::sample() {
 | 
			
		||||
  if (this->is_temperature_) {
 | 
			
		||||
    adc_set_temp_sensor_enabled(true);
 | 
			
		||||
    delay(1);
 | 
			
		||||
    adc_select_input(4);
 | 
			
		||||
    uint32_t raw = 0;
 | 
			
		||||
    for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
      raw += adc_read();
 | 
			
		||||
    }
 | 
			
		||||
    raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
    adc_set_temp_sensor_enabled(false);
 | 
			
		||||
    if (this->output_raw_) {
 | 
			
		||||
      return raw;
 | 
			
		||||
    }
 | 
			
		||||
    return raw * 3.3f / 4096.0f;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  uint8_t pin = this->pin_->get_pin();
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
  if (pin == PICO_VSYS_PIN) {
 | 
			
		||||
    // Measuring VSYS on Raspberry Pico W needs to be wrapped with
 | 
			
		||||
    // `cyw43_thread_enter()`/`cyw43_thread_exit()` as discussed in
 | 
			
		||||
    // https://github.com/raspberrypi/pico-sdk/issues/1222, since Wifi chip and
 | 
			
		||||
    // VSYS ADC both share GPIO29
 | 
			
		||||
    cyw43_thread_enter();
 | 
			
		||||
  }
 | 
			
		||||
#endif  // CYW43_USES_VSYS_PIN
 | 
			
		||||
 | 
			
		||||
  adc_gpio_init(pin);
 | 
			
		||||
  adc_select_input(pin - 26);
 | 
			
		||||
 | 
			
		||||
  uint32_t raw = 0;
 | 
			
		||||
  for (uint8_t sample = 0; sample < this->sample_count_; sample++) {
 | 
			
		||||
    raw += adc_read();
 | 
			
		||||
  }
 | 
			
		||||
  raw = (raw + (this->sample_count_ >> 1)) / this->sample_count_;  // NOLINT(clang-analyzer-core.DivideZero)
 | 
			
		||||
 | 
			
		||||
#ifdef CYW43_USES_VSYS_PIN
 | 
			
		||||
  if (pin == PICO_VSYS_PIN) {
 | 
			
		||||
    cyw43_thread_exit();
 | 
			
		||||
  }
 | 
			
		||||
#endif  // CYW43_USES_VSYS_PIN
 | 
			
		||||
 | 
			
		||||
  if (this->output_raw_) {
 | 
			
		||||
    return raw;
 | 
			
		||||
  }
 | 
			
		||||
  float coeff = pin == PICO_VSYS_PIN ? 3.0f : 1.0f;
 | 
			
		||||
  return raw * 3.3f / 4096.0f * coeff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace adc
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_RP2040
 | 
			
		||||
@@ -72,10 +72,9 @@ void AlarmControlPanelCall::validate_() {
 | 
			
		||||
      this->state_.reset();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    if (state == ACP_STATE_DISARMED &&
 | 
			
		||||
        !(this->parent_->is_state_armed(this->parent_->get_state()) ||
 | 
			
		||||
          this->parent_->get_state() == ACP_STATE_PENDING || this->parent_->get_state() == ACP_STATE_ARMING ||
 | 
			
		||||
          this->parent_->get_state() == ACP_STATE_TRIGGERED)) {
 | 
			
		||||
    if (state == ACP_STATE_DISARMED && !this->parent_->is_state_armed(this->parent_->get_state()) &&
 | 
			
		||||
        this->parent_->get_state() != ACP_STATE_PENDING && this->parent_->get_state() != ACP_STATE_ARMING &&
 | 
			
		||||
        this->parent_->get_state() != ACP_STATE_TRIGGERED) {
 | 
			
		||||
      ESP_LOGW(TAG, "Cannot disarm when not armed");
 | 
			
		||||
      this->state_.reset();
 | 
			
		||||
      return;
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,6 @@ import logging
 | 
			
		||||
 | 
			
		||||
from esphome import automation, core
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import font
 | 
			
		||||
import esphome.components.image as espImage
 | 
			
		||||
from esphome.components.image import (
 | 
			
		||||
    CONF_USE_TRANSPARENCY,
 | 
			
		||||
@@ -131,7 +130,7 @@ ANIMATION_SCHEMA = cv.Schema(
 | 
			
		||||
    )
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = cv.All(font.validate_pillow_installed, ANIMATION_SCHEMA)
 | 
			
		||||
CONFIG_SCHEMA = ANIMATION_SCHEMA
 | 
			
		||||
 | 
			
		||||
NEXT_FRAME_SCHEMA = automation.maybe_simple_id(
 | 
			
		||||
    {
 | 
			
		||||
@@ -271,7 +270,8 @@ async def to_code(config):
 | 
			
		||||
                pos += 1
 | 
			
		||||
 | 
			
		||||
    elif config[CONF_TYPE] in ["RGB565", "TRANSPARENT_IMAGE"]:
 | 
			
		||||
        data = [0 for _ in range(height * width * 2 * frames)]
 | 
			
		||||
        bytes_per_pixel = 3 if transparent else 2
 | 
			
		||||
        data = [0 for _ in range(height * width * bytes_per_pixel * frames)]
 | 
			
		||||
        pos = 0
 | 
			
		||||
        for frameIndex in range(frames):
 | 
			
		||||
            image.seek(frameIndex)
 | 
			
		||||
@@ -288,17 +288,13 @@ async def to_code(config):
 | 
			
		||||
                G = g >> 2
 | 
			
		||||
                B = b >> 3
 | 
			
		||||
                rgb = (R << 11) | (G << 5) | B
 | 
			
		||||
 | 
			
		||||
                if transparent:
 | 
			
		||||
                    if rgb == 0x0020:
 | 
			
		||||
                        rgb = 0
 | 
			
		||||
                    if a < 0x80:
 | 
			
		||||
                        rgb = 0x0020
 | 
			
		||||
 | 
			
		||||
                data[pos] = rgb >> 8
 | 
			
		||||
                pos += 1
 | 
			
		||||
                data[pos] = rgb & 0xFF
 | 
			
		||||
                pos += 1
 | 
			
		||||
                if transparent:
 | 
			
		||||
                    data[pos] = a
 | 
			
		||||
                    pos += 1
 | 
			
		||||
 | 
			
		||||
    elif config[CONF_TYPE] in ["BINARY", "TRANSPARENT_BINARY"]:
 | 
			
		||||
        width8 = ((width + 7) // 8) * 8
 | 
			
		||||
 
 | 
			
		||||
@@ -62,7 +62,7 @@ void Animation::set_frame(int frame) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Animation::update_data_start_() {
 | 
			
		||||
  const uint32_t image_size = image_type_to_width_stride(this->width_, this->type_) * this->height_;
 | 
			
		||||
  const uint32_t image_size = this->get_width_stride() * this->height_;
 | 
			
		||||
  this->data_start_ = this->animation_data_start_ + image_size * this->current_frame_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -122,7 +122,8 @@ void APDS9306::update() {
 | 
			
		||||
 | 
			
		||||
  this->status_clear_warning();
 | 
			
		||||
 | 
			
		||||
  if (!(status &= 0b00001000)) {  // No new data
 | 
			
		||||
  status &= 0b00001000;
 | 
			
		||||
  if (!status) {  // No new data
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										9
									
								
								esphome/components/audio/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								esphome/components/audio/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@kahrendt"]
 | 
			
		||||
audio_ns = cg.esphome_ns.namespace("audio")
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = cv.All(
 | 
			
		||||
    cv.Schema({}),
 | 
			
		||||
)
 | 
			
		||||
							
								
								
									
										21
									
								
								esphome/components/audio/audio.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								esphome/components/audio/audio.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include <cstddef>
 | 
			
		||||
#include <cstdint>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace audio {
 | 
			
		||||
 | 
			
		||||
struct AudioStreamInfo {
 | 
			
		||||
  bool operator==(const AudioStreamInfo &rhs) const {
 | 
			
		||||
    return (channels == rhs.channels) && (bits_per_sample == rhs.bits_per_sample) && (sample_rate == rhs.sample_rate);
 | 
			
		||||
  }
 | 
			
		||||
  bool operator!=(const AudioStreamInfo &rhs) const { return !operator==(rhs); }
 | 
			
		||||
  size_t get_bytes_per_sample() const { return bits_per_sample / 8; }
 | 
			
		||||
  uint8_t channels = 1;
 | 
			
		||||
  uint8_t bits_per_sample = 16;
 | 
			
		||||
  uint32_t sample_rate = 16000;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace audio
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										6
									
								
								esphome/components/axs15231/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								esphome/components/axs15231/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,6 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@clydebarrow"]
 | 
			
		||||
DEPENDENCIES = ["i2c"]
 | 
			
		||||
 | 
			
		||||
axs15231_ns = cg.esphome_ns.namespace("axs15231")
 | 
			
		||||
							
								
								
									
										36
									
								
								esphome/components/axs15231/touchscreen/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								esphome/components/axs15231/touchscreen/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import i2c, touchscreen
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_ID, CONF_INTERRUPT_PIN, CONF_RESET_PIN
 | 
			
		||||
 | 
			
		||||
from .. import axs15231_ns
 | 
			
		||||
 | 
			
		||||
AXS15231Touchscreen = axs15231_ns.class_(
 | 
			
		||||
    "AXS15231Touchscreen",
 | 
			
		||||
    touchscreen.Touchscreen,
 | 
			
		||||
    i2c.I2CDevice,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = (
 | 
			
		||||
    touchscreen.touchscreen_schema("50ms")
 | 
			
		||||
    .extend(
 | 
			
		||||
        {
 | 
			
		||||
            cv.GenerateID(): cv.declare_id(AXS15231Touchscreen),
 | 
			
		||||
            cv.Optional(CONF_INTERRUPT_PIN): pins.internal_gpio_input_pin_schema,
 | 
			
		||||
            cv.Optional(CONF_RESET_PIN): pins.gpio_output_pin_schema,
 | 
			
		||||
        }
 | 
			
		||||
    )
 | 
			
		||||
    .extend(i2c.i2c_device_schema(0x3B))
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await touchscreen.register_touchscreen(var, config)
 | 
			
		||||
    await i2c.register_i2c_device(var, config)
 | 
			
		||||
 | 
			
		||||
    if interrupt_pin := config.get(CONF_INTERRUPT_PIN):
 | 
			
		||||
        cg.add(var.set_interrupt_pin(await cg.gpio_pin_expression(interrupt_pin)))
 | 
			
		||||
    if reset_pin := config.get(CONF_RESET_PIN):
 | 
			
		||||
        cg.add(var.set_reset_pin(await cg.gpio_pin_expression(reset_pin)))
 | 
			
		||||
@@ -0,0 +1,64 @@
 | 
			
		||||
#include "axs15231_touchscreen.h"
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace axs15231 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "ax15231.touchscreen";
 | 
			
		||||
 | 
			
		||||
constexpr static const uint8_t AXS_READ_TOUCHPAD[11] = {0xb5, 0xab, 0xa5, 0x5a, 0x0, 0x0, 0x0, 0x8};
 | 
			
		||||
 | 
			
		||||
#define ERROR_CHECK(err) \
 | 
			
		||||
  if ((err) != i2c::ERROR_OK) { \
 | 
			
		||||
    this->status_set_warning("Failed to communicate"); \
 | 
			
		||||
    return; \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
void AXS15231Touchscreen::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up AXS15231 Touchscreen...");
 | 
			
		||||
  if (this->reset_pin_ != nullptr) {
 | 
			
		||||
    this->reset_pin_->setup();
 | 
			
		||||
    this->reset_pin_->digital_write(false);
 | 
			
		||||
    delay(5);
 | 
			
		||||
    this->reset_pin_->digital_write(true);
 | 
			
		||||
    delay(10);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->interrupt_pin_ != nullptr) {
 | 
			
		||||
    this->interrupt_pin_->pin_mode(gpio::FLAG_INPUT);
 | 
			
		||||
    this->interrupt_pin_->setup();
 | 
			
		||||
    this->attach_interrupt_(this->interrupt_pin_, gpio::INTERRUPT_FALLING_EDGE);
 | 
			
		||||
  }
 | 
			
		||||
  this->x_raw_max_ = this->display_->get_native_width();
 | 
			
		||||
  this->y_raw_max_ = this->display_->get_native_height();
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "AXS15231 Touchscreen setup complete");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void AXS15231Touchscreen::update_touches() {
 | 
			
		||||
  i2c::ErrorCode err;
 | 
			
		||||
  uint8_t data[8]{};
 | 
			
		||||
 | 
			
		||||
  err = this->write(AXS_READ_TOUCHPAD, sizeof(AXS_READ_TOUCHPAD), false);
 | 
			
		||||
  ERROR_CHECK(err);
 | 
			
		||||
  err = this->read(data, sizeof(data));
 | 
			
		||||
  ERROR_CHECK(err);
 | 
			
		||||
  this->status_clear_warning();
 | 
			
		||||
  if (data[0] != 0)  // no touches
 | 
			
		||||
    return;
 | 
			
		||||
  uint16_t x = encode_uint16(data[2] & 0xF, data[3]);
 | 
			
		||||
  uint16_t y = encode_uint16(data[4] & 0xF, data[5]);
 | 
			
		||||
  this->add_raw_touch_position_(0, x, y);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void AXS15231Touchscreen::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "AXS15231 Touchscreen:");
 | 
			
		||||
  LOG_I2C_DEVICE(this);
 | 
			
		||||
  LOG_PIN("  Interrupt Pin: ", this->interrupt_pin_);
 | 
			
		||||
  LOG_PIN("  Reset Pin: ", this->reset_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Width: %d", this->x_raw_max_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Height: %d", this->y_raw_max_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace axs15231
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -0,0 +1,27 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/components/i2c/i2c.h"
 | 
			
		||||
#include "esphome/components/touchscreen/touchscreen.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace axs15231 {
 | 
			
		||||
 | 
			
		||||
class AXS15231Touchscreen : public touchscreen::Touchscreen, public i2c::I2CDevice {
 | 
			
		||||
 public:
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
 | 
			
		||||
  void set_interrupt_pin(InternalGPIOPin *pin) { this->interrupt_pin_ = pin; }
 | 
			
		||||
  void set_reset_pin(GPIOPin *pin) { this->reset_pin_ = pin; }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void update_touches() override;
 | 
			
		||||
 | 
			
		||||
  InternalGPIOPin *interrupt_pin_{};
 | 
			
		||||
  GPIOPin *reset_pin_{};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace axs15231
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -58,7 +58,7 @@ class BinarySensor : public EntityBase, public EntityBase_DeviceClass {
 | 
			
		||||
  void publish_initial_state(bool state);
 | 
			
		||||
 | 
			
		||||
  /// The current reported state of the binary sensor.
 | 
			
		||||
  bool state;
 | 
			
		||||
  bool state{false};
 | 
			
		||||
 | 
			
		||||
  void add_filter(Filter *filter);
 | 
			
		||||
  void add_filters(const std::vector<Filter *> &filters);
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,7 @@ from esphome.components.libretiny.const import (
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
 | 
			
		||||
from .boards import BK72XX_BOARDS, BK72XX_BOARD_PINS
 | 
			
		||||
from .boards import BK72XX_BOARD_PINS, BK72XX_BOARDS
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@kuba2k2"]
 | 
			
		||||
AUTO_LOAD = ["libretiny"]
 | 
			
		||||
 
 | 
			
		||||
@@ -16,7 +16,7 @@ CODEOWNERS = ["@neffs", "@kbx81"]
 | 
			
		||||
 | 
			
		||||
DOMAIN = "bme68x_bsec2"
 | 
			
		||||
 | 
			
		||||
BSEC2_LIBRARY_VERSION = "v1.7.2502"
 | 
			
		||||
BSEC2_LIBRARY_VERSION = "v1.8.2610"
 | 
			
		||||
 | 
			
		||||
CONF_ALGORITHM_OUTPUT = "algorithm_output"
 | 
			
		||||
CONF_BME68X_BSEC2_ID = "bme68x_bsec2_id"
 | 
			
		||||
 
 | 
			
		||||
@@ -204,11 +204,11 @@ void BME68xBSEC2Component::update_subscription_() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void BME68xBSEC2Component::run_() {
 | 
			
		||||
  this->op_mode_ = this->bsec_settings_.op_mode;
 | 
			
		||||
  int64_t curr_time_ns = this->get_time_ns_();
 | 
			
		||||
  if (curr_time_ns < this->next_call_ns_) {
 | 
			
		||||
  if (curr_time_ns < this->bsec_settings_.next_call) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->op_mode_ = this->bsec_settings_.op_mode;
 | 
			
		||||
  uint8_t status;
 | 
			
		||||
 | 
			
		||||
  ESP_LOGV(TAG, "Performing sensor run");
 | 
			
		||||
@@ -219,57 +219,60 @@ void BME68xBSEC2Component::run_() {
 | 
			
		||||
    ESP_LOGW(TAG, "Failed to fetch sensor control settings (BSEC2 error code %d)", this->bsec_status_);
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->next_call_ns_ = this->bsec_settings_.next_call;
 | 
			
		||||
 | 
			
		||||
  if (this->bsec_settings_.trigger_measurement) {
 | 
			
		||||
    bme68x_get_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
  switch (this->bsec_settings_.op_mode) {
 | 
			
		||||
    case BME68X_FORCED_MODE:
 | 
			
		||||
      bme68x_get_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
 | 
			
		||||
    bme68x_conf.os_hum = this->bsec_settings_.humidity_oversampling;
 | 
			
		||||
    bme68x_conf.os_temp = this->bsec_settings_.temperature_oversampling;
 | 
			
		||||
    bme68x_conf.os_pres = this->bsec_settings_.pressure_oversampling;
 | 
			
		||||
    bme68x_set_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
      bme68x_conf.os_hum = this->bsec_settings_.humidity_oversampling;
 | 
			
		||||
      bme68x_conf.os_temp = this->bsec_settings_.temperature_oversampling;
 | 
			
		||||
      bme68x_conf.os_pres = this->bsec_settings_.pressure_oversampling;
 | 
			
		||||
      bme68x_set_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
      this->bme68x_heatr_conf_.enable = BME68X_ENABLE;
 | 
			
		||||
      this->bme68x_heatr_conf_.heatr_temp = this->bsec_settings_.heater_temperature;
 | 
			
		||||
      this->bme68x_heatr_conf_.heatr_dur = this->bsec_settings_.heater_duration;
 | 
			
		||||
 | 
			
		||||
      // status = bme68x_set_op_mode(this->bsec_settings_.op_mode, &this->bme68x_);
 | 
			
		||||
      status = bme68x_set_heatr_conf(BME68X_FORCED_MODE, &this->bme68x_heatr_conf_, &this->bme68x_);
 | 
			
		||||
      status = bme68x_set_op_mode(BME68X_FORCED_MODE, &this->bme68x_);
 | 
			
		||||
      this->op_mode_ = BME68X_FORCED_MODE;
 | 
			
		||||
      ESP_LOGV(TAG, "Using forced mode");
 | 
			
		||||
 | 
			
		||||
      break;
 | 
			
		||||
    case BME68X_PARALLEL_MODE:
 | 
			
		||||
      if (this->op_mode_ != this->bsec_settings_.op_mode) {
 | 
			
		||||
        bme68x_get_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
 | 
			
		||||
        bme68x_conf.os_hum = this->bsec_settings_.humidity_oversampling;
 | 
			
		||||
        bme68x_conf.os_temp = this->bsec_settings_.temperature_oversampling;
 | 
			
		||||
        bme68x_conf.os_pres = this->bsec_settings_.pressure_oversampling;
 | 
			
		||||
        bme68x_set_conf(&bme68x_conf, &this->bme68x_);
 | 
			
		||||
 | 
			
		||||
    switch (this->bsec_settings_.op_mode) {
 | 
			
		||||
      case BME68X_FORCED_MODE:
 | 
			
		||||
        this->bme68x_heatr_conf_.enable = BME68X_ENABLE;
 | 
			
		||||
        this->bme68x_heatr_conf_.heatr_temp = this->bsec_settings_.heater_temperature;
 | 
			
		||||
        this->bme68x_heatr_conf_.heatr_dur = this->bsec_settings_.heater_duration;
 | 
			
		||||
        this->bme68x_heatr_conf_.heatr_temp_prof = this->bsec_settings_.heater_temperature_profile;
 | 
			
		||||
        this->bme68x_heatr_conf_.heatr_dur_prof = this->bsec_settings_.heater_duration_profile;
 | 
			
		||||
        this->bme68x_heatr_conf_.profile_len = this->bsec_settings_.heater_profile_len;
 | 
			
		||||
        this->bme68x_heatr_conf_.shared_heatr_dur =
 | 
			
		||||
            BSEC_TOTAL_HEAT_DUR -
 | 
			
		||||
            (bme68x_get_meas_dur(BME68X_PARALLEL_MODE, &bme68x_conf, &this->bme68x_) / INT64_C(1000));
 | 
			
		||||
 | 
			
		||||
        status = bme68x_set_op_mode(this->bsec_settings_.op_mode, &this->bme68x_);
 | 
			
		||||
        status = bme68x_set_heatr_conf(BME68X_FORCED_MODE, &this->bme68x_heatr_conf_, &this->bme68x_);
 | 
			
		||||
        status = bme68x_set_op_mode(BME68X_FORCED_MODE, &this->bme68x_);
 | 
			
		||||
        this->op_mode_ = BME68X_FORCED_MODE;
 | 
			
		||||
        this->sleep_mode_ = false;
 | 
			
		||||
        ESP_LOGV(TAG, "Using forced mode");
 | 
			
		||||
        status = bme68x_set_heatr_conf(BME68X_PARALLEL_MODE, &this->bme68x_heatr_conf_, &this->bme68x_);
 | 
			
		||||
 | 
			
		||||
        break;
 | 
			
		||||
      case BME68X_PARALLEL_MODE:
 | 
			
		||||
        if (this->op_mode_ != this->bsec_settings_.op_mode) {
 | 
			
		||||
          this->bme68x_heatr_conf_.enable = BME68X_ENABLE;
 | 
			
		||||
          this->bme68x_heatr_conf_.heatr_temp_prof = this->bsec_settings_.heater_temperature_profile;
 | 
			
		||||
          this->bme68x_heatr_conf_.heatr_dur_prof = this->bsec_settings_.heater_duration_profile;
 | 
			
		||||
          this->bme68x_heatr_conf_.profile_len = this->bsec_settings_.heater_profile_len;
 | 
			
		||||
          this->bme68x_heatr_conf_.shared_heatr_dur =
 | 
			
		||||
              BSEC_TOTAL_HEAT_DUR -
 | 
			
		||||
              (bme68x_get_meas_dur(BME68X_PARALLEL_MODE, &bme68x_conf, &this->bme68x_) / INT64_C(1000));
 | 
			
		||||
 | 
			
		||||
          status = bme68x_set_heatr_conf(BME68X_PARALLEL_MODE, &this->bme68x_heatr_conf_, &this->bme68x_);
 | 
			
		||||
 | 
			
		||||
          status = bme68x_set_op_mode(BME68X_PARALLEL_MODE, &this->bme68x_);
 | 
			
		||||
          this->op_mode_ = BME68X_PARALLEL_MODE;
 | 
			
		||||
          this->sleep_mode_ = false;
 | 
			
		||||
          ESP_LOGV(TAG, "Using parallel mode");
 | 
			
		||||
        }
 | 
			
		||||
        break;
 | 
			
		||||
      case BME68X_SLEEP_MODE:
 | 
			
		||||
        if (!this->sleep_mode_) {
 | 
			
		||||
          bme68x_set_op_mode(BME68X_SLEEP_MODE, &this->bme68x_);
 | 
			
		||||
          this->sleep_mode_ = true;
 | 
			
		||||
          ESP_LOGV(TAG, "Using sleep mode");
 | 
			
		||||
        }
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
        status = bme68x_set_op_mode(BME68X_PARALLEL_MODE, &this->bme68x_);
 | 
			
		||||
        this->op_mode_ = BME68X_PARALLEL_MODE;
 | 
			
		||||
        ESP_LOGV(TAG, "Using parallel mode");
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
    case BME68X_SLEEP_MODE:
 | 
			
		||||
      if (this->op_mode_ != this->bsec_settings_.op_mode) {
 | 
			
		||||
        bme68x_set_op_mode(BME68X_SLEEP_MODE, &this->bme68x_);
 | 
			
		||||
        this->op_mode_ = BME68X_SLEEP_MODE;
 | 
			
		||||
        ESP_LOGV(TAG, "Using sleep mode");
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->bsec_settings_.trigger_measurement && this->bsec_settings_.op_mode != BME68X_SLEEP_MODE) {
 | 
			
		||||
    uint32_t meas_dur = 0;
 | 
			
		||||
    meas_dur = bme68x_get_meas_dur(this->op_mode_, &bme68x_conf, &this->bme68x_);
 | 
			
		||||
    ESP_LOGV(TAG, "Queueing read in %uus", meas_dur);
 | 
			
		||||
 
 | 
			
		||||
@@ -113,13 +113,11 @@ class BME68xBSEC2Component : public Component {
 | 
			
		||||
 | 
			
		||||
  struct bme68x_heatr_conf bme68x_heatr_conf_;
 | 
			
		||||
  uint8_t op_mode_;  // operating mode of sensor
 | 
			
		||||
  bool sleep_mode_;
 | 
			
		||||
  bsec_library_return_t bsec_status_{BSEC_OK};
 | 
			
		||||
  int8_t bme68x_status_{BME68X_OK};
 | 
			
		||||
 | 
			
		||||
  int64_t last_time_ms_{0};
 | 
			
		||||
  uint32_t millis_overflow_counter_{0};
 | 
			
		||||
  int64_t next_call_ns_{0};
 | 
			
		||||
 | 
			
		||||
  std::queue<std::function<void()>> queue_;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										5
									
								
								esphome/components/bytebuffer/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								esphome/components/bytebuffer/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
CODEOWNERS = ["@clydebarrow"]
 | 
			
		||||
 | 
			
		||||
# Allows bytebuffer to be configured in yaml, to allow use of the C++ api.
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = {}
 | 
			
		||||
							
								
								
									
										421
									
								
								esphome/components/bytebuffer/bytebuffer.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										421
									
								
								esphome/components/bytebuffer/bytebuffer.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,421 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <vector>
 | 
			
		||||
#include <cinttypes>
 | 
			
		||||
#include <cstddef>
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace bytebuffer {
 | 
			
		||||
 | 
			
		||||
enum Endian { LITTLE, BIG };
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A class modelled on the Java ByteBuffer class. It wraps a vector of bytes and permits putting and getting
 | 
			
		||||
 * items of various sizes, with an automatically incremented position.
 | 
			
		||||
 *
 | 
			
		||||
 * There are three variables maintained pointing into the buffer:
 | 
			
		||||
 *
 | 
			
		||||
 * capacity: the maximum amount of data that can be stored - set on construction and cannot be changed
 | 
			
		||||
 * limit: the limit of the data currently available to get or put
 | 
			
		||||
 * position: the current insert or extract position
 | 
			
		||||
 *
 | 
			
		||||
 * 0 <= position <= limit <= capacity
 | 
			
		||||
 *
 | 
			
		||||
 * In addition a mark can be set to the current position with mark(). A subsequent call to reset() will restore
 | 
			
		||||
 * the position to the mark.
 | 
			
		||||
 *
 | 
			
		||||
 * The buffer can be marked to be little-endian (default) or big-endian. All subsequent operations will use that order.
 | 
			
		||||
 *
 | 
			
		||||
 * The flip() operation will reset the position to 0 and limit to the current position. This is useful for reading
 | 
			
		||||
 * data from a buffer after it has been written.
 | 
			
		||||
 *
 | 
			
		||||
 * The code is defined here in the header file rather than in a .cpp file, so that it does not get compiled if not used.
 | 
			
		||||
 * The templated functions ensure that only those typed functions actually used are compiled. The functions
 | 
			
		||||
 * are implicitly inline-able which will aid performance.
 | 
			
		||||
 */
 | 
			
		||||
class ByteBuffer {
 | 
			
		||||
 public:
 | 
			
		||||
  // Default constructor (compatibility with TEMPLATABLE_VALUE)
 | 
			
		||||
  // Creates a zero-length ByteBuffer which is little use to anybody.
 | 
			
		||||
  ByteBuffer() : ByteBuffer(std::vector<uint8_t>()) {}
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Create a new Bytebuffer with the given capacity
 | 
			
		||||
   */
 | 
			
		||||
  ByteBuffer(size_t capacity, Endian endianness = LITTLE)
 | 
			
		||||
      : data_(std::vector<uint8_t>(capacity)), endianness_(endianness), limit_(capacity){};
 | 
			
		||||
 | 
			
		||||
  // templated functions to implement putting and getting data of various types. There are two flavours of all
 | 
			
		||||
  // functions - one that uses the position as the offset, and updates the position accordingly, and one that
 | 
			
		||||
  // takes an explicit offset and does not update the position.
 | 
			
		||||
  // Separate temnplates are provided for types that fit into 32 bits and those that are bigger. These delegate
 | 
			
		||||
  // the actual put/get to common code based around those sizes.
 | 
			
		||||
  // This reduces the code size and execution time for smaller types. A similar structure for e.g. 16 bits is unlikely
 | 
			
		||||
  // to provide any further benefit given that all target platforms are native 32 bit.
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    // integral types that fit into 32 bit
 | 
			
		||||
    return static_cast<T>(this->get_uint32_(sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(size_t offset, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    return static_cast<T>(this->get_uint32_(offset, sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint32_(static_cast<uint32_t>(value), sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, size_t offset, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint32_(static_cast<uint32_t>(value), offset, sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // integral types that do not fit into 32 bit (basically only 64 bit types)
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    return static_cast<T>(this->get_uint64_(sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(size_t offset, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    return static_cast<T>(this->get_uint64_(offset, sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint64_(value, sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, size_t offset, typename std::enable_if<std::is_integral<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint64_(static_cast<uint64_t>(value), offset, sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // floating point types. Caters for 32 and 64 bit floating point.
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    return bit_cast<T>(this->get_uint32_(sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    return bit_cast<T>(this->get_uint64_(sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(size_t offset, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    return bit_cast<T>(this->get_uint32_(offset, sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  T get(size_t offset, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
        typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    return bit_cast<T>(this->get_uint64_(offset, sizeof(T)));
 | 
			
		||||
  }
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint32_(bit_cast<uint32_t>(value), sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint64_(bit_cast<uint64_t>(value), sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, size_t offset, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) <= sizeof(uint32_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint32_(bit_cast<uint32_t>(value), offset, sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  void put(const T &value, size_t offset, typename std::enable_if<std::is_floating_point<T>::value, T>::type * = 0,
 | 
			
		||||
           typename std::enable_if<(sizeof(T) == sizeof(uint64_t)), T>::type * = 0) {
 | 
			
		||||
    this->put_uint64_(bit_cast<uint64_t>(value), offset, sizeof(T));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  template<typename T> static ByteBuffer wrap(T value, Endian endianness = LITTLE) {
 | 
			
		||||
    ByteBuffer buffer = ByteBuffer(sizeof(T), endianness);
 | 
			
		||||
    buffer.put(value);
 | 
			
		||||
    buffer.flip();
 | 
			
		||||
    return buffer;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static ByteBuffer wrap(std::vector<uint8_t> const &data, Endian endianness = LITTLE) {
 | 
			
		||||
    ByteBuffer buffer = {data};
 | 
			
		||||
    buffer.endianness_ = endianness;
 | 
			
		||||
    return buffer;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static ByteBuffer wrap(const uint8_t *ptr, size_t len, Endian endianness = LITTLE) {
 | 
			
		||||
    return wrap(std::vector<uint8_t>(ptr, ptr + len), endianness);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // convenience functions with explicit types named..
 | 
			
		||||
  void put_float(float value) { this->put(value); }
 | 
			
		||||
  void put_double(double value) { this->put(value); }
 | 
			
		||||
 | 
			
		||||
  uint8_t get_uint8() { return this->data_[this->position_++]; }
 | 
			
		||||
  // Get a 16 bit unsigned value, increment by 2
 | 
			
		||||
  uint16_t get_uint16() { return this->get<uint16_t>(); }
 | 
			
		||||
  // Get a 24 bit unsigned value, increment by 3
 | 
			
		||||
  uint32_t get_uint24() { return this->get_uint32_(3); };
 | 
			
		||||
  // Get a 32 bit unsigned value, increment by 4
 | 
			
		||||
  uint32_t get_uint32() { return this->get<uint32_t>(); };
 | 
			
		||||
  // Get a 64 bit unsigned value, increment by 8
 | 
			
		||||
  uint64_t get_uint64() { return this->get<uint64_t>(); };
 | 
			
		||||
  // Signed versions of the get functions
 | 
			
		||||
  uint8_t get_int8() { return static_cast<int8_t>(this->get_uint8()); };
 | 
			
		||||
  int16_t get_int16() { return this->get<uint16_t>(); }
 | 
			
		||||
  int32_t get_int32() { return this->get<int32_t>(); }
 | 
			
		||||
  int64_t get_int64() { return this->get<int64_t>(); }
 | 
			
		||||
  // Get a float value, increment by 4
 | 
			
		||||
  float get_float() { return this->get<float>(); }
 | 
			
		||||
  // Get a double value, increment by 8
 | 
			
		||||
  double get_double() { return this->get<double>(); }
 | 
			
		||||
 | 
			
		||||
  // Get a bool value, increment by 1
 | 
			
		||||
  bool get_bool() { return static_cast<bool>(this->get_uint8()); }
 | 
			
		||||
 | 
			
		||||
  uint32_t get_int24(size_t offset) {
 | 
			
		||||
    auto value = this->get_uint24(offset);
 | 
			
		||||
    uint32_t mask = (~static_cast<uint32_t>(0)) << 23;
 | 
			
		||||
    if ((value & mask) != 0)
 | 
			
		||||
      value |= mask;
 | 
			
		||||
    return value;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  uint32_t get_int24() {
 | 
			
		||||
    auto value = this->get_uint24();
 | 
			
		||||
    uint32_t mask = (~static_cast<uint32_t>(0)) << 23;
 | 
			
		||||
    if ((value & mask) != 0)
 | 
			
		||||
      value |= mask;
 | 
			
		||||
    return value;
 | 
			
		||||
  }
 | 
			
		||||
  std::vector<uint8_t> get_vector(size_t length, size_t offset) {
 | 
			
		||||
    auto start = this->data_.begin() + offset;
 | 
			
		||||
    return {start, start + length};
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  std::vector<uint8_t> get_vector(size_t length) {
 | 
			
		||||
    auto result = this->get_vector(length, this->position_);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
    return result;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Convenience named functions
 | 
			
		||||
  void put_uint8(uint8_t value) { this->data_[this->position_++] = value; }
 | 
			
		||||
  void put_uint16(uint16_t value) { this->put(value); }
 | 
			
		||||
  void put_uint24(uint32_t value) { this->put_uint32_(value, 3); }
 | 
			
		||||
  void put_uint32(uint32_t value) { this->put(value); }
 | 
			
		||||
  void put_uint64(uint64_t value) { this->put(value); }
 | 
			
		||||
  // Signed versions of the put functions
 | 
			
		||||
  void put_int8(int8_t value) { this->put_uint8(static_cast<uint8_t>(value)); }
 | 
			
		||||
  void put_int16(int16_t value) { this->put(value); }
 | 
			
		||||
  void put_int24(int32_t value) { this->put_uint32_(value, 3); }
 | 
			
		||||
  void put_int32(int32_t value) { this->put(value); }
 | 
			
		||||
  void put_int64(int64_t value) { this->put(value); }
 | 
			
		||||
  // Extra put functions
 | 
			
		||||
  void put_bool(bool value) { this->put_uint8(value); }
 | 
			
		||||
 | 
			
		||||
  // versions of the above with an offset, these do not update the position
 | 
			
		||||
 | 
			
		||||
  uint64_t get_uint64(size_t offset) { return this->get<uint64_t>(offset); }
 | 
			
		||||
  uint32_t get_uint24(size_t offset) { return this->get_uint32_(offset, 3); };
 | 
			
		||||
  double get_double(size_t offset) { return get<double>(offset); }
 | 
			
		||||
 | 
			
		||||
  // Get one byte from the buffer, increment position by 1
 | 
			
		||||
  uint8_t get_uint8(size_t offset) { return this->data_[offset]; }
 | 
			
		||||
  // Get a 16 bit unsigned value, increment by 2
 | 
			
		||||
  uint16_t get_uint16(size_t offset) { return get<uint16_t>(offset); }
 | 
			
		||||
  // Get a 24 bit unsigned value, increment by 3
 | 
			
		||||
  uint32_t get_uint32(size_t offset) { return this->get<uint32_t>(offset); };
 | 
			
		||||
  // Get a 64 bit unsigned value, increment by 8
 | 
			
		||||
  uint8_t get_int8(size_t offset) { return get<int8_t>(offset); }
 | 
			
		||||
  int16_t get_int16(size_t offset) { return get<int16_t>(offset); }
 | 
			
		||||
  int32_t get_int32(size_t offset) { return get<int32_t>(offset); }
 | 
			
		||||
  int64_t get_int64(size_t offset) { return get<int64_t>(offset); }
 | 
			
		||||
  // Get a float value, increment by 4
 | 
			
		||||
  float get_float(size_t offset) { return get<float>(offset); }
 | 
			
		||||
  // Get a double value, increment by 8
 | 
			
		||||
 | 
			
		||||
  // Get a bool value, increment by 1
 | 
			
		||||
  bool get_bool(size_t offset) { return this->get_uint8(offset); }
 | 
			
		||||
 | 
			
		||||
  void put_uint8(uint8_t value, size_t offset) { this->data_[offset] = value; }
 | 
			
		||||
  void put_uint16(uint16_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_uint24(uint32_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_uint32(uint32_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_uint64(uint64_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  // Signed versions of the put functions
 | 
			
		||||
  void put_int8(int8_t value, size_t offset) { this->put_uint8(static_cast<uint8_t>(value), offset); }
 | 
			
		||||
  void put_int16(int16_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_int24(int32_t value, size_t offset) { this->put_uint32_(value, offset, 3); }
 | 
			
		||||
  void put_int32(int32_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_int64(int64_t value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  // Extra put functions
 | 
			
		||||
  void put_float(float value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_double(double value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put_bool(bool value, size_t offset) { this->put_uint8(value, offset); }
 | 
			
		||||
  void put(const std::vector<uint8_t> &value, size_t offset) {
 | 
			
		||||
    std::copy(value.begin(), value.end(), this->data_.begin() + offset);
 | 
			
		||||
  }
 | 
			
		||||
  void put_vector(const std::vector<uint8_t> &value, size_t offset) { this->put(value, offset); }
 | 
			
		||||
  void put(const std::vector<uint8_t> &value) {
 | 
			
		||||
    this->put_vector(value, this->position_);
 | 
			
		||||
    this->position_ += value.size();
 | 
			
		||||
  }
 | 
			
		||||
  void put_vector(const std::vector<uint8_t> &value) { this->put(value); }
 | 
			
		||||
 | 
			
		||||
  // Getters
 | 
			
		||||
 | 
			
		||||
  inline size_t get_capacity() const { return this->data_.size(); }
 | 
			
		||||
  inline size_t get_position() const { return this->position_; }
 | 
			
		||||
  inline size_t get_limit() const { return this->limit_; }
 | 
			
		||||
  inline size_t get_remaining() const { return this->get_limit() - this->get_position(); }
 | 
			
		||||
  inline Endian get_endianness() const { return this->endianness_; }
 | 
			
		||||
  inline void mark() { this->mark_ = this->position_; }
 | 
			
		||||
  inline void big_endian() { this->endianness_ = BIG; }
 | 
			
		||||
  inline void little_endian() { this->endianness_ = LITTLE; }
 | 
			
		||||
  // retrieve a pointer to the underlying data.
 | 
			
		||||
  std::vector<uint8_t> get_data() { return this->data_; };
 | 
			
		||||
 | 
			
		||||
  void get_bytes(void *dest, size_t length) {
 | 
			
		||||
    std::copy(this->data_.begin() + this->position_, this->data_.begin() + this->position_ + length, (uint8_t *) dest);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  void get_bytes(void *dest, size_t length, size_t offset) {
 | 
			
		||||
    std::copy(this->data_.begin() + offset, this->data_.begin() + offset + length, (uint8_t *) dest);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  void rewind() { this->position_ = 0; }
 | 
			
		||||
  void reset() { this->position_ = this->mark_; }
 | 
			
		||||
 | 
			
		||||
  void set_limit(size_t limit) { this->limit_ = limit; }
 | 
			
		||||
  void set_position(size_t position) { this->position_ = position; }
 | 
			
		||||
  void clear() {
 | 
			
		||||
    this->limit_ = this->get_capacity();
 | 
			
		||||
    this->position_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
  void flip() {
 | 
			
		||||
    this->limit_ = this->position_;
 | 
			
		||||
    this->position_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  uint64_t get_uint64_(size_t offset, size_t length) const {
 | 
			
		||||
    uint64_t value = 0;
 | 
			
		||||
    if (this->endianness_ == LITTLE) {
 | 
			
		||||
      offset += length;
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        value <<= 8;
 | 
			
		||||
        value |= this->data_[--offset];
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        value <<= 8;
 | 
			
		||||
        value |= this->data_[offset++];
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    return value;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  uint64_t get_uint64_(size_t length) {
 | 
			
		||||
    auto result = this->get_uint64_(this->position_, length);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
    return result;
 | 
			
		||||
  }
 | 
			
		||||
  uint32_t get_uint32_(size_t offset, size_t length) const {
 | 
			
		||||
    uint32_t value = 0;
 | 
			
		||||
    if (this->endianness_ == LITTLE) {
 | 
			
		||||
      offset += length;
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        value <<= 8;
 | 
			
		||||
        value |= this->data_[--offset];
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        value <<= 8;
 | 
			
		||||
        value |= this->data_[offset++];
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    return value;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  uint32_t get_uint32_(size_t length) {
 | 
			
		||||
    auto result = this->get_uint32_(this->position_, length);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
    return result;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /// Putters
 | 
			
		||||
 | 
			
		||||
  void put_uint64_(uint64_t value, size_t length) {
 | 
			
		||||
    this->put_uint64_(value, this->position_, length);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
  }
 | 
			
		||||
  void put_uint32_(uint32_t value, size_t length) {
 | 
			
		||||
    this->put_uint32_(value, this->position_, length);
 | 
			
		||||
    this->position_ += length;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  void put_uint64_(uint64_t value, size_t offset, size_t length) {
 | 
			
		||||
    if (this->endianness_ == LITTLE) {
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        this->data_[offset++] = static_cast<uint8_t>(value);
 | 
			
		||||
        value >>= 8;
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      offset += length;
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        this->data_[--offset] = static_cast<uint8_t>(value);
 | 
			
		||||
        value >>= 8;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  void put_uint32_(uint32_t value, size_t offset, size_t length) {
 | 
			
		||||
    if (this->endianness_ == LITTLE) {
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        this->data_[offset++] = static_cast<uint8_t>(value);
 | 
			
		||||
        value >>= 8;
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      offset += length;
 | 
			
		||||
      while (length-- != 0) {
 | 
			
		||||
        this->data_[--offset] = static_cast<uint8_t>(value);
 | 
			
		||||
        value >>= 8;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  ByteBuffer(std::vector<uint8_t> const &data) : data_(data), limit_(data.size()) {}
 | 
			
		||||
 | 
			
		||||
  std::vector<uint8_t> data_;
 | 
			
		||||
  Endian endianness_{LITTLE};
 | 
			
		||||
  size_t position_{0};
 | 
			
		||||
  size_t mark_{0};
 | 
			
		||||
  size_t limit_{0};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace bytebuffer
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -119,10 +119,21 @@ visual_temperature = cv.float_with_unit(
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def single_visual_temperature(value):
 | 
			
		||||
    if isinstance(value, dict):
 | 
			
		||||
        return value
 | 
			
		||||
VISUAL_TEMPERATURE_STEP_SCHEMA = cv.Schema(
 | 
			
		||||
    {
 | 
			
		||||
        cv.Required(CONF_TARGET_TEMPERATURE): visual_temperature,
 | 
			
		||||
        cv.Required(CONF_CURRENT_TEMPERATURE): visual_temperature,
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def visual_temperature_step(value):
 | 
			
		||||
 | 
			
		||||
    # Allow defining target/current temperature steps separately
 | 
			
		||||
    if isinstance(value, dict):
 | 
			
		||||
        return VISUAL_TEMPERATURE_STEP_SCHEMA(value)
 | 
			
		||||
 | 
			
		||||
    # Otherwise, use the single value for both properties
 | 
			
		||||
    value = visual_temperature(value)
 | 
			
		||||
    return VISUAL_TEMPERATURE_STEP_SCHEMA(
 | 
			
		||||
        {
 | 
			
		||||
@@ -141,16 +152,6 @@ ControlTrigger = climate_ns.class_(
 | 
			
		||||
    "ControlTrigger", automation.Trigger.template(ClimateCall.operator("ref"))
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
VISUAL_TEMPERATURE_STEP_SCHEMA = cv.Any(
 | 
			
		||||
    single_visual_temperature,
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
            cv.Required(CONF_TARGET_TEMPERATURE): visual_temperature,
 | 
			
		||||
            cv.Required(CONF_CURRENT_TEMPERATURE): visual_temperature,
 | 
			
		||||
        }
 | 
			
		||||
    ),
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CLIMATE_SCHEMA = (
 | 
			
		||||
    cv.ENTITY_BASE_SCHEMA.extend(web_server.WEBSERVER_SORTING_SCHEMA)
 | 
			
		||||
    .extend(cv.MQTT_COMMAND_COMPONENT_SCHEMA)
 | 
			
		||||
@@ -162,7 +163,7 @@ CLIMATE_SCHEMA = (
 | 
			
		||||
                {
 | 
			
		||||
                    cv.Optional(CONF_MIN_TEMPERATURE): cv.temperature,
 | 
			
		||||
                    cv.Optional(CONF_MAX_TEMPERATURE): cv.temperature,
 | 
			
		||||
                    cv.Optional(CONF_TEMPERATURE_STEP): VISUAL_TEMPERATURE_STEP_SCHEMA,
 | 
			
		||||
                    cv.Optional(CONF_TEMPERATURE_STEP): visual_temperature_step,
 | 
			
		||||
                    cv.Optional(CONF_MIN_HUMIDITY): cv.percentage_int,
 | 
			
		||||
                    cv.Optional(CONF_MAX_HUMIDITY): cv.percentage_int,
 | 
			
		||||
                }
 | 
			
		||||
 
 | 
			
		||||
@@ -43,7 +43,7 @@ bool CSE7766Component::check_byte_() {
 | 
			
		||||
  uint8_t index = this->raw_data_index_;
 | 
			
		||||
  uint8_t byte = this->raw_data_[index];
 | 
			
		||||
  if (index == 0) {
 | 
			
		||||
    return !((byte != 0x55) && ((byte & 0xF0) != 0xF0) && (byte != 0xAA));
 | 
			
		||||
    return (byte == 0x55) || ((byte & 0xF0) == 0xF0) || (byte == 0xAA);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (index == 1) {
 | 
			
		||||
 
 | 
			
		||||
@@ -70,8 +70,6 @@ def _validate_time_present(config):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
_DATETIME_SCHEMA = cv.ENTITY_BASE_SCHEMA.extend(
 | 
			
		||||
    web_server.WEBSERVER_SORTING_SCHEMA,
 | 
			
		||||
    cv.MQTT_COMMAND_COMPONENT_SCHEMA,
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
            cv.Optional(CONF_ON_VALUE): automation.validate_automation(
 | 
			
		||||
@@ -81,7 +79,9 @@ _DATETIME_SCHEMA = cv.ENTITY_BASE_SCHEMA.extend(
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_TIME_ID): cv.use_id(time.RealTimeClock),
 | 
			
		||||
        }
 | 
			
		||||
    ),
 | 
			
		||||
    )
 | 
			
		||||
    .extend(web_server.WEBSERVER_SORTING_SCHEMA)
 | 
			
		||||
    .extend(cv.MQTT_COMMAND_COMPONENT_SCHEMA)
 | 
			
		||||
).add_extra(_validate_time_present)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -60,9 +60,7 @@ ESPTime DateTimeEntity::state_as_esptime() const {
 | 
			
		||||
  obj.hour = this->hour_;
 | 
			
		||||
  obj.minute = this->minute_;
 | 
			
		||||
  obj.second = this->second_;
 | 
			
		||||
  obj.day_of_week = 1;  // Required to be valid for recalc_timestamp_local but not used.
 | 
			
		||||
  obj.day_of_year = 1;  // Required to be valid for recalc_timestamp_local but not used.
 | 
			
		||||
  obj.recalc_timestamp_local(false);
 | 
			
		||||
  obj.recalc_timestamp_local();
 | 
			
		||||
  return obj;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -36,7 +36,8 @@ std::string DebugComponent::get_reset_reason_() {
 | 
			
		||||
      break;
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32)
 | 
			
		||||
    case SW_RESET:
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32S3) || defined(USE_ESP32_VARIANT_ESP32C6)
 | 
			
		||||
    case RTC_SW_SYS_RESET:
 | 
			
		||||
#endif
 | 
			
		||||
      reset_reason = "Software Reset Digital Core";
 | 
			
		||||
@@ -72,14 +73,16 @@ std::string DebugComponent::get_reset_reason_() {
 | 
			
		||||
    case TGWDT_CPU_RESET:
 | 
			
		||||
      reset_reason = "Timer Group Reset CPU";
 | 
			
		||||
      break;
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32S3) || defined(USE_ESP32_VARIANT_ESP32C6)
 | 
			
		||||
    case TG0WDT_CPU_RESET:
 | 
			
		||||
      reset_reason = "Timer Group 0 Reset CPU";
 | 
			
		||||
      break;
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32)
 | 
			
		||||
    case SW_CPU_RESET:
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32S3) || defined(USE_ESP32_VARIANT_ESP32C6)
 | 
			
		||||
    case RTC_SW_CPU_RESET:
 | 
			
		||||
#endif
 | 
			
		||||
      reset_reason = "Software Reset CPU";
 | 
			
		||||
@@ -98,27 +101,32 @@ std::string DebugComponent::get_reset_reason_() {
 | 
			
		||||
    case RTCWDT_RTC_RESET:
 | 
			
		||||
      reset_reason = "RTC Watch Dog Reset Digital Core And RTC Module";
 | 
			
		||||
      break;
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32C6)
 | 
			
		||||
    case TG1WDT_CPU_RESET:
 | 
			
		||||
      reset_reason = "Timer Group 1 Reset CPU";
 | 
			
		||||
      break;
 | 
			
		||||
    case SUPER_WDT_RESET:
 | 
			
		||||
      reset_reason = "Super Watchdog Reset Digital Core And RTC Module";
 | 
			
		||||
      break;
 | 
			
		||||
    case GLITCH_RTC_RESET:
 | 
			
		||||
      reset_reason = "Glitch Reset Digital Core And RTC Module";
 | 
			
		||||
      break;
 | 
			
		||||
    case EFUSE_RESET:
 | 
			
		||||
      reset_reason = "eFuse Reset Digital Core";
 | 
			
		||||
      break;
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case GLITCH_RTC_RESET:
 | 
			
		||||
      reset_reason = "Glitch Reset Digital Core And RTC Module";
 | 
			
		||||
      break;
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S3) || defined(USE_ESP32_VARIANT_ESP32C6)
 | 
			
		||||
    case USB_UART_CHIP_RESET:
 | 
			
		||||
      reset_reason = "USB UART Reset Digital Core";
 | 
			
		||||
      break;
 | 
			
		||||
    case USB_JTAG_CHIP_RESET:
 | 
			
		||||
      reset_reason = "USB JTAG Reset Digital Core";
 | 
			
		||||
      break;
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case POWER_GLITCH_RESET:
 | 
			
		||||
      reset_reason = "Power Glitch Reset Digital Core And RTC Module";
 | 
			
		||||
      break;
 | 
			
		||||
 
 | 
			
		||||
@@ -52,11 +52,11 @@ void DeepSleepComponent::dump_config_platform_() {
 | 
			
		||||
 | 
			
		||||
bool DeepSleepComponent::prepare_to_sleep_() {
 | 
			
		||||
  if (this->wakeup_pin_mode_ == WAKEUP_PIN_MODE_KEEP_AWAKE && this->wakeup_pin_ != nullptr &&
 | 
			
		||||
      !this->sleep_duration_.has_value() && this->wakeup_pin_->digital_read()) {
 | 
			
		||||
      this->wakeup_pin_->digital_read()) {
 | 
			
		||||
    // Defer deep sleep until inactive
 | 
			
		||||
    if (!this->next_enter_deep_sleep_) {
 | 
			
		||||
      this->status_set_warning();
 | 
			
		||||
      ESP_LOGW(TAG, "Waiting for pin_ to switch state to enter deep sleep...");
 | 
			
		||||
      ESP_LOGW(TAG, "Waiting wakeup pin state change to enter deep sleep...");
 | 
			
		||||
    }
 | 
			
		||||
    this->next_enter_deep_sleep_ = true;
 | 
			
		||||
    return false;
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,104 @@ namespace dfplayer {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "dfplayer";
 | 
			
		||||
 | 
			
		||||
void DFPlayer::next() {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing next track");
 | 
			
		||||
  this->send_cmd_(0x01);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::previous() {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing previous track");
 | 
			
		||||
  this->send_cmd_(0x02);
 | 
			
		||||
}
 | 
			
		||||
void DFPlayer::play_mp3(uint16_t file) {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing file %d in mp3 folder", file);
 | 
			
		||||
  this->send_cmd_(0x12, file);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::play_file(uint16_t file) {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing file %d", file);
 | 
			
		||||
  this->send_cmd_(0x03, file);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::play_file_loop(uint16_t file) {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing file %d in loop", file);
 | 
			
		||||
  this->send_cmd_(0x08, file);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::play_folder_loop(uint16_t folder) {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing folder %d in loop", folder);
 | 
			
		||||
  this->send_cmd_(0x17, folder);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::volume_up() {
 | 
			
		||||
  ESP_LOGD(TAG, "Increasing volume");
 | 
			
		||||
  this->send_cmd_(0x04);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::volume_down() {
 | 
			
		||||
  ESP_LOGD(TAG, "Decreasing volume");
 | 
			
		||||
  this->send_cmd_(0x05);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::set_device(Device device) {
 | 
			
		||||
  ESP_LOGD(TAG, "Setting device to %d", device);
 | 
			
		||||
  this->send_cmd_(0x09, device);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::set_volume(uint8_t volume) {
 | 
			
		||||
  ESP_LOGD(TAG, "Setting volume to %d", volume);
 | 
			
		||||
  this->send_cmd_(0x06, volume);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::set_eq(EqPreset preset) {
 | 
			
		||||
  ESP_LOGD(TAG, "Setting EQ to %d", preset);
 | 
			
		||||
  this->send_cmd_(0x07, preset);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::sleep() {
 | 
			
		||||
  this->ack_reset_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Putting DFPlayer to sleep");
 | 
			
		||||
  this->send_cmd_(0x0A);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::reset() {
 | 
			
		||||
  this->ack_reset_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Resetting DFPlayer");
 | 
			
		||||
  this->send_cmd_(0x0C);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::start() {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Starting playback");
 | 
			
		||||
  this->send_cmd_(0x0D);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::pause() {
 | 
			
		||||
  this->ack_reset_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Pausing playback");
 | 
			
		||||
  this->send_cmd_(0x0E);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::stop() {
 | 
			
		||||
  this->ack_reset_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Stopping playback");
 | 
			
		||||
  this->send_cmd_(0x16);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::random() {
 | 
			
		||||
  this->ack_set_is_playing_ = true;
 | 
			
		||||
  ESP_LOGD(TAG, "Playing random file");
 | 
			
		||||
  this->send_cmd_(0x18);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void DFPlayer::play_folder(uint16_t folder, uint16_t file) {
 | 
			
		||||
  ESP_LOGD(TAG, "Playing file %d in folder %d", file, folder);
 | 
			
		||||
  if (folder < 100 && file < 256) {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x0F, (uint8_t) folder, (uint8_t) file);
 | 
			
		||||
@@ -29,7 +126,7 @@ void DFPlayer::send_cmd_(uint8_t cmd, uint16_t argument) {
 | 
			
		||||
 | 
			
		||||
  this->sent_cmd_ = cmd;
 | 
			
		||||
 | 
			
		||||
  ESP_LOGD(TAG, "Send Command %#02x arg %#04x", cmd, argument);
 | 
			
		||||
  ESP_LOGV(TAG, "Send Command %#02x arg %#04x", cmd, argument);
 | 
			
		||||
  this->write_array(buffer, 10);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -101,9 +198,37 @@ void DFPlayer::loop() {
 | 
			
		||||
            ESP_LOGV(TAG, "Nack");
 | 
			
		||||
            this->ack_set_is_playing_ = false;
 | 
			
		||||
            this->ack_reset_is_playing_ = false;
 | 
			
		||||
            if (argument == 6) {
 | 
			
		||||
              ESP_LOGV(TAG, "File not found");
 | 
			
		||||
              this->is_playing_ = false;
 | 
			
		||||
            switch (argument) {
 | 
			
		||||
              case 0x01:
 | 
			
		||||
                ESP_LOGE(TAG, "Module is busy or uninitialized");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x02:
 | 
			
		||||
                ESP_LOGE(TAG, "Module is in sleep mode");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x03:
 | 
			
		||||
                ESP_LOGE(TAG, "Serial receive error");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x04:
 | 
			
		||||
                ESP_LOGE(TAG, "Checksum incorrect");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x05:
 | 
			
		||||
                ESP_LOGE(TAG, "Specified track is out of current track scope");
 | 
			
		||||
                this->is_playing_ = false;
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x06:
 | 
			
		||||
                ESP_LOGE(TAG, "Specified track is not found");
 | 
			
		||||
                this->is_playing_ = false;
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x07:
 | 
			
		||||
                ESP_LOGE(TAG, "Insertion error (an inserting operation only can be done when a track is being played)");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x08:
 | 
			
		||||
                ESP_LOGE(TAG, "SD card reading failed (SD card pulled out or damaged)");
 | 
			
		||||
                break;
 | 
			
		||||
              case 0x09:
 | 
			
		||||
                ESP_LOGE(TAG, "Entered into sleep mode");
 | 
			
		||||
                this->is_playing_ = false;
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
          case 0x41:
 | 
			
		||||
@@ -113,12 +238,13 @@ void DFPlayer::loop() {
 | 
			
		||||
            this->ack_set_is_playing_ = false;
 | 
			
		||||
            this->ack_reset_is_playing_ = false;
 | 
			
		||||
            break;
 | 
			
		||||
          case 0x3D:  // Playback finished
 | 
			
		||||
          case 0x3D:
 | 
			
		||||
            ESP_LOGV(TAG, "Playback finished");
 | 
			
		||||
            this->is_playing_ = false;
 | 
			
		||||
            this->on_finished_playback_callback_.call();
 | 
			
		||||
            break;
 | 
			
		||||
          default:
 | 
			
		||||
            ESP_LOGD(TAG, "Command %#02x arg %#04x", cmd, argument);
 | 
			
		||||
            ESP_LOGV(TAG, "Received unknown cmd %#02x arg %#04x", cmd, argument);
 | 
			
		||||
        }
 | 
			
		||||
        this->sent_cmd_ = 0;
 | 
			
		||||
        this->read_pos_ = 0;
 | 
			
		||||
 
 | 
			
		||||
@@ -23,64 +23,30 @@ enum Device {
 | 
			
		||||
  TF_CARD = 2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// See the datasheet here:
 | 
			
		||||
// https://github.com/DFRobot/DFRobotDFPlayerMini/blob/master/doc/FN-M16P%2BEmbedded%2BMP3%2BAudio%2BModule%2BDatasheet.pdf
 | 
			
		||||
class DFPlayer : public uart::UARTDevice, public Component {
 | 
			
		||||
 public:
 | 
			
		||||
  void loop() override;
 | 
			
		||||
 | 
			
		||||
  void next() {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x01);
 | 
			
		||||
  }
 | 
			
		||||
  void previous() {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x02);
 | 
			
		||||
  }
 | 
			
		||||
  void play_mp3(uint16_t file) {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x12, file);
 | 
			
		||||
  }
 | 
			
		||||
  void play_file(uint16_t file) {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x03, file);
 | 
			
		||||
  }
 | 
			
		||||
  void play_file_loop(uint16_t file) {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x08, file);
 | 
			
		||||
  }
 | 
			
		||||
  void next();
 | 
			
		||||
  void previous();
 | 
			
		||||
  void play_mp3(uint16_t file);
 | 
			
		||||
  void play_file(uint16_t file);
 | 
			
		||||
  void play_file_loop(uint16_t file);
 | 
			
		||||
  void play_folder(uint16_t folder, uint16_t file);
 | 
			
		||||
  void play_folder_loop(uint16_t folder) {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x17, folder);
 | 
			
		||||
  }
 | 
			
		||||
  void volume_up() { this->send_cmd_(0x04); }
 | 
			
		||||
  void volume_down() { this->send_cmd_(0x05); }
 | 
			
		||||
  void set_device(Device device) { this->send_cmd_(0x09, device); }
 | 
			
		||||
  void set_volume(uint8_t volume) { this->send_cmd_(0x06, volume); }
 | 
			
		||||
  void set_eq(EqPreset preset) { this->send_cmd_(0x07, preset); }
 | 
			
		||||
  void sleep() {
 | 
			
		||||
    this->ack_reset_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x0A);
 | 
			
		||||
  }
 | 
			
		||||
  void reset() {
 | 
			
		||||
    this->ack_reset_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x0C);
 | 
			
		||||
  }
 | 
			
		||||
  void start() {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x0D);
 | 
			
		||||
  }
 | 
			
		||||
  void pause() {
 | 
			
		||||
    this->ack_reset_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x0E);
 | 
			
		||||
  }
 | 
			
		||||
  void stop() {
 | 
			
		||||
    this->ack_reset_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x16);
 | 
			
		||||
  }
 | 
			
		||||
  void random() {
 | 
			
		||||
    this->ack_set_is_playing_ = true;
 | 
			
		||||
    this->send_cmd_(0x18);
 | 
			
		||||
  }
 | 
			
		||||
  void play_folder_loop(uint16_t folder);
 | 
			
		||||
  void volume_up();
 | 
			
		||||
  void volume_down();
 | 
			
		||||
  void set_device(Device device);
 | 
			
		||||
  void set_volume(uint8_t volume);
 | 
			
		||||
  void set_eq(EqPreset preset);
 | 
			
		||||
  void sleep();
 | 
			
		||||
  void reset();
 | 
			
		||||
  void start();
 | 
			
		||||
  void pause();
 | 
			
		||||
  void stop();
 | 
			
		||||
  void random();
 | 
			
		||||
 | 
			
		||||
  bool is_playing() { return is_playing_; }
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
 
 | 
			
		||||
@@ -135,7 +135,8 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
 | 
			
		||||
 | 
			
		||||
      // Wait for falling edge
 | 
			
		||||
      while (this->pin_->digital_read()) {
 | 
			
		||||
        if ((end_time = micros()) - start_time > 90) {
 | 
			
		||||
        end_time = micros();
 | 
			
		||||
        if (end_time - start_time > 90) {
 | 
			
		||||
          if (i < 0) {
 | 
			
		||||
            error_code = 3;
 | 
			
		||||
          } else {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
#include "display.h"
 | 
			
		||||
#include "display_color_utils.h"
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include "display_color_utils.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
@@ -156,6 +156,148 @@ void Display::filled_circle(int center_x, int center_y, int radius, Color color)
 | 
			
		||||
    }
 | 
			
		||||
  } while (dx <= 0);
 | 
			
		||||
}
 | 
			
		||||
void Display::filled_ring(int center_x, int center_y, int radius1, int radius2, Color color) {
 | 
			
		||||
  int rmax = radius1 > radius2 ? radius1 : radius2;
 | 
			
		||||
  int rmin = radius1 < radius2 ? radius1 : radius2;
 | 
			
		||||
  int dxmax = -int32_t(rmax), dxmin = -int32_t(rmin);
 | 
			
		||||
  int dymax = 0, dymin = 0;
 | 
			
		||||
  int errmax = 2 - 2 * rmax, errmin = 2 - 2 * rmin;
 | 
			
		||||
  int e2max, e2min;
 | 
			
		||||
  do {
 | 
			
		||||
    // 8 dots for borders
 | 
			
		||||
    this->draw_pixel_at(center_x - dxmax, center_y + dymax, color);
 | 
			
		||||
    this->draw_pixel_at(center_x + dxmax, center_y + dymax, color);
 | 
			
		||||
    this->draw_pixel_at(center_x - dxmin, center_y + dymin, color);
 | 
			
		||||
    this->draw_pixel_at(center_x + dxmin, center_y + dymin, color);
 | 
			
		||||
    this->draw_pixel_at(center_x + dxmax, center_y - dymax, color);
 | 
			
		||||
    this->draw_pixel_at(center_x - dxmax, center_y - dymax, color);
 | 
			
		||||
    this->draw_pixel_at(center_x + dxmin, center_y - dymin, color);
 | 
			
		||||
    this->draw_pixel_at(center_x - dxmin, center_y - dymin, color);
 | 
			
		||||
    if (dymin < rmin) {
 | 
			
		||||
      // two parts - four lines
 | 
			
		||||
      int hline_width = -(dxmax - dxmin) + 1;
 | 
			
		||||
      this->horizontal_line(center_x + dxmax, center_y + dymax, hline_width, color);
 | 
			
		||||
      this->horizontal_line(center_x - dxmin, center_y + dymax, hline_width, color);
 | 
			
		||||
      this->horizontal_line(center_x + dxmax, center_y - dymax, hline_width, color);
 | 
			
		||||
      this->horizontal_line(center_x - dxmin, center_y - dymax, hline_width, color);
 | 
			
		||||
    } else {
 | 
			
		||||
      // one part - top and bottom
 | 
			
		||||
      int hline_width = 2 * (-dxmax) + 1;
 | 
			
		||||
      this->horizontal_line(center_x + dxmax, center_y + dymax, hline_width, color);
 | 
			
		||||
      this->horizontal_line(center_x + dxmax, center_y - dymax, hline_width, color);
 | 
			
		||||
    }
 | 
			
		||||
    e2max = errmax;
 | 
			
		||||
    // tune external
 | 
			
		||||
    if (e2max < dymax) {
 | 
			
		||||
      errmax += ++dymax * 2 + 1;
 | 
			
		||||
      if (-dxmax == dymax && e2max <= dxmax) {
 | 
			
		||||
        e2max = 0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (e2max > dxmax) {
 | 
			
		||||
      errmax += ++dxmax * 2 + 1;
 | 
			
		||||
    }
 | 
			
		||||
    // tune internal
 | 
			
		||||
    while (dymin < dymax && dymin < rmin) {
 | 
			
		||||
      e2min = errmin;
 | 
			
		||||
      if (e2min < dymin) {
 | 
			
		||||
        errmin += ++dymin * 2 + 1;
 | 
			
		||||
        if (-dxmin == dymin && e2min <= dxmin) {
 | 
			
		||||
          e2min = 0;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      if (e2min > dxmin) {
 | 
			
		||||
        errmin += ++dxmin * 2 + 1;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  } while (dxmax <= 0);
 | 
			
		||||
}
 | 
			
		||||
void Display::filled_gauge(int center_x, int center_y, int radius1, int radius2, int progress, Color color) {
 | 
			
		||||
  int rmax = radius1 > radius2 ? radius1 : radius2;
 | 
			
		||||
  int rmin = radius1 < radius2 ? radius1 : radius2;
 | 
			
		||||
  int dxmax = -int32_t(rmax), dxmin = -int32_t(rmin), upd_dxmax, upd_dxmin;
 | 
			
		||||
  int dymax = 0, dymin = 0;
 | 
			
		||||
  int errmax = 2 - 2 * rmax, errmin = 2 - 2 * rmin;
 | 
			
		||||
  int e2max, e2min;
 | 
			
		||||
  progress = std::max(0, std::min(progress, 100));  // 0..100
 | 
			
		||||
  int draw_progress = progress > 50 ? (100 - progress) : progress;
 | 
			
		||||
  float tan_a = (progress == 50) ? 65535 : tan(float(draw_progress) * M_PI / 100);  // slope
 | 
			
		||||
 | 
			
		||||
  do {
 | 
			
		||||
    // outer dots
 | 
			
		||||
    this->draw_pixel_at(center_x + dxmax, center_y - dymax, color);
 | 
			
		||||
    this->draw_pixel_at(center_x - dxmax, center_y - dymax, color);
 | 
			
		||||
    if (dymin < rmin) {  // side parts
 | 
			
		||||
      int lhline_width = -(dxmax - dxmin) + 1;
 | 
			
		||||
      if (progress >= 50) {
 | 
			
		||||
        if (float(dymax) < float(-dxmax) * tan_a) {
 | 
			
		||||
          upd_dxmax = ceil(float(dymax) / tan_a);
 | 
			
		||||
        } else {
 | 
			
		||||
          upd_dxmax = -dxmax;
 | 
			
		||||
        }
 | 
			
		||||
        this->horizontal_line(center_x + dxmax, center_y - dymax, lhline_width, color);  // left
 | 
			
		||||
        if (!dymax)
 | 
			
		||||
          this->horizontal_line(center_x - dxmin, center_y, lhline_width, color);  // right horizontal border
 | 
			
		||||
        if (upd_dxmax > -dxmin) {                                                  // right
 | 
			
		||||
          int rhline_width = (upd_dxmax + dxmin) + 1;
 | 
			
		||||
          this->horizontal_line(center_x - dxmin, center_y - dymax,
 | 
			
		||||
                                rhline_width > lhline_width ? lhline_width : rhline_width, color);
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        if (float(dymin) > float(-dxmin) * tan_a) {
 | 
			
		||||
          upd_dxmin = ceil(float(dymin) / tan_a);
 | 
			
		||||
        } else {
 | 
			
		||||
          upd_dxmin = -dxmin;
 | 
			
		||||
        }
 | 
			
		||||
        lhline_width = -(dxmax + upd_dxmin) + 1;
 | 
			
		||||
        if (!dymax)
 | 
			
		||||
          this->horizontal_line(center_x - dxmin, center_y, lhline_width, color);  // right horizontal border
 | 
			
		||||
        if (lhline_width > 0)
 | 
			
		||||
          this->horizontal_line(center_x + dxmax, center_y - dymax, lhline_width, color);
 | 
			
		||||
      }
 | 
			
		||||
    } else {  // top part
 | 
			
		||||
      int hline_width = 2 * (-dxmax) + 1;
 | 
			
		||||
      if (progress >= 50) {
 | 
			
		||||
        if (dymax < float(-dxmax) * tan_a) {
 | 
			
		||||
          upd_dxmax = ceil(float(dymax) / tan_a);
 | 
			
		||||
          hline_width = -dxmax + upd_dxmax + 1;
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        if (dymax < float(-dxmax) * tan_a) {
 | 
			
		||||
          upd_dxmax = ceil(float(dymax) / tan_a);
 | 
			
		||||
          hline_width = -dxmax - upd_dxmax + 1;
 | 
			
		||||
        } else
 | 
			
		||||
          hline_width = 0;
 | 
			
		||||
      }
 | 
			
		||||
      if (hline_width > 0)
 | 
			
		||||
        this->horizontal_line(center_x + dxmax, center_y - dymax, hline_width, color);
 | 
			
		||||
    }
 | 
			
		||||
    e2max = errmax;
 | 
			
		||||
    if (e2max < dymax) {
 | 
			
		||||
      errmax += ++dymax * 2 + 1;
 | 
			
		||||
      if (-dxmax == dymax && e2max <= dxmax) {
 | 
			
		||||
        e2max = 0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (e2max > dxmax) {
 | 
			
		||||
      errmax += ++dxmax * 2 + 1;
 | 
			
		||||
    }
 | 
			
		||||
    while (dymin <= dymax && dymin <= rmin && dxmin <= 0) {
 | 
			
		||||
      this->draw_pixel_at(center_x + dxmin, center_y - dymin, color);
 | 
			
		||||
      this->draw_pixel_at(center_x - dxmin, center_y - dymin, color);
 | 
			
		||||
      e2min = errmin;
 | 
			
		||||
      if (e2min < dymin) {
 | 
			
		||||
        errmin += ++dymin * 2 + 1;
 | 
			
		||||
        if (-dxmin == dymin && e2min <= dxmin) {
 | 
			
		||||
          e2min = 0;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      if (e2min > dxmin) {
 | 
			
		||||
        errmin += ++dxmin * 2 + 1;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  } while (dxmax <= 0);
 | 
			
		||||
}
 | 
			
		||||
void HOT Display::triangle(int x1, int y1, int x2, int y2, int x3, int y3, Color color) {
 | 
			
		||||
  this->line(x1, y1, x2, y2, color);
 | 
			
		||||
  this->line(x1, y1, x3, y3, color);
 | 
			
		||||
@@ -520,20 +662,24 @@ void DisplayOnPageChangeTrigger::process(DisplayPage *from, DisplayPage *to) {
 | 
			
		||||
  if ((this->from_ == nullptr || this->from_ == from) && (this->to_ == nullptr || this->to_ == to))
 | 
			
		||||
    this->trigger(from, to);
 | 
			
		||||
}
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, Color color, TextAlign align, const char *format, ESPTime time) {
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, Color color, Color background, TextAlign align, const char *format,
 | 
			
		||||
                       ESPTime time) {
 | 
			
		||||
  char buffer[64];
 | 
			
		||||
  size_t ret = time.strftime(buffer, sizeof(buffer), format);
 | 
			
		||||
  if (ret > 0)
 | 
			
		||||
    this->print(x, y, font, color, align, buffer);
 | 
			
		||||
    this->print(x, y, font, color, align, buffer, background);
 | 
			
		||||
}
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, Color color, TextAlign align, const char *format, ESPTime time) {
 | 
			
		||||
  this->strftime(x, y, font, color, COLOR_OFF, align, format, time);
 | 
			
		||||
}
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, Color color, const char *format, ESPTime time) {
 | 
			
		||||
  this->strftime(x, y, font, color, TextAlign::TOP_LEFT, format, time);
 | 
			
		||||
  this->strftime(x, y, font, color, COLOR_OFF, TextAlign::TOP_LEFT, format, time);
 | 
			
		||||
}
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, TextAlign align, const char *format, ESPTime time) {
 | 
			
		||||
  this->strftime(x, y, font, COLOR_ON, align, format, time);
 | 
			
		||||
  this->strftime(x, y, font, COLOR_ON, COLOR_OFF, align, format, time);
 | 
			
		||||
}
 | 
			
		||||
void Display::strftime(int x, int y, BaseFont *font, const char *format, ESPTime time) {
 | 
			
		||||
  this->strftime(x, y, font, COLOR_ON, TextAlign::TOP_LEFT, format, time);
 | 
			
		||||
  this->strftime(x, y, font, COLOR_ON, COLOR_OFF, TextAlign::TOP_LEFT, format, time);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Display::start_clipping(Rect rect) {
 | 
			
		||||
 
 | 
			
		||||
@@ -285,6 +285,13 @@ class Display : public PollingComponent {
 | 
			
		||||
  /// Fill a circle centered around [center_x,center_y] with the radius radius with the given color.
 | 
			
		||||
  void filled_circle(int center_x, int center_y, int radius, Color color = COLOR_ON);
 | 
			
		||||
 | 
			
		||||
  /// Fill a ring centered around [center_x,center_y] between two circles with the radius1 and radius2 with the given
 | 
			
		||||
  /// color.
 | 
			
		||||
  void filled_ring(int center_x, int center_y, int radius1, int radius2, Color color = COLOR_ON);
 | 
			
		||||
  /// Fill a half-ring "gauge" centered around [center_x,center_y] between two circles with the radius1 and radius2
 | 
			
		||||
  /// with he given color and filled up to 'progress' percent
 | 
			
		||||
  void filled_gauge(int center_x, int center_y, int radius1, int radius2, int progress, Color color = COLOR_ON);
 | 
			
		||||
 | 
			
		||||
  /// Draw the outline of a triangle contained between the points [x1,y1], [x2,y2] and [x3,y3] with the given color.
 | 
			
		||||
  void triangle(int x1, int y1, int x2, int y2, int x3, int y3, Color color = COLOR_ON);
 | 
			
		||||
 | 
			
		||||
@@ -430,6 +437,20 @@ class Display : public PollingComponent {
 | 
			
		||||
   */
 | 
			
		||||
  void printf(int x, int y, BaseFont *font, const char *format, ...) __attribute__((format(printf, 5, 6)));
 | 
			
		||||
 | 
			
		||||
  /** Evaluate the strftime-format `format` and print the result with the anchor point at [x,y] with `font`.
 | 
			
		||||
   *
 | 
			
		||||
   * @param x The x coordinate of the text alignment anchor point.
 | 
			
		||||
   * @param y The y coordinate of the text alignment anchor point.
 | 
			
		||||
   * @param font The font to draw the text with.
 | 
			
		||||
   * @param color The color to draw the text with.
 | 
			
		||||
   * @param background The background color to draw the text with.
 | 
			
		||||
   * @param align The alignment of the text.
 | 
			
		||||
   * @param format The format to use.
 | 
			
		||||
   * @param ... The arguments to use for the text formatting.
 | 
			
		||||
   */
 | 
			
		||||
  void strftime(int x, int y, BaseFont *font, Color color, Color background, TextAlign align, const char *format,
 | 
			
		||||
                ESPTime time) __attribute__((format(strftime, 8, 0)));
 | 
			
		||||
 | 
			
		||||
  /** Evaluate the strftime-format `format` and print the result with the anchor point at [x,y] with `font`.
 | 
			
		||||
   *
 | 
			
		||||
   * @param x The x coordinate of the text alignment anchor point.
 | 
			
		||||
 
 | 
			
		||||
@@ -68,8 +68,6 @@ IsActiveCondition = display_menu_base_ns.class_(
 | 
			
		||||
    "IsActiveCondition", automation.Condition
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
MULTI_CONF = True
 | 
			
		||||
 | 
			
		||||
MenuItemType = display_menu_base_ns.enum("MenuItemType")
 | 
			
		||||
 | 
			
		||||
MENU_ITEM_TYPES = {
 | 
			
		||||
 
 | 
			
		||||
@@ -280,7 +280,7 @@ bool DisplayMenuComponent::cursor_down_() {
 | 
			
		||||
bool DisplayMenuComponent::enter_menu_() {
 | 
			
		||||
  this->displayed_item_->on_leave();
 | 
			
		||||
  this->displayed_item_ = static_cast<MenuItemMenu *>(this->get_selected_item_());
 | 
			
		||||
  this->selection_stack_.push_front({this->top_index_, this->cursor_index_});
 | 
			
		||||
  this->selection_stack_.emplace_front(this->top_index_, this->cursor_index_);
 | 
			
		||||
  this->cursor_index_ = this->top_index_ = 0;
 | 
			
		||||
  this->displayed_item_->on_enter();
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -296,7 +296,7 @@ void Dsmr::dump_config() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Dsmr::set_decryption_key(const std::string &decryption_key) {
 | 
			
		||||
  if (decryption_key.length() == 0) {
 | 
			
		||||
  if (decryption_key.empty()) {
 | 
			
		||||
    ESP_LOGI(TAG, "Disabling decryption");
 | 
			
		||||
    this->decryption_key_.clear();
 | 
			
		||||
    if (this->crypt_telegram_ != nullptr) {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										0
									
								
								esphome/components/es8311/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								esphome/components/es8311/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										70
									
								
								esphome/components/es8311/audio_dac.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								esphome/components/es8311/audio_dac.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,70 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import i2c
 | 
			
		||||
from esphome.components.audio_dac import AudioDac
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_BITS_PER_SAMPLE, CONF_ID, CONF_SAMPLE_RATE
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@kroimon", "@kahrendt"]
 | 
			
		||||
DEPENDENCIES = ["i2c"]
 | 
			
		||||
 | 
			
		||||
es8311_ns = cg.esphome_ns.namespace("es8311")
 | 
			
		||||
ES8311 = es8311_ns.class_("ES8311", AudioDac, cg.Component, i2c.I2CDevice)
 | 
			
		||||
 | 
			
		||||
CONF_MIC_GAIN = "mic_gain"
 | 
			
		||||
CONF_USE_MCLK = "use_mclk"
 | 
			
		||||
CONF_USE_MICROPHONE = "use_microphone"
 | 
			
		||||
 | 
			
		||||
es8311_resolution = es8311_ns.enum("ES8311Resolution")
 | 
			
		||||
ES8311_BITS_PER_SAMPLE_ENUM = {
 | 
			
		||||
    16: es8311_resolution.ES8311_RESOLUTION_16,
 | 
			
		||||
    24: es8311_resolution.ES8311_RESOLUTION_24,
 | 
			
		||||
    32: es8311_resolution.ES8311_RESOLUTION_32,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
es8311_mic_gain = es8311_ns.enum("ES8311MicGain")
 | 
			
		||||
ES8311_MIC_GAIN_ENUM = {
 | 
			
		||||
    "MIN": es8311_mic_gain.ES8311_MIC_GAIN_MIN,
 | 
			
		||||
    "0DB": es8311_mic_gain.ES8311_MIC_GAIN_0DB,
 | 
			
		||||
    "6DB": es8311_mic_gain.ES8311_MIC_GAIN_6DB,
 | 
			
		||||
    "12DB": es8311_mic_gain.ES8311_MIC_GAIN_12DB,
 | 
			
		||||
    "18DB": es8311_mic_gain.ES8311_MIC_GAIN_18DB,
 | 
			
		||||
    "24DB": es8311_mic_gain.ES8311_MIC_GAIN_24DB,
 | 
			
		||||
    "30DB": es8311_mic_gain.ES8311_MIC_GAIN_30DB,
 | 
			
		||||
    "36DB": es8311_mic_gain.ES8311_MIC_GAIN_36DB,
 | 
			
		||||
    "42DB": es8311_mic_gain.ES8311_MIC_GAIN_42DB,
 | 
			
		||||
    "MAX": es8311_mic_gain.ES8311_MIC_GAIN_MAX,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
_validate_bits = cv.float_with_unit("bits", "bit")
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = (
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
            cv.GenerateID(): cv.declare_id(ES8311),
 | 
			
		||||
            cv.Optional(CONF_BITS_PER_SAMPLE, default="16bit"): cv.All(
 | 
			
		||||
                _validate_bits, cv.enum(ES8311_BITS_PER_SAMPLE_ENUM)
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_MIC_GAIN, default="42DB"): cv.enum(
 | 
			
		||||
                ES8311_MIC_GAIN_ENUM, upper=True
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_SAMPLE_RATE, default=16000): cv.int_range(min=1),
 | 
			
		||||
            cv.Optional(CONF_USE_MCLK, default=True): cv.boolean,
 | 
			
		||||
            cv.Optional(CONF_USE_MICROPHONE, default=False): cv.boolean,
 | 
			
		||||
        }
 | 
			
		||||
    )
 | 
			
		||||
    .extend(cv.COMPONENT_SCHEMA)
 | 
			
		||||
    .extend(i2c.i2c_device_schema(0x18))
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
    await i2c.register_i2c_device(var, config)
 | 
			
		||||
 | 
			
		||||
    cg.add(var.set_bits_per_sample(config[CONF_BITS_PER_SAMPLE]))
 | 
			
		||||
    cg.add(var.set_mic_gain(config[CONF_MIC_GAIN]))
 | 
			
		||||
    cg.add(var.set_sample_frequency(config[CONF_SAMPLE_RATE]))
 | 
			
		||||
    cg.add(var.set_use_mclk(config[CONF_USE_MCLK]))
 | 
			
		||||
    cg.add(var.set_use_mic(config[CONF_USE_MICROPHONE]))
 | 
			
		||||
							
								
								
									
										227
									
								
								esphome/components/es8311/es8311.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										227
									
								
								esphome/components/es8311/es8311.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,227 @@
 | 
			
		||||
#include "es8311.h"
 | 
			
		||||
#include "es8311_const.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include <cinttypes>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace es8311 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "es8311";
 | 
			
		||||
 | 
			
		||||
// Mark the component as failed; use only in setup
 | 
			
		||||
#define ES8311_ERROR_FAILED(func) \
 | 
			
		||||
  if (!(func)) { \
 | 
			
		||||
    this->mark_failed(); \
 | 
			
		||||
    return; \
 | 
			
		||||
  }
 | 
			
		||||
// Return false; use outside of setup
 | 
			
		||||
#define ES8311_ERROR_CHECK(func) \
 | 
			
		||||
  if (!(func)) { \
 | 
			
		||||
    return false; \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
void ES8311::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up ES8311...");
 | 
			
		||||
 | 
			
		||||
  // Reset
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG00_RESET, 0x1F));
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG00_RESET, 0x00));
 | 
			
		||||
 | 
			
		||||
  ES8311_ERROR_FAILED(this->configure_clock_());
 | 
			
		||||
  ES8311_ERROR_FAILED(this->configure_format_());
 | 
			
		||||
  ES8311_ERROR_FAILED(this->configure_mic_());
 | 
			
		||||
 | 
			
		||||
  // Set initial volume
 | 
			
		||||
  this->set_volume(0.75);  // 0.75 = 0xBF = 0dB
 | 
			
		||||
 | 
			
		||||
  // Power up analog circuitry
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG0D_SYSTEM, 0x01));
 | 
			
		||||
  // Enable analog PGA, enable ADC modulator
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG0E_SYSTEM, 0x02));
 | 
			
		||||
  // Power up DAC
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG12_SYSTEM, 0x00));
 | 
			
		||||
  // Enable output to HP drive
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG13_SYSTEM, 0x10));
 | 
			
		||||
  // ADC Equalizer bypass, cancel DC offset in digital domain
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG1C_ADC, 0x6A));
 | 
			
		||||
  // Bypass DAC equalizer
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG37_DAC, 0x08));
 | 
			
		||||
  // Power On
 | 
			
		||||
  ES8311_ERROR_FAILED(this->write_byte(ES8311_REG00_RESET, 0x80));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ES8311::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "ES8311 Audio Codec:");
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Use MCLK: %s", YESNO(this->use_mclk_));
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Use Microphone: %s", YESNO(this->use_mic_));
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  DAC Bits per Sample: %" PRIu8, this->resolution_out_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Sample Rate: %" PRIu32, this->sample_frequency_);
 | 
			
		||||
 | 
			
		||||
  if (this->is_failed()) {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Failed to initialize!");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool ES8311::set_volume(float volume) {
 | 
			
		||||
  volume = clamp(volume, 0.0f, 1.0f);
 | 
			
		||||
  uint8_t reg32 = remap<uint8_t, float>(volume, 0.0f, 1.0f, 0, 255);
 | 
			
		||||
  return this->write_byte(ES8311_REG32_DAC, reg32);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float ES8311::volume() {
 | 
			
		||||
  uint8_t reg32;
 | 
			
		||||
  this->read_byte(ES8311_REG32_DAC, ®32);
 | 
			
		||||
  return remap<float, uint8_t>(reg32, 0, 255, 0.0f, 1.0f);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint8_t ES8311::calculate_resolution_value(ES8311Resolution resolution) {
 | 
			
		||||
  switch (resolution) {
 | 
			
		||||
    case ES8311_RESOLUTION_16:
 | 
			
		||||
      return (3 << 2);
 | 
			
		||||
    case ES8311_RESOLUTION_18:
 | 
			
		||||
      return (2 << 2);
 | 
			
		||||
    case ES8311_RESOLUTION_20:
 | 
			
		||||
      return (1 << 2);
 | 
			
		||||
    case ES8311_RESOLUTION_24:
 | 
			
		||||
      return (0 << 2);
 | 
			
		||||
    case ES8311_RESOLUTION_32:
 | 
			
		||||
      return (4 << 2);
 | 
			
		||||
    default:
 | 
			
		||||
      return 0;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const ES8311Coefficient *ES8311::get_coefficient(uint32_t mclk, uint32_t rate) {
 | 
			
		||||
  for (const auto &coefficient : ES8311_COEFFICIENTS) {
 | 
			
		||||
    if (coefficient.mclk == mclk && coefficient.rate == rate)
 | 
			
		||||
      return &coefficient;
 | 
			
		||||
  }
 | 
			
		||||
  return nullptr;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool ES8311::configure_clock_() {
 | 
			
		||||
  // Register 0x01: select clock source for internal MCLK and determine its frequency
 | 
			
		||||
  uint8_t reg01 = 0x3F;  // Enable all clocks
 | 
			
		||||
 | 
			
		||||
  uint32_t mclk_frequency = this->sample_frequency_ * this->mclk_multiple_;
 | 
			
		||||
  if (!this->use_mclk_) {
 | 
			
		||||
    reg01 |= BIT(7);  // Use SCLK
 | 
			
		||||
    mclk_frequency = this->sample_frequency_ * (int) this->resolution_out_ * 2;
 | 
			
		||||
  }
 | 
			
		||||
  if (this->mclk_inverted_) {
 | 
			
		||||
    reg01 |= BIT(6);  // Invert MCLK pin
 | 
			
		||||
  }
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG01_CLK_MANAGER, reg01));
 | 
			
		||||
 | 
			
		||||
  // Get clock coefficients from coefficient table
 | 
			
		||||
  auto *coefficient = get_coefficient(mclk_frequency, this->sample_frequency_);
 | 
			
		||||
  if (coefficient == nullptr) {
 | 
			
		||||
    ESP_LOGE(TAG, "Unable to configure sample rate %" PRIu32 "Hz with %" PRIu32 "Hz MCLK", this->sample_frequency_,
 | 
			
		||||
             mclk_frequency);
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Register 0x02
 | 
			
		||||
  uint8_t reg02;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->read_byte(ES8311_REG02_CLK_MANAGER, ®02));
 | 
			
		||||
  reg02 &= 0x07;
 | 
			
		||||
  reg02 |= (coefficient->pre_div - 1) << 5;
 | 
			
		||||
  reg02 |= coefficient->pre_mult << 3;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG02_CLK_MANAGER, reg02));
 | 
			
		||||
 | 
			
		||||
  // Register 0x03
 | 
			
		||||
  const uint8_t reg03 = (coefficient->fs_mode << 6) | coefficient->adc_osr;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG03_CLK_MANAGER, reg03));
 | 
			
		||||
 | 
			
		||||
  // Register 0x04
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG04_CLK_MANAGER, coefficient->dac_osr));
 | 
			
		||||
 | 
			
		||||
  // Register 0x05
 | 
			
		||||
  const uint8_t reg05 = ((coefficient->adc_div - 1) << 4) | (coefficient->dac_div - 1);
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG05_CLK_MANAGER, reg05));
 | 
			
		||||
 | 
			
		||||
  // Register 0x06
 | 
			
		||||
  uint8_t reg06;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->read_byte(ES8311_REG06_CLK_MANAGER, ®06));
 | 
			
		||||
  if (this->sclk_inverted_) {
 | 
			
		||||
    reg06 |= BIT(5);
 | 
			
		||||
  } else {
 | 
			
		||||
    reg06 &= ~BIT(5);
 | 
			
		||||
  }
 | 
			
		||||
  reg06 &= 0xE0;
 | 
			
		||||
  if (coefficient->bclk_div < 19) {
 | 
			
		||||
    reg06 |= (coefficient->bclk_div - 1) << 0;
 | 
			
		||||
  } else {
 | 
			
		||||
    reg06 |= (coefficient->bclk_div) << 0;
 | 
			
		||||
  }
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG06_CLK_MANAGER, reg06));
 | 
			
		||||
 | 
			
		||||
  // Register 0x07
 | 
			
		||||
  uint8_t reg07;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->read_byte(ES8311_REG07_CLK_MANAGER, ®07));
 | 
			
		||||
  reg07 &= 0xC0;
 | 
			
		||||
  reg07 |= coefficient->lrck_h << 0;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG07_CLK_MANAGER, reg07));
 | 
			
		||||
 | 
			
		||||
  // Register 0x08
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG08_CLK_MANAGER, coefficient->lrck_l));
 | 
			
		||||
 | 
			
		||||
  // Successfully configured the clock
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool ES8311::configure_format_() {
 | 
			
		||||
  // Configure I2S mode and format
 | 
			
		||||
  uint8_t reg00;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->read_byte(ES8311_REG00_RESET, ®00));
 | 
			
		||||
  reg00 &= 0xBF;
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG00_RESET, reg00));
 | 
			
		||||
 | 
			
		||||
  // Configure SDP in resolution
 | 
			
		||||
  uint8_t reg09 = calculate_resolution_value(this->resolution_in_);
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG09_SDPIN, reg09));
 | 
			
		||||
 | 
			
		||||
  // Configure SDP out resolution
 | 
			
		||||
  uint8_t reg0a = calculate_resolution_value(this->resolution_out_);
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG0A_SDPOUT, reg0a));
 | 
			
		||||
 | 
			
		||||
  // Successfully configured the format
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool ES8311::configure_mic_() {
 | 
			
		||||
  uint8_t reg14 = 0x1A;  // Enable analog MIC and max PGA gain
 | 
			
		||||
  if (this->use_mic_) {
 | 
			
		||||
    reg14 |= BIT(6);  // Enable PDM digital microphone
 | 
			
		||||
  }
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG14_SYSTEM, reg14));
 | 
			
		||||
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG16_ADC, this->mic_gain_));  // ADC gain scale up
 | 
			
		||||
  ES8311_ERROR_CHECK(this->write_byte(ES8311_REG17_ADC, 0xC8));             // Set ADC gain
 | 
			
		||||
 | 
			
		||||
  // Successfully configured the microphones
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool ES8311::set_mute_state_(bool mute_state) {
 | 
			
		||||
  uint8_t reg31;
 | 
			
		||||
 | 
			
		||||
  this->is_muted_ = mute_state;
 | 
			
		||||
 | 
			
		||||
  if (!this->read_byte(ES8311_REG31_DAC, ®31)) {
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (mute_state) {
 | 
			
		||||
    reg31 |= BIT(6) | BIT(5);
 | 
			
		||||
  } else {
 | 
			
		||||
    reg31 &= ~(BIT(6) | BIT(5));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return this->write_byte(ES8311_REG31_DAC, reg31);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace es8311
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										135
									
								
								esphome/components/es8311/es8311.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								esphome/components/es8311/es8311.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,135 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/components/audio_dac/audio_dac.h"
 | 
			
		||||
#include "esphome/components/i2c/i2c.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace es8311 {
 | 
			
		||||
 | 
			
		||||
enum ES8311MicGain {
 | 
			
		||||
  ES8311_MIC_GAIN_MIN = -1,
 | 
			
		||||
  ES8311_MIC_GAIN_0DB,
 | 
			
		||||
  ES8311_MIC_GAIN_6DB,
 | 
			
		||||
  ES8311_MIC_GAIN_12DB,
 | 
			
		||||
  ES8311_MIC_GAIN_18DB,
 | 
			
		||||
  ES8311_MIC_GAIN_24DB,
 | 
			
		||||
  ES8311_MIC_GAIN_30DB,
 | 
			
		||||
  ES8311_MIC_GAIN_36DB,
 | 
			
		||||
  ES8311_MIC_GAIN_42DB,
 | 
			
		||||
  ES8311_MIC_GAIN_MAX
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum ES8311Resolution : uint8_t {
 | 
			
		||||
  ES8311_RESOLUTION_16 = 16,
 | 
			
		||||
  ES8311_RESOLUTION_18 = 18,
 | 
			
		||||
  ES8311_RESOLUTION_20 = 20,
 | 
			
		||||
  ES8311_RESOLUTION_24 = 24,
 | 
			
		||||
  ES8311_RESOLUTION_32 = 32
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct ES8311Coefficient {
 | 
			
		||||
  uint32_t mclk;     // mclk frequency
 | 
			
		||||
  uint32_t rate;     // sample rate
 | 
			
		||||
  uint8_t pre_div;   // the pre divider with range from 1 to 8
 | 
			
		||||
  uint8_t pre_mult;  // the pre multiplier with x1, x2, x4 and x8 selection
 | 
			
		||||
  uint8_t adc_div;   // adcclk divider
 | 
			
		||||
  uint8_t dac_div;   // dacclk divider
 | 
			
		||||
  uint8_t fs_mode;   // single speed (0) or double speed (1)
 | 
			
		||||
  uint8_t lrck_h;    // adc lrck divider and dac lrck divider
 | 
			
		||||
  uint8_t lrck_l;    //
 | 
			
		||||
  uint8_t bclk_div;  // sclk divider
 | 
			
		||||
  uint8_t adc_osr;   // adc osr
 | 
			
		||||
  uint8_t dac_osr;   // dac osr
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class ES8311 : public audio_dac::AudioDac, public Component, public i2c::I2CDevice {
 | 
			
		||||
 public:
 | 
			
		||||
  /////////////////////////
 | 
			
		||||
  // Component overrides //
 | 
			
		||||
  /////////////////////////
 | 
			
		||||
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  float get_setup_priority() const override { return setup_priority::DATA; }
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
 | 
			
		||||
  ////////////////////////
 | 
			
		||||
  // AudioDac overrides //
 | 
			
		||||
  ////////////////////////
 | 
			
		||||
 | 
			
		||||
  /// @brief Writes the volume out to the DAC
 | 
			
		||||
  /// @param volume floating point between 0.0 and 1.0
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool set_volume(float volume) override;
 | 
			
		||||
 | 
			
		||||
  /// @brief Gets the current volume out from the DAC
 | 
			
		||||
  /// @return floating point between 0.0 and 1.0
 | 
			
		||||
  float volume() override;
 | 
			
		||||
 | 
			
		||||
  /// @brief Disables mute for audio out
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool set_mute_off() override { return this->set_mute_state_(false); }
 | 
			
		||||
 | 
			
		||||
  /// @brief Enables mute for audio out
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool set_mute_on() override { return this->set_mute_state_(true); }
 | 
			
		||||
 | 
			
		||||
  bool is_muted() override { return this->is_muted_; }
 | 
			
		||||
 | 
			
		||||
  //////////////////////////////////
 | 
			
		||||
  // ES8311 configuration setters //
 | 
			
		||||
  //////////////////////////////////
 | 
			
		||||
 | 
			
		||||
  void set_use_mclk(bool use_mclk) { this->use_mclk_ = use_mclk; }
 | 
			
		||||
  void set_bits_per_sample(ES8311Resolution resolution) {
 | 
			
		||||
    this->resolution_in_ = resolution;
 | 
			
		||||
    this->resolution_out_ = resolution;
 | 
			
		||||
  }
 | 
			
		||||
  void set_sample_frequency(uint32_t sample_frequency) { this->sample_frequency_ = sample_frequency; }
 | 
			
		||||
  void set_use_mic(bool use_mic) { this->use_mic_ = use_mic; }
 | 
			
		||||
  void set_mic_gain(ES8311MicGain mic_gain) { this->mic_gain_ = mic_gain; }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  /// @brief Computes the register value for the configured resolution (bits per sample)
 | 
			
		||||
  /// @param resolution bits per sample enum for both audio in and audio out
 | 
			
		||||
  /// @return register value
 | 
			
		||||
  static uint8_t calculate_resolution_value(ES8311Resolution resolution);
 | 
			
		||||
 | 
			
		||||
  /// @brief Retrieves the appropriate registers values for the configured mclk and rate
 | 
			
		||||
  /// @param mclk mlck frequency in Hz
 | 
			
		||||
  /// @param rate sample rate frequency in Hz
 | 
			
		||||
  /// @return ES8311Coeffecient containing appropriate register values to configure the ES8311 or nullptr if impossible
 | 
			
		||||
  static const ES8311Coefficient *get_coefficient(uint32_t mclk, uint32_t rate);
 | 
			
		||||
 | 
			
		||||
  /// @brief Configures the ES8311 registers for the chosen sample rate
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool configure_clock_();
 | 
			
		||||
 | 
			
		||||
  /// @brief Configures the ES8311 registers for the chosen bits per sample
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool configure_format_();
 | 
			
		||||
 | 
			
		||||
  /// @brief Configures the ES8311 microphone registers
 | 
			
		||||
  /// @return True if successful and false otherwise
 | 
			
		||||
  bool configure_mic_();
 | 
			
		||||
 | 
			
		||||
  /// @brief Mutes or unmute the DAC audio out
 | 
			
		||||
  /// @param mute_state True to mute, false to unmute
 | 
			
		||||
  /// @return
 | 
			
		||||
  bool set_mute_state_(bool mute_state);
 | 
			
		||||
 | 
			
		||||
  bool use_mic_;
 | 
			
		||||
  ES8311MicGain mic_gain_;
 | 
			
		||||
 | 
			
		||||
  bool use_mclk_;                // true = use dedicated MCLK pin, false = use SCLK
 | 
			
		||||
  bool sclk_inverted_{false};    // SCLK is inverted
 | 
			
		||||
  bool mclk_inverted_{false};    // MCLK is inverted (ignored if use_mclk_ == false)
 | 
			
		||||
  uint32_t mclk_multiple_{256};  // MCLK frequency is sample rate * mclk_multiple_ (ignored if use_mclk_ == false)
 | 
			
		||||
 | 
			
		||||
  uint32_t sample_frequency_;  // in Hz
 | 
			
		||||
  ES8311Resolution resolution_in_;
 | 
			
		||||
  ES8311Resolution resolution_out_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace es8311
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										195
									
								
								esphome/components/es8311/es8311_const.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										195
									
								
								esphome/components/es8311/es8311_const.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,195 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "es8311.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace es8311 {
 | 
			
		||||
 | 
			
		||||
// ES8311 register addresses
 | 
			
		||||
static const uint8_t ES8311_REG00_RESET = 0x00;        // Reset
 | 
			
		||||
static const uint8_t ES8311_REG01_CLK_MANAGER = 0x01;  // Clock Manager: select clk src for mclk, enable clock for codec
 | 
			
		||||
static const uint8_t ES8311_REG02_CLK_MANAGER = 0x02;  // Clock Manager: clk divider and clk multiplier
 | 
			
		||||
static const uint8_t ES8311_REG03_CLK_MANAGER = 0x03;  // Clock Manager: adc fsmode and osr
 | 
			
		||||
static const uint8_t ES8311_REG04_CLK_MANAGER = 0x04;  // Clock Manager: dac osr
 | 
			
		||||
static const uint8_t ES8311_REG05_CLK_MANAGER = 0x05;  // Clock Manager: clk divider for adc and dac
 | 
			
		||||
static const uint8_t ES8311_REG06_CLK_MANAGER = 0x06;  // Clock Manager: bclk inverter BIT(5) and divider
 | 
			
		||||
static const uint8_t ES8311_REG07_CLK_MANAGER = 0x07;  // Clock Manager: tri-state, lrck divider
 | 
			
		||||
static const uint8_t ES8311_REG08_CLK_MANAGER = 0x08;  // Clock Manager: lrck divider
 | 
			
		||||
static const uint8_t ES8311_REG09_SDPIN = 0x09;        // Serial Digital Port: DAC
 | 
			
		||||
static const uint8_t ES8311_REG0A_SDPOUT = 0x0A;       // Serial Digital Port: ADC
 | 
			
		||||
static const uint8_t ES8311_REG0B_SYSTEM = 0x0B;       // System
 | 
			
		||||
static const uint8_t ES8311_REG0C_SYSTEM = 0x0C;       // System
 | 
			
		||||
static const uint8_t ES8311_REG0D_SYSTEM = 0x0D;       // System: power up/down
 | 
			
		||||
static const uint8_t ES8311_REG0E_SYSTEM = 0x0E;       // System: power up/down
 | 
			
		||||
static const uint8_t ES8311_REG0F_SYSTEM = 0x0F;       // System: low power
 | 
			
		||||
static const uint8_t ES8311_REG10_SYSTEM = 0x10;       // System
 | 
			
		||||
static const uint8_t ES8311_REG11_SYSTEM = 0x11;       // System
 | 
			
		||||
static const uint8_t ES8311_REG12_SYSTEM = 0x12;       // System: Enable DAC
 | 
			
		||||
static const uint8_t ES8311_REG13_SYSTEM = 0x13;       // System
 | 
			
		||||
static const uint8_t ES8311_REG14_SYSTEM = 0x14;       // System: select DMIC, select analog pga gain
 | 
			
		||||
static const uint8_t ES8311_REG15_ADC = 0x15;          // ADC: adc ramp rate, dmic sense
 | 
			
		||||
static const uint8_t ES8311_REG16_ADC = 0x16;          // ADC
 | 
			
		||||
static const uint8_t ES8311_REG17_ADC = 0x17;          // ADC: volume
 | 
			
		||||
static const uint8_t ES8311_REG18_ADC = 0x18;          // ADC: alc enable and winsize
 | 
			
		||||
static const uint8_t ES8311_REG19_ADC = 0x19;          // ADC: alc maxlevel
 | 
			
		||||
static const uint8_t ES8311_REG1A_ADC = 0x1A;          // ADC: alc automute
 | 
			
		||||
static const uint8_t ES8311_REG1B_ADC = 0x1B;          // ADC: alc automute, adc hpf s1
 | 
			
		||||
static const uint8_t ES8311_REG1C_ADC = 0x1C;          // ADC: equalizer, hpf s2
 | 
			
		||||
static const uint8_t ES8311_REG1D_ADCEQ = 0x1D;        // ADCEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG1E_ADCEQ = 0x1E;        // ADCEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG1F_ADCEQ = 0x1F;        // ADCEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG20_ADCEQ = 0x20;        // ADCEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG21_ADCEQ = 0x21;        // ADCEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG22_ADCEQ = 0x22;        // ADCEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG23_ADCEQ = 0x23;        // ADCEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG24_ADCEQ = 0x24;        // ADCEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG25_ADCEQ = 0x25;        // ADCEQ: equalizer A2
 | 
			
		||||
static const uint8_t ES8311_REG26_ADCEQ = 0x26;        // ADCEQ: equalizer A2
 | 
			
		||||
static const uint8_t ES8311_REG27_ADCEQ = 0x27;        // ADCEQ: equalizer A2
 | 
			
		||||
static const uint8_t ES8311_REG28_ADCEQ = 0x28;        // ADCEQ: equalizer A2
 | 
			
		||||
static const uint8_t ES8311_REG29_ADCEQ = 0x29;        // ADCEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG2A_ADCEQ = 0x2A;        // ADCEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG2B_ADCEQ = 0x2B;        // ADCEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG2C_ADCEQ = 0x2C;        // ADCEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG2D_ADCEQ = 0x2D;        // ADCEQ: equalizer B2
 | 
			
		||||
static const uint8_t ES8311_REG2E_ADCEQ = 0x2E;        // ADCEQ: equalizer B2
 | 
			
		||||
static const uint8_t ES8311_REG2F_ADCEQ = 0x2F;        // ADCEQ: equalizer B2
 | 
			
		||||
static const uint8_t ES8311_REG30_ADCEQ = 0x30;        // ADCEQ: equalizer B2
 | 
			
		||||
static const uint8_t ES8311_REG31_DAC = 0x31;          // DAC: mute
 | 
			
		||||
static const uint8_t ES8311_REG32_DAC = 0x32;          // DAC: volume
 | 
			
		||||
static const uint8_t ES8311_REG33_DAC = 0x33;          // DAC: offset
 | 
			
		||||
static const uint8_t ES8311_REG34_DAC = 0x34;          // DAC: drc enable, drc winsize
 | 
			
		||||
static const uint8_t ES8311_REG35_DAC = 0x35;          // DAC: drc maxlevel, minilevel
 | 
			
		||||
static const uint8_t ES8311_REG36_DAC = 0x36;          // DAC
 | 
			
		||||
static const uint8_t ES8311_REG37_DAC = 0x37;          // DAC: ramprate
 | 
			
		||||
static const uint8_t ES8311_REG38_DACEQ = 0x38;        // DACEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG39_DACEQ = 0x39;        // DACEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG3A_DACEQ = 0x3A;        // DACEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG3B_DACEQ = 0x3B;        // DACEQ: equalizer B0
 | 
			
		||||
static const uint8_t ES8311_REG3C_DACEQ = 0x3C;        // DACEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG3D_DACEQ = 0x3D;        // DACEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG3E_DACEQ = 0x3E;        // DACEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG3F_DACEQ = 0x3F;        // DACEQ: equalizer B1
 | 
			
		||||
static const uint8_t ES8311_REG40_DACEQ = 0x40;        // DACEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG41_DACEQ = 0x41;        // DACEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG42_DACEQ = 0x42;        // DACEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG43_DACEQ = 0x43;        // DACEQ: equalizer A1
 | 
			
		||||
static const uint8_t ES8311_REG44_GPIO = 0x44;         // GPIO: dac2adc for test
 | 
			
		||||
static const uint8_t ES8311_REG45_GP = 0x45;           // GPIO: GP control
 | 
			
		||||
static const uint8_t ES8311_REGFA_I2C = 0xFA;          // I2C: reset registers
 | 
			
		||||
static const uint8_t ES8311_REGFC_FLAG = 0xFC;         // Flag
 | 
			
		||||
static const uint8_t ES8311_REGFD_CHD1 = 0xFD;         // Chip: ID1
 | 
			
		||||
static const uint8_t ES8311_REGFE_CHD2 = 0xFE;         // Chip: ID2
 | 
			
		||||
static const uint8_t ES8311_REGFF_CHVER = 0xFF;        // Chip: Version
 | 
			
		||||
 | 
			
		||||
// ES8311 clock divider coefficients
 | 
			
		||||
static const ES8311Coefficient ES8311_COEFFICIENTS[] = {
 | 
			
		||||
    // clang-format off
 | 
			
		||||
 | 
			
		||||
  //   mclk,  rate, pre_  pre_  adc_  dac_  fs_   lrck  lrck bclk_  adc_  dac_
 | 
			
		||||
  //                 div, mult,  div,  div, mode,   _h,   _l,  div,  osr,  osr
 | 
			
		||||
 | 
			
		||||
  // 8k
 | 
			
		||||
  {12288000,  8000, 0x06, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  {18432000,  8000, 0x03, 0x02, 0x03, 0x03, 0x00, 0x05, 0xff, 0x18, 0x10, 0x20},
 | 
			
		||||
  {16384000,  8000, 0x08, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 8192000,  8000, 0x04, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 6144000,  8000, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 4096000,  8000, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 3072000,  8000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 2048000,  8000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1536000,  8000, 0x03, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1024000,  8000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
 | 
			
		||||
  // 11.025k
 | 
			
		||||
  {11289600, 11025, 0x04, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 5644800, 11025, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 2822400, 11025, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1411200, 11025, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
 | 
			
		||||
  // 12k
 | 
			
		||||
  {12288000, 12000, 0x04, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 6144000, 12000, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 3072000, 12000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1536000, 12000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
 | 
			
		||||
  // 16k
 | 
			
		||||
  {12288000, 16000, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  {18432000, 16000, 0x03, 0x02, 0x03, 0x03, 0x00, 0x02, 0xff, 0x0c, 0x10, 0x20},
 | 
			
		||||
  {16384000, 16000, 0x04, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 8192000, 16000, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 6144000, 16000, 0x03, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 4096000, 16000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 3072000, 16000, 0x03, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 2048000, 16000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1536000, 16000, 0x03, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
  { 1024000, 16000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x20},
 | 
			
		||||
 | 
			
		||||
  // 22.05k
 | 
			
		||||
  {11289600, 22050, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 5644800, 22050, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 2822400, 22050, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1411200, 22050, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 24k
 | 
			
		||||
  {12288000, 24000, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  {18432000, 24000, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 6144000, 24000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 3072000, 24000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1536000, 24000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 32k
 | 
			
		||||
  {12288000, 32000, 0x03, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  {18432000, 32000, 0x03, 0x04, 0x03, 0x03, 0x00, 0x02, 0xff, 0x0c, 0x10, 0x10},
 | 
			
		||||
  {16384000, 32000, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 8192000, 32000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 6144000, 32000, 0x03, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 4096000, 32000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 3072000, 32000, 0x03, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 2048000, 32000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1536000, 32000, 0x03, 0x08, 0x01, 0x01, 0x01, 0x00, 0x7f, 0x02, 0x10, 0x10},
 | 
			
		||||
  { 1024000, 32000, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 44.1k
 | 
			
		||||
  {11289600, 44100, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 5644800, 44100, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 2822400, 44100, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1411200, 44100, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 48k
 | 
			
		||||
  {12288000, 48000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  {18432000, 48000, 0x03, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 6144000, 48000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 3072000, 48000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1536000, 48000, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 64k
 | 
			
		||||
  {12288000, 64000, 0x03, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  {18432000, 64000, 0x03, 0x04, 0x03, 0x03, 0x01, 0x01, 0x7f, 0x06, 0x10, 0x10},
 | 
			
		||||
  {16384000, 64000, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 8192000, 64000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 6144000, 64000, 0x01, 0x04, 0x03, 0x03, 0x01, 0x01, 0x7f, 0x06, 0x10, 0x10},
 | 
			
		||||
  { 4096000, 64000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 3072000, 64000, 0x01, 0x08, 0x03, 0x03, 0x01, 0x01, 0x7f, 0x06, 0x10, 0x10},
 | 
			
		||||
  { 2048000, 64000, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1536000, 64000, 0x01, 0x08, 0x01, 0x01, 0x01, 0x00, 0xbf, 0x03, 0x18, 0x18},
 | 
			
		||||
  { 1024000, 64000, 0x01, 0x08, 0x01, 0x01, 0x01, 0x00, 0x7f, 0x02, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 88.2k
 | 
			
		||||
  {11289600, 88200, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 5644800, 88200, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 2822400, 88200, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1411200, 88200, 0x01, 0x08, 0x01, 0x01, 0x01, 0x00, 0x7f, 0x02, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
  // 96k
 | 
			
		||||
  {12288000, 96000, 0x01, 0x02, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  {18432000, 96000, 0x03, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 6144000, 96000, 0x01, 0x04, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 3072000, 96000, 0x01, 0x08, 0x01, 0x01, 0x00, 0x00, 0xff, 0x04, 0x10, 0x10},
 | 
			
		||||
  { 1536000, 96000, 0x01, 0x08, 0x01, 0x01, 0x01, 0x00, 0x7f, 0x02, 0x10, 0x10},
 | 
			
		||||
 | 
			
		||||
    // clang-format on
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace es8311
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -65,6 +65,8 @@ _LOGGER = logging.getLogger(__name__)
 | 
			
		||||
CODEOWNERS = ["@esphome/core"]
 | 
			
		||||
AUTO_LOAD = ["preferences"]
 | 
			
		||||
 | 
			
		||||
CONF_RELEASE = "release"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def set_core_data(config):
 | 
			
		||||
    CORE.data[KEY_ESP32] = {}
 | 
			
		||||
@@ -216,11 +218,17 @@ def _format_framework_arduino_version(ver: cv.Version) -> str:
 | 
			
		||||
    return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _format_framework_espidf_version(ver: cv.Version) -> str:
 | 
			
		||||
def _format_framework_espidf_version(
 | 
			
		||||
    ver: cv.Version, release: str, for_platformio: bool
 | 
			
		||||
) -> str:
 | 
			
		||||
    # format the given arduino (https://github.com/espressif/esp-idf/releases) version to
 | 
			
		||||
    # a PIO platformio/framework-espidf value
 | 
			
		||||
    # List of package versions: https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf
 | 
			
		||||
    return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
 | 
			
		||||
    if for_platformio:
 | 
			
		||||
        return f"platformio/framework-espidf@~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
 | 
			
		||||
    if release:
 | 
			
		||||
        return f"pioarduino/framework-espidf@https://github.com/pioarduino/esp-idf/releases/download/v{str(ver)}.{release}/esp-idf-v{str(ver)}.zip"
 | 
			
		||||
    return f"pioarduino/framework-espidf@https://github.com/pioarduino/esp-idf/releases/download/v{str(ver)}/esp-idf-v{str(ver)}.zip"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# NOTE: Keep this in mind when updating the recommended version:
 | 
			
		||||
@@ -241,11 +249,33 @@ ARDUINO_PLATFORM_VERSION = cv.Version(5, 4, 0)
 | 
			
		||||
# The default/recommended esp-idf framework version
 | 
			
		||||
#  - https://github.com/espressif/esp-idf/releases
 | 
			
		||||
#  - https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf
 | 
			
		||||
RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION = cv.Version(4, 4, 8)
 | 
			
		||||
RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION = cv.Version(5, 1, 5)
 | 
			
		||||
# The platformio/espressif32 version to use for esp-idf frameworks
 | 
			
		||||
#  - https://github.com/platformio/platform-espressif32/releases
 | 
			
		||||
#  - https://api.registry.platformio.org/v3/packages/platformio/platform/espressif32
 | 
			
		||||
ESP_IDF_PLATFORM_VERSION = cv.Version(5, 4, 0)
 | 
			
		||||
ESP_IDF_PLATFORM_VERSION = cv.Version(51, 3, 7)
 | 
			
		||||
 | 
			
		||||
# List based on https://registry.platformio.org/tools/platformio/framework-espidf/versions
 | 
			
		||||
SUPPORTED_PLATFORMIO_ESP_IDF_5X = [
 | 
			
		||||
    cv.Version(5, 3, 1),
 | 
			
		||||
    cv.Version(5, 3, 0),
 | 
			
		||||
    cv.Version(5, 2, 2),
 | 
			
		||||
    cv.Version(5, 2, 1),
 | 
			
		||||
    cv.Version(5, 1, 2),
 | 
			
		||||
    cv.Version(5, 1, 1),
 | 
			
		||||
    cv.Version(5, 1, 0),
 | 
			
		||||
    cv.Version(5, 0, 2),
 | 
			
		||||
    cv.Version(5, 0, 1),
 | 
			
		||||
    cv.Version(5, 0, 0),
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
# pioarduino versions that don't require a release number
 | 
			
		||||
# List based on https://github.com/pioarduino/esp-idf/releases
 | 
			
		||||
SUPPORTED_PIOARDUINO_ESP_IDF_5X = [
 | 
			
		||||
    cv.Version(5, 3, 1),
 | 
			
		||||
    cv.Version(5, 3, 0),
 | 
			
		||||
    cv.Version(5, 1, 5),
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _arduino_check_versions(value):
 | 
			
		||||
@@ -286,8 +316,8 @@ def _arduino_check_versions(value):
 | 
			
		||||
def _esp_idf_check_versions(value):
 | 
			
		||||
    value = value.copy()
 | 
			
		||||
    lookups = {
 | 
			
		||||
        "dev": (cv.Version(5, 1, 2), "https://github.com/espressif/esp-idf.git"),
 | 
			
		||||
        "latest": (cv.Version(5, 1, 2), None),
 | 
			
		||||
        "dev": (cv.Version(5, 1, 5), "https://github.com/espressif/esp-idf.git"),
 | 
			
		||||
        "latest": (cv.Version(5, 1, 5), None),
 | 
			
		||||
        "recommended": (RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION, None),
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -305,13 +335,51 @@ def _esp_idf_check_versions(value):
 | 
			
		||||
    if version < cv.Version(4, 0, 0):
 | 
			
		||||
        raise cv.Invalid("Only ESP-IDF 4.0+ is supported.")
 | 
			
		||||
 | 
			
		||||
    value[CONF_VERSION] = str(version)
 | 
			
		||||
    value[CONF_SOURCE] = source or _format_framework_espidf_version(version)
 | 
			
		||||
    # flag this for later *before* we set value[CONF_PLATFORM_VERSION] below
 | 
			
		||||
    has_platform_ver = CONF_PLATFORM_VERSION in value
 | 
			
		||||
 | 
			
		||||
    value[CONF_PLATFORM_VERSION] = value.get(
 | 
			
		||||
        CONF_PLATFORM_VERSION, _parse_platform_version(str(ESP_IDF_PLATFORM_VERSION))
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
        (is_platformio := _platform_is_platformio(value[CONF_PLATFORM_VERSION]))
 | 
			
		||||
        and version.major >= 5
 | 
			
		||||
        and version not in SUPPORTED_PLATFORMIO_ESP_IDF_5X
 | 
			
		||||
    ):
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"ESP-IDF {str(version)} not supported by platformio/espressif32"
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
        version.major < 5
 | 
			
		||||
        or (
 | 
			
		||||
            version in SUPPORTED_PLATFORMIO_ESP_IDF_5X
 | 
			
		||||
            and version not in SUPPORTED_PIOARDUINO_ESP_IDF_5X
 | 
			
		||||
        )
 | 
			
		||||
    ) and not has_platform_ver:
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"ESP-IDF {value[CONF_VERSION]} may be supported by platformio/espressif32; please specify '{CONF_PLATFORM_VERSION}'"
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
        not is_platformio
 | 
			
		||||
        and CONF_RELEASE not in value
 | 
			
		||||
        and version not in SUPPORTED_PIOARDUINO_ESP_IDF_5X
 | 
			
		||||
    ):
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"ESP-IDF {value[CONF_VERSION]} is not available with pioarduino; you may need to specify '{CONF_RELEASE}'"
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    value[CONF_VERSION] = str(version)
 | 
			
		||||
    value[CONF_SOURCE] = source or _format_framework_espidf_version(
 | 
			
		||||
        version, value.get(CONF_RELEASE, None), is_platformio
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    if value[CONF_SOURCE].startswith("http"):
 | 
			
		||||
        # prefix is necessary or platformio will complain with a cryptic error
 | 
			
		||||
        value[CONF_SOURCE] = f"framework-espidf@{value[CONF_SOURCE]}"
 | 
			
		||||
 | 
			
		||||
    if version != RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION:
 | 
			
		||||
        _LOGGER.warning(
 | 
			
		||||
            "The selected ESP-IDF framework version is not the recommended one. "
 | 
			
		||||
@@ -323,6 +391,12 @@ def _esp_idf_check_versions(value):
 | 
			
		||||
 | 
			
		||||
def _parse_platform_version(value):
 | 
			
		||||
    try:
 | 
			
		||||
        ver = cv.Version.parse(cv.version_number(value))
 | 
			
		||||
        if ver.major >= 50:  # a pioarduino version
 | 
			
		||||
            if "-" in value:
 | 
			
		||||
                # maybe a release candidate?...definitely not our default, just use it as-is...
 | 
			
		||||
                return f"https://github.com/pioarduino/platform-espressif32.git#{value}"
 | 
			
		||||
            return f"https://github.com/pioarduino/platform-espressif32.git#{ver.major}.{ver.minor:02d}.{ver.patch:02d}"
 | 
			
		||||
        # if platform version is a valid version constraint, prefix the default package
 | 
			
		||||
        cv.platformio_version_constraint(value)
 | 
			
		||||
        return f"platformio/espressif32@{value}"
 | 
			
		||||
@@ -330,6 +404,14 @@ def _parse_platform_version(value):
 | 
			
		||||
        return value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _platform_is_platformio(value):
 | 
			
		||||
    try:
 | 
			
		||||
        ver = cv.Version.parse(cv.version_number(value))
 | 
			
		||||
        return ver.major < 50
 | 
			
		||||
    except cv.Invalid:
 | 
			
		||||
        return "platformio" in value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _detect_variant(value):
 | 
			
		||||
    board = value[CONF_BOARD]
 | 
			
		||||
    if board in BOARDS:
 | 
			
		||||
@@ -355,24 +437,20 @@ def _detect_variant(value):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def final_validate(config):
 | 
			
		||||
    if CONF_PLATFORMIO_OPTIONS not in fv.full_config.get()[CONF_ESPHOME]:
 | 
			
		||||
    if not (
 | 
			
		||||
        pio_options := fv.full_config.get()[CONF_ESPHOME].get(CONF_PLATFORMIO_OPTIONS)
 | 
			
		||||
    ):
 | 
			
		||||
        # Not specified or empty
 | 
			
		||||
        return config
 | 
			
		||||
 | 
			
		||||
    pio_flash_size_key = "board_upload.flash_size"
 | 
			
		||||
    pio_partitions_key = "board_build.partitions"
 | 
			
		||||
    if (
 | 
			
		||||
        CONF_PARTITIONS in config
 | 
			
		||||
        and pio_partitions_key
 | 
			
		||||
        in fv.full_config.get()[CONF_ESPHOME][CONF_PLATFORMIO_OPTIONS]
 | 
			
		||||
    ):
 | 
			
		||||
    if CONF_PARTITIONS in config and pio_partitions_key in pio_options:
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"Do not specify '{pio_partitions_key}' in '{CONF_PLATFORMIO_OPTIONS}' with '{CONF_PARTITIONS}' in esp32"
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
        pio_flash_size_key
 | 
			
		||||
        in fv.full_config.get()[CONF_ESPHOME][CONF_PLATFORMIO_OPTIONS]
 | 
			
		||||
    ):
 | 
			
		||||
    if pio_flash_size_key in pio_options:
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"Please specify {CONF_FLASH_SIZE} within esp32 configuration only"
 | 
			
		||||
        )
 | 
			
		||||
@@ -412,6 +490,7 @@ ESP_IDF_FRAMEWORK_SCHEMA = cv.All(
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
            cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
 | 
			
		||||
            cv.Optional(CONF_RELEASE): cv.string_strict,
 | 
			
		||||
            cv.Optional(CONF_SOURCE): cv.string_strict,
 | 
			
		||||
            cv.Optional(CONF_PLATFORM_VERSION): _parse_platform_version,
 | 
			
		||||
            cv.Optional(CONF_SDKCONFIG_OPTIONS, default={}): {
 | 
			
		||||
@@ -515,10 +594,9 @@ async def to_code(config):
 | 
			
		||||
        cg.add_build_flag("-DUSE_ESP_IDF")
 | 
			
		||||
        cg.add_build_flag("-DUSE_ESP32_FRAMEWORK_ESP_IDF")
 | 
			
		||||
        cg.add_build_flag("-Wno-nonnull-compare")
 | 
			
		||||
        cg.add_platformio_option(
 | 
			
		||||
            "platform_packages",
 | 
			
		||||
            [f"platformio/framework-espidf@{conf[CONF_SOURCE]}"],
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
        cg.add_platformio_option("platform_packages", [conf[CONF_SOURCE]])
 | 
			
		||||
 | 
			
		||||
        # platformio/toolchain-esp32ulp does not support linux_aarch64 yet and has not been updated for over 2 years
 | 
			
		||||
        # This is espressif's own published version which is more up to date.
 | 
			
		||||
        cg.add_platformio_option(
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,12 @@
 | 
			
		||||
from .const import VARIANT_ESP32, VARIANT_ESP32C3, VARIANT_ESP32S2, VARIANT_ESP32S3
 | 
			
		||||
from .const import (
 | 
			
		||||
    VARIANT_ESP32,
 | 
			
		||||
    VARIANT_ESP32C2,
 | 
			
		||||
    VARIANT_ESP32C3,
 | 
			
		||||
    VARIANT_ESP32C6,
 | 
			
		||||
    VARIANT_ESP32H2,
 | 
			
		||||
    VARIANT_ESP32S2,
 | 
			
		||||
    VARIANT_ESP32S3,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
ESP32_BASE_PINS = {
 | 
			
		||||
    "TX": 1,
 | 
			
		||||
@@ -103,6 +111,173 @@ ESP32_BOARD_PINS = {
 | 
			
		||||
        "LED": 13,
 | 
			
		||||
        "LED_BUILTIN": 13,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s3": {
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "A0": 18,
 | 
			
		||||
        "A1": 17,
 | 
			
		||||
        "A2": 16,
 | 
			
		||||
        "A3": 15,
 | 
			
		||||
        "A4": 14,
 | 
			
		||||
        "A5": 8,
 | 
			
		||||
        "SCK": 36,
 | 
			
		||||
        "MOSI": 35,
 | 
			
		||||
        "MISO": 37,
 | 
			
		||||
        "RX": 38,
 | 
			
		||||
        "TX": 39,
 | 
			
		||||
        "SCL": 4,
 | 
			
		||||
        "SDA": 3,
 | 
			
		||||
        "NEOPIXEL": 33,
 | 
			
		||||
        "PIN_NEOPIXEL": 33,
 | 
			
		||||
        "NEOPIXEL_POWER": 21,
 | 
			
		||||
        "I2C_POWER": 7,
 | 
			
		||||
        "LED": 13,
 | 
			
		||||
        "LED_BUILTIN": 13,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s3_nopsram": {
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "A0": 18,
 | 
			
		||||
        "A1": 17,
 | 
			
		||||
        "A2": 16,
 | 
			
		||||
        "A3": 15,
 | 
			
		||||
        "A4": 14,
 | 
			
		||||
        "A5": 8,
 | 
			
		||||
        "SCK": 36,
 | 
			
		||||
        "MOSI": 35,
 | 
			
		||||
        "MISO": 37,
 | 
			
		||||
        "RX": 38,
 | 
			
		||||
        "TX": 39,
 | 
			
		||||
        "SCL": 4,
 | 
			
		||||
        "SDA": 3,
 | 
			
		||||
        "NEOPIXEL": 33,
 | 
			
		||||
        "PIN_NEOPIXEL": 33,
 | 
			
		||||
        "NEOPIXEL_POWER": 21,
 | 
			
		||||
        "I2C_POWER": 7,
 | 
			
		||||
        "LED": 13,
 | 
			
		||||
        "LED_BUILTIN": 13,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s3_tft": {
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "A0": 18,
 | 
			
		||||
        "A1": 17,
 | 
			
		||||
        "A2": 16,
 | 
			
		||||
        "A3": 15,
 | 
			
		||||
        "A4": 14,
 | 
			
		||||
        "A5": 8,
 | 
			
		||||
        "SCK": 36,
 | 
			
		||||
        "MOSI": 35,
 | 
			
		||||
        "MISO": 37,
 | 
			
		||||
        "RX": 2,
 | 
			
		||||
        "TX": 1,
 | 
			
		||||
        "SCL": 41,
 | 
			
		||||
        "SDA": 42,
 | 
			
		||||
        "NEOPIXEL": 33,
 | 
			
		||||
        "PIN_NEOPIXEL": 33,
 | 
			
		||||
        "NEOPIXEL_POWER": 34,
 | 
			
		||||
        "TFT_I2C_POWER": 21,
 | 
			
		||||
        "TFT_CS": 7,
 | 
			
		||||
        "TFT_DC": 39,
 | 
			
		||||
        "TFT_RESET": 40,
 | 
			
		||||
        "TFT_BACKLIGHT": 45,
 | 
			
		||||
        "LED": 13,
 | 
			
		||||
        "LED_BUILTIN": 13,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_funhouse_esp32s2": {
 | 
			
		||||
        "BUTTON_UP": 5,
 | 
			
		||||
        "BUTTON_DOWN": 3,
 | 
			
		||||
        "BUTTON_SELECT": 4,
 | 
			
		||||
        "DOTSTAR_DATA": 14,
 | 
			
		||||
        "DOTSTAR_CLOCK": 15,
 | 
			
		||||
        "PIR_SENSE": 16,
 | 
			
		||||
        "A0": 17,
 | 
			
		||||
        "A1": 2,
 | 
			
		||||
        "A2": 1,
 | 
			
		||||
        "CAP6": 6,
 | 
			
		||||
        "CAP7": 7,
 | 
			
		||||
        "CAP8": 8,
 | 
			
		||||
        "CAP9": 9,
 | 
			
		||||
        "CAP10": 10,
 | 
			
		||||
        "CAP11": 11,
 | 
			
		||||
        "CAP12": 12,
 | 
			
		||||
        "CAP13": 13,
 | 
			
		||||
        "SPEAKER": 42,
 | 
			
		||||
        "LED": 37,
 | 
			
		||||
        "LIGHT": 18,
 | 
			
		||||
        "TFT_MOSI": 35,
 | 
			
		||||
        "TFT_SCK": 36,
 | 
			
		||||
        "TFT_CS": 40,
 | 
			
		||||
        "TFT_DC": 39,
 | 
			
		||||
        "TFT_RESET": 41,
 | 
			
		||||
        "TFT_BACKLIGHT": 21,
 | 
			
		||||
        "RED_LED": 31,
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_itsybitsy_esp32": {
 | 
			
		||||
        "A0": 25,
 | 
			
		||||
        "A1": 26,
 | 
			
		||||
        "A2": 4,
 | 
			
		||||
        "A3": 38,
 | 
			
		||||
        "A4": 37,
 | 
			
		||||
        "A5": 36,
 | 
			
		||||
        "SCK": 19,
 | 
			
		||||
        "MOSI": 21,
 | 
			
		||||
        "MISO": 22,
 | 
			
		||||
        "SCL": 27,
 | 
			
		||||
        "SDA": 15,
 | 
			
		||||
        "TX": 20,
 | 
			
		||||
        "RX": 8,
 | 
			
		||||
        "NEOPIXEL": 0,
 | 
			
		||||
        "PIN_NEOPIXEL": 0,
 | 
			
		||||
        "NEOPIXEL_POWER": 2,
 | 
			
		||||
        "BUTTON": 35,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_magtag29_esp32s2": {
 | 
			
		||||
        "A1": 18,
 | 
			
		||||
        "BUTTON_A": 15,
 | 
			
		||||
        "BUTTON_B": 14,
 | 
			
		||||
        "BUTTON_C": 12,
 | 
			
		||||
        "BUTTON_D": 11,
 | 
			
		||||
        "SDA": 33,
 | 
			
		||||
        "SCL": 34,
 | 
			
		||||
        "SPEAKER": 17,
 | 
			
		||||
        "SPEAKER_ENABLE": 16,
 | 
			
		||||
        "VOLTAGE_MONITOR": 4,
 | 
			
		||||
        "ACCELEROMETER_INT": 9,
 | 
			
		||||
        "ACCELEROMETER_INTERRUPT": 9,
 | 
			
		||||
        "LIGHT": 3,
 | 
			
		||||
        "NEOPIXEL": 1,
 | 
			
		||||
        "PIN_NEOPIXEL": 1,
 | 
			
		||||
        "NEOPIXEL_POWER": 21,
 | 
			
		||||
        "EPD_BUSY": 5,
 | 
			
		||||
        "EPD_RESET": 6,
 | 
			
		||||
        "EPD_DC": 7,
 | 
			
		||||
        "EPD_CS": 8,
 | 
			
		||||
        "EPD_MOSI": 35,
 | 
			
		||||
        "EPD_SCK": 36,
 | 
			
		||||
        "EPD_MISO": 37,
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "LED": 13,
 | 
			
		||||
        "LED_BUILTIN": 13,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_metro_esp32s2": {
 | 
			
		||||
        "A0": 17,
 | 
			
		||||
        "A1": 18,
 | 
			
		||||
        "A2": 1,
 | 
			
		||||
        "A3": 2,
 | 
			
		||||
        "A4": 3,
 | 
			
		||||
        "A5": 4,
 | 
			
		||||
        "RX": 38,
 | 
			
		||||
        "TX": 37,
 | 
			
		||||
        "SCL": 34,
 | 
			
		||||
        "SDA": 33,
 | 
			
		||||
        "MISO": 37,
 | 
			
		||||
        "SCK": 36,
 | 
			
		||||
        "MOSI": 35,
 | 
			
		||||
        "NEOPIXEL": 45,
 | 
			
		||||
        "PIN_NEOPIXEL": 45,
 | 
			
		||||
        "LED": 42,
 | 
			
		||||
        "LED_BUILTIN": 42,
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32c3": {
 | 
			
		||||
        "A0": 4,
 | 
			
		||||
        "A1": 3,
 | 
			
		||||
@@ -141,6 +316,26 @@ ESP32_BOARD_PINS = {
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "SWITCH": 0,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32s3_nopsram": {
 | 
			
		||||
        "A0": 18,
 | 
			
		||||
        "A1": 17,
 | 
			
		||||
        "A2": 9,
 | 
			
		||||
        "A3": 8,
 | 
			
		||||
        "SDA": 7,
 | 
			
		||||
        "SCL": 6,
 | 
			
		||||
        "MOSI": 35,
 | 
			
		||||
        "MISO": 37,
 | 
			
		||||
        "SCK": 36,
 | 
			
		||||
        "RX": 16,
 | 
			
		||||
        "TX": 5,
 | 
			
		||||
        "SDA1": 41,
 | 
			
		||||
        "SCL1": 40,
 | 
			
		||||
        "NEOPIXEL": 39,
 | 
			
		||||
        "PIN_NEOPIXEL": 39,
 | 
			
		||||
        "NEOPIXEL_POWER": 38,
 | 
			
		||||
        "BUTTON": 0,
 | 
			
		||||
        "SWITCH": 0,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32": {
 | 
			
		||||
        "A0": 26,
 | 
			
		||||
        "A1": 25,
 | 
			
		||||
@@ -1068,7 +1263,18 @@ ESP32_BOARD_PINS = {
 | 
			
		||||
        "_VBAT": 35,
 | 
			
		||||
    },
 | 
			
		||||
    "wemosbat": {"LED": 16},
 | 
			
		||||
    "wesp32": {"MISO": 32, "SCL": 4, "SDA": 15},
 | 
			
		||||
    "wesp32": {
 | 
			
		||||
        "MISO": 32,
 | 
			
		||||
        "MOSI": 23,
 | 
			
		||||
        "SCK": 18,
 | 
			
		||||
        "SCL": 4,
 | 
			
		||||
        "SDA": 15,
 | 
			
		||||
        "MISO1": 12,
 | 
			
		||||
        "MOSI1": 13,
 | 
			
		||||
        "SCK1": 14,
 | 
			
		||||
        "SCL1": 5,
 | 
			
		||||
        "SDA1": 33,
 | 
			
		||||
    },
 | 
			
		||||
    "widora-air": {
 | 
			
		||||
        "A1": 39,
 | 
			
		||||
        "A2": 35,
 | 
			
		||||
@@ -1146,6 +1352,26 @@ done | sort
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
BOARDS = {
 | 
			
		||||
    "4d_systems_esp32s3_gen4_r8n16": {
 | 
			
		||||
        "name": "4D Systems GEN4-ESP32 16MB (ESP32S3-R8N16)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_camera_esp32s3": {
 | 
			
		||||
        "name": "Adafruit pyCamera S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32c6": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-C6",
 | 
			
		||||
        "variant": VARIANT_ESP32C6,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s2": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s2_reversetft": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S2 Reverse TFT",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s2_tft": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S2 TFT",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
@@ -1158,6 +1384,10 @@ BOARDS = {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S3 No PSRAM",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s3_reversetft": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S3 Reverse TFT",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_feather_esp32s3_tft": {
 | 
			
		||||
        "name": "Adafruit Feather ESP32-S3 TFT",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
@@ -1178,10 +1408,18 @@ BOARDS = {
 | 
			
		||||
        "name": "Adafruit MagTag 2.9",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_matrixportal_esp32s3": {
 | 
			
		||||
        "name": "Adafruit MatrixPortal ESP32-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_metro_esp32s2": {
 | 
			
		||||
        "name": "Adafruit Metro ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_metro_esp32s3": {
 | 
			
		||||
        "name": "Adafruit Metro ESP32-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32c3": {
 | 
			
		||||
        "name": "Adafruit QT Py ESP32-C3",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
@@ -1194,10 +1432,18 @@ BOARDS = {
 | 
			
		||||
        "name": "Adafruit QT Py ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32s3_n4r2": {
 | 
			
		||||
        "name": "Adafruit QT Py ESP32-S3 (4M Flash 2M PSRAM)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qtpy_esp32s3_nopsram": {
 | 
			
		||||
        "name": "Adafruit QT Py ESP32-S3 No PSRAM",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "adafruit_qualia_s3_rgb666": {
 | 
			
		||||
        "name": "Adafruit Qualia ESP32-S3 RGB666",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "airm2m_core_esp32c3": {
 | 
			
		||||
        "name": "AirM2M CORE ESP32C3",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
@@ -1206,14 +1452,30 @@ BOARDS = {
 | 
			
		||||
        "name": "ALKS ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "arduino_nano_esp32": {
 | 
			
		||||
        "name": "Arduino Nano ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "atd147_s3": {
 | 
			
		||||
        "name": "ArtronShop ATD1.47-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "atmegazero_esp32s2": {
 | 
			
		||||
        "name": "EspinalLab ATMegaZero ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "aventen_s3_sync": {
 | 
			
		||||
        "name": "Aventen S3 Sync",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "az-delivery-devkit-v4": {
 | 
			
		||||
        "name": "AZ-Delivery ESP-32 Dev Kit C V4",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "bee_data_logger": {
 | 
			
		||||
        "name": "Smart Bee Data Logger",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "bee_motion_mini": {
 | 
			
		||||
        "name": "Smart Bee Motion Mini",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
@@ -1238,14 +1500,6 @@ BOARDS = {
 | 
			
		||||
        "name": "BPI-Leaf-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "briki_abc_esp32": {
 | 
			
		||||
        "name": "Briki ABC (MBC-WB) - ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "briki_mbc-wb_esp32": {
 | 
			
		||||
        "name": "Briki MBC-WB - ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "cnrs_aw2eth": {
 | 
			
		||||
        "name": "CNRS AW2ETH",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1298,18 +1552,38 @@ BOARDS = {
 | 
			
		||||
        "name": "DFRobot Beetle ESP32-C3",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "dfrobot_firebeetle2_esp32e": {
 | 
			
		||||
        "name": "DFRobot Firebeetle 2 ESP32-E",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "dfrobot_firebeetle2_esp32s3": {
 | 
			
		||||
        "name": "DFRobot Firebeetle 2 ESP32-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "dfrobot_romeo_esp32s3": {
 | 
			
		||||
        "name": "DFRobot Romeo ESP32-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "dpu_esp32": {
 | 
			
		||||
        "name": "TAMC DPU ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "edgebox-esp-100": {
 | 
			
		||||
        "name": "Seeed Studio Edgebox-ESP-100",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp320": {
 | 
			
		||||
        "name": "Electronic SweetPeas ESP320",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-c2-devkitm-1": {
 | 
			
		||||
        "name": "Espressif ESP32-C2-DevKitM-1",
 | 
			
		||||
        "variant": VARIANT_ESP32C2,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-c3-devkitc-02": {
 | 
			
		||||
        "name": "Espressif ESP32-C3-DevKitC-02",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-c3-devkitm-1": {
 | 
			
		||||
        "name": "Espressif ESP32-C3-DevKitM-1",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
@@ -1318,6 +1592,14 @@ BOARDS = {
 | 
			
		||||
        "name": "Ai-Thinker ESP-C3-M1-I-Kit",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-c6-devkitc-1": {
 | 
			
		||||
        "name": "Espressif ESP32-C6-DevKitC-1",
 | 
			
		||||
        "variant": VARIANT_ESP32C6,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-c6-devkitm-1": {
 | 
			
		||||
        "name": "Espressif ESP32-C6-DevKitM-1",
 | 
			
		||||
        "variant": VARIANT_ESP32C6,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32cam": {
 | 
			
		||||
        "name": "AI Thinker ESP32-CAM",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1346,6 +1628,14 @@ BOARDS = {
 | 
			
		||||
        "name": "OLIMEX ESP32-GATEWAY",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-h2-devkitm-1": {
 | 
			
		||||
        "name": "Espressif ESP32-H2-DevKit",
 | 
			
		||||
        "variant": VARIANT_ESP32H2,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-pico-devkitm-2": {
 | 
			
		||||
        "name": "Espressif ESP32-PICO-DevKitM-2",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-poe-iso": {
 | 
			
		||||
        "name": "OLIMEX ESP32-PoE-ISO",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1382,10 +1672,22 @@ BOARDS = {
 | 
			
		||||
        "name": "Espressif ESP32-S3-DevKitC-1-N8 (8 MB QD, No PSRAM)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-s3-korvo-2": {
 | 
			
		||||
        "name": "Espressif ESP32-S3-Korvo-2",
 | 
			
		||||
    "esp32-s3-devkitm-1": {
 | 
			
		||||
        "name": "Espressif ESP32-S3-DevKitM-1",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32s3_powerfeather": {
 | 
			
		||||
        "name": "ESP32-S3 PowerFeather",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32s3usbotg": {
 | 
			
		||||
        "name": "Espressif ESP32-S3-USB-OTG",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32-solo1": {
 | 
			
		||||
        "name": "Espressif Generic ESP32-solo1 4M Flash",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "esp32thing": {
 | 
			
		||||
        "name": "SparkFun ESP32 Thing",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1454,9 +1756,9 @@ BOARDS = {
 | 
			
		||||
        "name": "Heltec WiFi Kit 32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "heltec_wifi_kit_32_v2": {
 | 
			
		||||
        "name": "Heltec WiFi Kit 32 (V2)",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    "heltec_wifi_kit_32_V3": {
 | 
			
		||||
        "name": "Heltec WiFi Kit 32 (V3)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "heltec_wifi_lora_32": {
 | 
			
		||||
        "name": "Heltec WiFi LoRa 32",
 | 
			
		||||
@@ -1466,6 +1768,10 @@ BOARDS = {
 | 
			
		||||
        "name": "Heltec WiFi LoRa 32 (V2)",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "heltec_wifi_lora_32_V3": {
 | 
			
		||||
        "name": "Heltec WiFi LoRa 32 (V3)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "heltec_wireless_stick_lite": {
 | 
			
		||||
        "name": "Heltec Wireless Stick Lite",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1510,6 +1816,14 @@ BOARDS = {
 | 
			
		||||
        "name": "oddWires IoT-Bus Proteus",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "ioxesp32": {
 | 
			
		||||
        "name": "ArtronShop IOXESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "ioxesp32ps": {
 | 
			
		||||
        "name": "ArtronShop IOXESP32PS",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "kb32-ft": {
 | 
			
		||||
        "name": "MakerAsia KB32-FT",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1522,10 +1836,26 @@ BOARDS = {
 | 
			
		||||
        "name": "Labplus mPython",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "lilka_v2": {
 | 
			
		||||
        "name": "Lilka v2",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lilygo-t-display": {
 | 
			
		||||
        "name": "LilyGo T-Display",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "lilygo-t-display-s3": {
 | 
			
		||||
        "name": "LilyGo T-Display-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lionbit": {
 | 
			
		||||
        "name": "Lion:Bit Dev Board",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "lionbits3": {
 | 
			
		||||
        "name": "Lion:Bit S3 STEM Dev Board",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lolin32_lite": {
 | 
			
		||||
        "name": "WEMOS LOLIN32 Lite",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1554,10 +1884,18 @@ BOARDS = {
 | 
			
		||||
        "name": "WEMOS LOLIN S2 PICO",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "lolin_s3_mini": {
 | 
			
		||||
        "name": "WEMOS LOLIN S3 Mini",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lolin_s3": {
 | 
			
		||||
        "name": "WEMOS LOLIN S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lolin_s3_pro": {
 | 
			
		||||
        "name": "WEMOS LOLIN S3 PRO",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "lopy4": {
 | 
			
		||||
        "name": "Pycom LoPy4",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1570,10 +1908,18 @@ BOARDS = {
 | 
			
		||||
        "name": "M5Stack-ATOM",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-atoms3": {
 | 
			
		||||
        "name": "M5Stack AtomS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-core2": {
 | 
			
		||||
        "name": "M5Stack Core2",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-core-esp32-16M": {
 | 
			
		||||
        "name": "M5Stack Core ESP32 16M",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-core-esp32": {
 | 
			
		||||
        "name": "M5Stack Core ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1582,6 +1928,10 @@ BOARDS = {
 | 
			
		||||
        "name": "M5Stack-Core Ink",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-cores3": {
 | 
			
		||||
        "name": "M5Stack CoreS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-fire": {
 | 
			
		||||
        "name": "M5Stack FIRE",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1590,6 +1940,14 @@ BOARDS = {
 | 
			
		||||
        "name": "M5Stack GREY ESP32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack_paper": {
 | 
			
		||||
        "name": "M5Stack Paper",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-stamps3": {
 | 
			
		||||
        "name": "M5Stack StampS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stack-station": {
 | 
			
		||||
        "name": "M5Stack Station",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1598,6 +1956,10 @@ BOARDS = {
 | 
			
		||||
        "name": "M5Stack Timer CAM",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stamp-pico": {
 | 
			
		||||
        "name": "M5Stamp-Pico",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "m5stick-c": {
 | 
			
		||||
        "name": "M5Stick-C",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1634,10 +1996,26 @@ BOARDS = {
 | 
			
		||||
        "name": "Deparment of Alchemy MiniMain ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "motorgo_mini_1": {
 | 
			
		||||
        "name": "MotorGo Mini 1 (ESP32-S3)",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "namino_arancio": {
 | 
			
		||||
        "name": "Namino Arancio",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "namino_rosso": {
 | 
			
		||||
        "name": "Namino Rosso",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "nano32": {
 | 
			
		||||
        "name": "MakerAsia Nano32",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "nebulas3": {
 | 
			
		||||
        "name": "Kinetic Dynamics Nebula S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "nina_w10": {
 | 
			
		||||
        "name": "u-blox NINA-W10 series",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1698,10 +2076,22 @@ BOARDS = {
 | 
			
		||||
        "name": "Munich Labs RedPill ESP32-S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "roboheart_hercules": {
 | 
			
		||||
        "name": "RoboHeart Hercules",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "seeed_xiao_esp32c3": {
 | 
			
		||||
        "name": "Seeed Studio XIAO ESP32C3",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "seeed_xiao_esp32s3": {
 | 
			
		||||
        "name": "Seeed Studio XIAO ESP32S3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "sensebox_mcu_esp32s2": {
 | 
			
		||||
        "name": "senseBox MCU-S2 ESP32-S2",
 | 
			
		||||
        "variant": VARIANT_ESP32S2,
 | 
			
		||||
    },
 | 
			
		||||
    "sensesiot_weizen": {
 | 
			
		||||
        "name": "LOGISENSES Senses Weizen",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1714,6 +2104,10 @@ BOARDS = {
 | 
			
		||||
        "name": "S.ODI Ultra v1",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "sparkfun_esp32c6_thing_plus": {
 | 
			
		||||
        "name": "Sparkfun ESP32-C6 Thing Plus",
 | 
			
		||||
        "variant": VARIANT_ESP32C6,
 | 
			
		||||
    },
 | 
			
		||||
    "sparkfun_esp32_iot_redboard": {
 | 
			
		||||
        "name": "SparkFun ESP32 IoT RedBoard",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
@@ -1806,6 +2200,10 @@ BOARDS = {
 | 
			
		||||
        "name": "Unexpected Maker FeatherS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "um_nanos3": {
 | 
			
		||||
        "name": "Unexpected Maker NanoS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
    },
 | 
			
		||||
    "um_pros3": {
 | 
			
		||||
        "name": "Unexpected Maker PROS3",
 | 
			
		||||
        "variant": VARIANT_ESP32S3,
 | 
			
		||||
@@ -1842,6 +2240,14 @@ BOARDS = {
 | 
			
		||||
        "name": "uPesy ESP32 Wrover DevKit",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
    },
 | 
			
		||||
    "valtrack_v4_mfw_esp32_c3": {
 | 
			
		||||
        "name": "Valetron Systems VALTRACK-V4MVF",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "valtrack_v4_vts_esp32_c3": {
 | 
			
		||||
        "name": "Valetron Systems VALTRACK-V4VTS",
 | 
			
		||||
        "variant": VARIANT_ESP32C3,
 | 
			
		||||
    },
 | 
			
		||||
    "vintlabs-devkit-v1": {
 | 
			
		||||
        "name": "VintLabs ESP32 Devkit",
 | 
			
		||||
        "variant": VARIANT_ESP32,
 | 
			
		||||
 
 | 
			
		||||
@@ -67,8 +67,10 @@ def _translate_pin(value):
 | 
			
		||||
            "This variable only supports pin numbers, not full pin schemas "
 | 
			
		||||
            "(with inverted and mode)."
 | 
			
		||||
        )
 | 
			
		||||
    if isinstance(value, int):
 | 
			
		||||
    if isinstance(value, int) and not isinstance(value, bool):
 | 
			
		||||
        return value
 | 
			
		||||
    if not isinstance(value, str):
 | 
			
		||||
        raise cv.Invalid(f"Invalid pin number: {value}")
 | 
			
		||||
    try:
 | 
			
		||||
        return int(value)
 | 
			
		||||
    except ValueError:
 | 
			
		||||
 
 | 
			
		||||
@@ -2,8 +2,10 @@ from esphome import automation
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components.esp32 import add_idf_sdkconfig_option, const, get_esp32_variant
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_ENABLE_ON_BOOT, CONF_ID
 | 
			
		||||
from esphome.const import CONF_ENABLE_ON_BOOT, CONF_ESPHOME, CONF_ID, CONF_NAME
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
from esphome.core.config import CONF_NAME_ADD_MAC_SUFFIX
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
 | 
			
		||||
DEPENDENCIES = ["esp32"]
 | 
			
		||||
CODEOWNERS = ["@jesserockz", "@Rapsssito"]
 | 
			
		||||
@@ -50,6 +52,7 @@ TX_POWER_LEVELS = {
 | 
			
		||||
CONFIG_SCHEMA = cv.Schema(
 | 
			
		||||
    {
 | 
			
		||||
        cv.GenerateID(): cv.declare_id(ESP32BLE),
 | 
			
		||||
        cv.Optional(CONF_NAME): cv.All(cv.string, cv.Length(max=20)),
 | 
			
		||||
        cv.Optional(CONF_IO_CAPABILITY, default="none"): cv.enum(
 | 
			
		||||
            IO_CAPABILITY, lower=True
 | 
			
		||||
        ),
 | 
			
		||||
@@ -67,7 +70,22 @@ def validate_variant(_):
 | 
			
		||||
        raise cv.Invalid(f"{variant} does not support Bluetooth")
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FINAL_VALIDATE_SCHEMA = validate_variant
 | 
			
		||||
def final_validation(config):
 | 
			
		||||
    validate_variant(config)
 | 
			
		||||
    if (name := config.get(CONF_NAME)) is not None:
 | 
			
		||||
        full_config = fv.full_config.get()
 | 
			
		||||
        max_length = 20
 | 
			
		||||
        if full_config[CONF_ESPHOME][CONF_NAME_ADD_MAC_SUFFIX]:
 | 
			
		||||
            max_length -= 7  # "-AABBCC" is appended when add mac suffix option is used
 | 
			
		||||
        if len(name) > max_length:
 | 
			
		||||
            raise cv.Invalid(
 | 
			
		||||
                f"Name '{name}' is too long, maximum length is {max_length} characters"
 | 
			
		||||
            )
 | 
			
		||||
 | 
			
		||||
    return config
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FINAL_VALIDATE_SCHEMA = final_validation
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
@@ -75,6 +93,8 @@ async def to_code(config):
 | 
			
		||||
    cg.add(var.set_enable_on_boot(config[CONF_ENABLE_ON_BOOT]))
 | 
			
		||||
    cg.add(var.set_io_capability(config[CONF_IO_CAPABILITY]))
 | 
			
		||||
    cg.add(var.set_advertising_cycle_time(config[CONF_ADVERTISING_CYCLE_TIME]))
 | 
			
		||||
    if (name := config.get(CONF_NAME)) is not None:
 | 
			
		||||
        cg.add(var.set_name(name))
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
 | 
			
		||||
    if CORE.using_esp_idf:
 | 
			
		||||
 
 | 
			
		||||
@@ -188,12 +188,20 @@ bool ESP32BLE::ble_setup_() {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  std::string name = App.get_name();
 | 
			
		||||
  if (name.length() > 20) {
 | 
			
		||||
  std::string name;
 | 
			
		||||
  if (this->name_.has_value()) {
 | 
			
		||||
    name = this->name_.value();
 | 
			
		||||
    if (App.is_name_add_mac_suffix_enabled()) {
 | 
			
		||||
      name.erase(name.begin() + 13, name.end() - 7);  // Remove characters between 13 and the mac address
 | 
			
		||||
    } else {
 | 
			
		||||
      name = name.substr(0, 20);
 | 
			
		||||
      name += "-" + get_mac_address().substr(6);
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    name = App.get_name();
 | 
			
		||||
    if (name.length() > 20) {
 | 
			
		||||
      if (App.is_name_add_mac_suffix_enabled()) {
 | 
			
		||||
        name.erase(name.begin() + 13, name.end() - 7);  // Remove characters between 13 and the mac address
 | 
			
		||||
      } else {
 | 
			
		||||
        name = name.substr(0, 20);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -90,6 +90,7 @@ class ESP32BLE : public Component {
 | 
			
		||||
  void loop() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
  float get_setup_priority() const override;
 | 
			
		||||
  void set_name(const std::string &name) { this->name_ = name; }
 | 
			
		||||
 | 
			
		||||
  void advertising_start();
 | 
			
		||||
  void advertising_set_service_data(const std::vector<uint8_t> &data);
 | 
			
		||||
@@ -131,6 +132,7 @@ class ESP32BLE : public Component {
 | 
			
		||||
  esp_ble_io_cap_t io_cap_{ESP_IO_CAP_NONE};
 | 
			
		||||
  uint32_t advertising_cycle_time_;
 | 
			
		||||
  bool enable_on_boot_;
 | 
			
		||||
  optional<std::string> name_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
 
 | 
			
		||||
@@ -83,7 +83,7 @@ esp_err_t BLEAdvertising::services_advertisement_() {
 | 
			
		||||
  esp_err_t err;
 | 
			
		||||
 | 
			
		||||
  this->advertising_data_.set_scan_rsp = false;
 | 
			
		||||
  this->advertising_data_.include_name = !this->scan_response_;
 | 
			
		||||
  this->advertising_data_.include_name = true;
 | 
			
		||||
  this->advertising_data_.include_txpower = !this->scan_response_;
 | 
			
		||||
  err = esp_ble_gap_config_adv_data(&this->advertising_data_);
 | 
			
		||||
  if (err != ESP_OK) {
 | 
			
		||||
 
 | 
			
		||||
@@ -34,7 +34,7 @@ ESPBTUUID ESPBTUUID::from_raw(const uint8_t *data) {
 | 
			
		||||
ESPBTUUID ESPBTUUID::from_raw_reversed(const uint8_t *data) {
 | 
			
		||||
  ESPBTUUID ret;
 | 
			
		||||
  ret.uuid_.len = ESP_UUID_LEN_128;
 | 
			
		||||
  for (int i = 0; i < ESP_UUID_LEN_128; i++)
 | 
			
		||||
  for (uint8_t i = 0; i < ESP_UUID_LEN_128; i++)
 | 
			
		||||
    ret.uuid_.uuid.uuid128[ESP_UUID_LEN_128 - 1 - i] = data[i];
 | 
			
		||||
  return ret;
 | 
			
		||||
}
 | 
			
		||||
@@ -43,30 +43,30 @@ ESPBTUUID ESPBTUUID::from_raw(const std::string &data) {
 | 
			
		||||
  if (data.length() == 4) {
 | 
			
		||||
    ret.uuid_.len = ESP_UUID_LEN_16;
 | 
			
		||||
    ret.uuid_.uuid.uuid16 = 0;
 | 
			
		||||
    for (int i = 0; i < data.length();) {
 | 
			
		||||
    for (uint i = 0; i < data.length(); i += 2) {
 | 
			
		||||
      uint8_t msb = data.c_str()[i];
 | 
			
		||||
      uint8_t lsb = data.c_str()[i + 1];
 | 
			
		||||
      uint8_t lsb_shift = i <= 2 ? (2 - i) * 4 : 0;
 | 
			
		||||
 | 
			
		||||
      if (msb > '9')
 | 
			
		||||
        msb -= 7;
 | 
			
		||||
      if (lsb > '9')
 | 
			
		||||
        lsb -= 7;
 | 
			
		||||
      ret.uuid_.uuid.uuid16 += (((msb & 0x0F) << 4) | (lsb & 0x0F)) << (2 - i) * 4;
 | 
			
		||||
      i += 2;
 | 
			
		||||
      ret.uuid_.uuid.uuid16 += (((msb & 0x0F) << 4) | (lsb & 0x0F)) << lsb_shift;
 | 
			
		||||
    }
 | 
			
		||||
  } else if (data.length() == 8) {
 | 
			
		||||
    ret.uuid_.len = ESP_UUID_LEN_32;
 | 
			
		||||
    ret.uuid_.uuid.uuid32 = 0;
 | 
			
		||||
    for (int i = 0; i < data.length();) {
 | 
			
		||||
    for (uint i = 0; i < data.length(); i += 2) {
 | 
			
		||||
      uint8_t msb = data.c_str()[i];
 | 
			
		||||
      uint8_t lsb = data.c_str()[i + 1];
 | 
			
		||||
      uint8_t lsb_shift = i <= 6 ? (6 - i) * 4 : 0;
 | 
			
		||||
 | 
			
		||||
      if (msb > '9')
 | 
			
		||||
        msb -= 7;
 | 
			
		||||
      if (lsb > '9')
 | 
			
		||||
        lsb -= 7;
 | 
			
		||||
      ret.uuid_.uuid.uuid32 += (((msb & 0x0F) << 4) | (lsb & 0x0F)) << (6 - i) * 4;
 | 
			
		||||
      i += 2;
 | 
			
		||||
      ret.uuid_.uuid.uuid32 += (((msb & 0x0F) << 4) | (lsb & 0x0F)) << lsb_shift;
 | 
			
		||||
    }
 | 
			
		||||
  } else if (data.length() == 16) {  // how we can have 16 byte length string reprezenting 128 bit uuid??? needs to be
 | 
			
		||||
                                     // investigated (lack of time)
 | 
			
		||||
@@ -77,7 +77,7 @@ ESPBTUUID ESPBTUUID::from_raw(const std::string &data) {
 | 
			
		||||
    // UUID format.
 | 
			
		||||
    ret.uuid_.len = ESP_UUID_LEN_128;
 | 
			
		||||
    int n = 0;
 | 
			
		||||
    for (int i = 0; i < data.length();) {
 | 
			
		||||
    for (uint i = 0; i < data.length(); i += 2) {
 | 
			
		||||
      if (data.c_str()[i] == '-')
 | 
			
		||||
        i++;
 | 
			
		||||
      uint8_t msb = data.c_str()[i];
 | 
			
		||||
@@ -88,7 +88,6 @@ ESPBTUUID ESPBTUUID::from_raw(const std::string &data) {
 | 
			
		||||
      if (lsb > '9')
 | 
			
		||||
        lsb -= 7;
 | 
			
		||||
      ret.uuid_.uuid.uuid128[15 - n++] = ((msb & 0x0F) << 4) | (lsb & 0x0F);
 | 
			
		||||
      i += 2;
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    ESP_LOGE(TAG, "ERROR: UUID value not 2, 4, 16 or 36 bytes - %s", data.c_str());
 | 
			
		||||
@@ -155,7 +154,7 @@ bool ESPBTUUID::operator==(const ESPBTUUID &uuid) const {
 | 
			
		||||
        }
 | 
			
		||||
        break;
 | 
			
		||||
      case ESP_UUID_LEN_128:
 | 
			
		||||
        for (int i = 0; i < ESP_UUID_LEN_128; i++) {
 | 
			
		||||
        for (uint8_t i = 0; i < ESP_UUID_LEN_128; i++) {
 | 
			
		||||
          if (uuid.uuid_.uuid.uuid128[i] != this->uuid_.uuid.uuid128[i]) {
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
 
 | 
			
		||||
@@ -40,6 +40,9 @@ static const esp_bt_controller_config_t BT_CONTROLLER_CONFIG = {
 | 
			
		||||
    .controller_run_cpu = 0,
 | 
			
		||||
    .enable_qa_test = RUN_QA_TEST,
 | 
			
		||||
    .enable_bqb_test = RUN_BQB_TEST,
 | 
			
		||||
#if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 3, 1)
 | 
			
		||||
    // The following fields have been removed since ESP IDF version 5.3.1, see commit:
 | 
			
		||||
    // https://github.com/espressif/esp-idf/commit/e761c1de8f9c0777829d597b4d5a33bb070a30a8
 | 
			
		||||
    .enable_uart_hci = HCI_UART_EN,
 | 
			
		||||
    .ble_hci_uart_port = DEFAULT_BT_LE_HCI_UART_PORT,
 | 
			
		||||
    .ble_hci_uart_baud = DEFAULT_BT_LE_HCI_UART_BAUD,
 | 
			
		||||
@@ -47,6 +50,7 @@ static const esp_bt_controller_config_t BT_CONTROLLER_CONFIG = {
 | 
			
		||||
    .ble_hci_uart_stop_bits = DEFAULT_BT_LE_HCI_UART_STOP_BITS,
 | 
			
		||||
    .ble_hci_uart_flow_ctrl = DEFAULT_BT_LE_HCI_UART_FLOW_CTRL,
 | 
			
		||||
    .ble_hci_uart_uart_parity = DEFAULT_BT_LE_HCI_UART_PARITY,
 | 
			
		||||
#endif
 | 
			
		||||
    .enable_tx_cca = DEFAULT_BT_LE_TX_CCA_ENABLED,
 | 
			
		||||
    .cca_rssi_thresh = 256 - DEFAULT_BT_LE_CCA_RSSI_THRESH,
 | 
			
		||||
    .sleep_en = NIMBLE_SLEEP_ENABLE,
 | 
			
		||||
@@ -58,6 +62,9 @@ static const esp_bt_controller_config_t BT_CONTROLLER_CONFIG = {
 | 
			
		||||
    .cpu_freq_mhz = CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ,
 | 
			
		||||
    .ignore_wl_for_direct_adv = 0,
 | 
			
		||||
    .enable_pcl = DEFAULT_BT_LE_POWER_CONTROL_ENABLED,
 | 
			
		||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 1, 3)
 | 
			
		||||
    .csa2_select = DEFAULT_BT_LE_50_FEATURE_SUPPORT,
 | 
			
		||||
#endif
 | 
			
		||||
    .config_magic = CONFIG_MAGIC,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -35,7 +35,7 @@ class BLEClientBase : public espbt::ESPBTClient, public Component {
 | 
			
		||||
  void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override;
 | 
			
		||||
  void connect() override;
 | 
			
		||||
  esp_err_t pair();
 | 
			
		||||
  void disconnect();
 | 
			
		||||
  void disconnect() override;
 | 
			
		||||
  void release_services();
 | 
			
		||||
 | 
			
		||||
  bool connected() { return this->state_ == espbt::ClientState::ESTABLISHED; }
 | 
			
		||||
 
 | 
			
		||||
@@ -65,6 +65,9 @@ void ESP32BLETracker::setup() {
 | 
			
		||||
      [this](ota::OTAState state, float progress, uint8_t error, ota::OTAComponent *comp) {
 | 
			
		||||
        if (state == ota::OTA_STARTED) {
 | 
			
		||||
          this->stop_scan();
 | 
			
		||||
          for (auto *client : this->clients_) {
 | 
			
		||||
            client->disconnect();
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
#endif
 | 
			
		||||
@@ -429,7 +432,7 @@ void ESPBTDevice::parse_scan_rst(const esp_ble_gap_cb_param_t::ble_scan_result_e
 | 
			
		||||
 | 
			
		||||
#ifdef ESPHOME_LOG_HAS_VERY_VERBOSE
 | 
			
		||||
  ESP_LOGVV(TAG, "Parse Result:");
 | 
			
		||||
  const char *address_type = "";
 | 
			
		||||
  const char *address_type;
 | 
			
		||||
  switch (this->address_type_) {
 | 
			
		||||
    case BLE_ADDR_TYPE_PUBLIC:
 | 
			
		||||
      address_type = "PUBLIC";
 | 
			
		||||
@@ -443,6 +446,9 @@ void ESPBTDevice::parse_scan_rst(const esp_ble_gap_cb_param_t::ble_scan_result_e
 | 
			
		||||
    case BLE_ADDR_TYPE_RPA_RANDOM:
 | 
			
		||||
      address_type = "RPA_RANDOM";
 | 
			
		||||
      break;
 | 
			
		||||
    default:
 | 
			
		||||
      address_type = "UNKNOWN";
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGVV(TAG, "  Address: %02X:%02X:%02X:%02X:%02X:%02X (%s)", this->address_[0], this->address_[1],
 | 
			
		||||
            this->address_[2], this->address_[3], this->address_[4], this->address_[5], address_type);
 | 
			
		||||
 
 | 
			
		||||
@@ -11,9 +11,9 @@
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
 | 
			
		||||
#include <esp_bt_defs.h>
 | 
			
		||||
#include <esp_gap_ble_api.h>
 | 
			
		||||
#include <esp_gattc_api.h>
 | 
			
		||||
#include <esp_bt_defs.h>
 | 
			
		||||
 | 
			
		||||
#include <freertos/FreeRTOS.h>
 | 
			
		||||
#include <freertos/semphr.h>
 | 
			
		||||
@@ -172,6 +172,7 @@ class ESPBTClient : public ESPBTDeviceListener {
 | 
			
		||||
                                   esp_ble_gattc_cb_param_t *param) = 0;
 | 
			
		||||
  virtual void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) = 0;
 | 
			
		||||
  virtual void connect() = 0;
 | 
			
		||||
  virtual void disconnect() = 0;
 | 
			
		||||
  virtual void set_state(ClientState st) { this->state_ = st; }
 | 
			
		||||
  ClientState state() const { return state_; }
 | 
			
		||||
  int app_id;
 | 
			
		||||
 
 | 
			
		||||
@@ -11,7 +11,7 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/preferences.h"
 | 
			
		||||
 | 
			
		||||
struct httpd_req;
 | 
			
		||||
struct httpd_req;  // NOLINT(readability-identifier-naming)
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace esp32_camera_web_server {
 | 
			
		||||
 
 | 
			
		||||
@@ -22,7 +22,7 @@ void ESP32RMTLEDStripLightOutput::setup() {
 | 
			
		||||
 | 
			
		||||
  size_t buffer_size = this->get_buffer_size_();
 | 
			
		||||
 | 
			
		||||
  ExternalRAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
 | 
			
		||||
  RAMAllocator<uint8_t> allocator(this->use_psram_ ? 0 : RAMAllocator<uint8_t>::ALLOC_INTERNAL);
 | 
			
		||||
  this->buf_ = allocator.allocate(buffer_size);
 | 
			
		||||
  if (this->buf_ == nullptr) {
 | 
			
		||||
    ESP_LOGE(TAG, "Cannot allocate LED buffer!");
 | 
			
		||||
@@ -37,7 +37,7 @@ void ESP32RMTLEDStripLightOutput::setup() {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ExternalRAMAllocator<rmt_item32_t> rmt_allocator(ExternalRAMAllocator<rmt_item32_t>::ALLOW_FAILURE);
 | 
			
		||||
  RAMAllocator<rmt_item32_t> rmt_allocator(this->use_psram_ ? 0 : RAMAllocator<rmt_item32_t>::ALLOC_INTERNAL);
 | 
			
		||||
  this->rmt_buf_ = rmt_allocator.allocate(buffer_size * 8 +
 | 
			
		||||
                                          1);  // 8 bits per byte, 1 rmt_item32_t per bit + 1 rmt_item32_t for reset
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -45,6 +45,7 @@ class ESP32RMTLEDStripLightOutput : public light::AddressableLight {
 | 
			
		||||
  void set_num_leds(uint16_t num_leds) { this->num_leds_ = num_leds; }
 | 
			
		||||
  void set_is_rgbw(bool is_rgbw) { this->is_rgbw_ = is_rgbw; }
 | 
			
		||||
  void set_is_wrgb(bool is_wrgb) { this->is_wrgb_ = is_wrgb; }
 | 
			
		||||
  void set_use_psram(bool use_psram) { this->use_psram_ = use_psram; }
 | 
			
		||||
 | 
			
		||||
  /// Set a maximum refresh rate in µs as some lights do not like being updated too often.
 | 
			
		||||
  void set_max_refresh_rate(uint32_t interval_us) { this->max_refresh_rate_ = interval_us; }
 | 
			
		||||
@@ -75,6 +76,7 @@ class ESP32RMTLEDStripLightOutput : public light::AddressableLight {
 | 
			
		||||
  uint16_t num_leds_;
 | 
			
		||||
  bool is_rgbw_;
 | 
			
		||||
  bool is_wrgb_;
 | 
			
		||||
  bool use_psram_;
 | 
			
		||||
 | 
			
		||||
  rmt_item32_t bit0_, bit1_, reset_;
 | 
			
		||||
  RGBOrder rgb_order_;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,9 @@
 | 
			
		||||
from dataclasses import dataclass
 | 
			
		||||
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import esp32_rmt, light
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_CHIPSET,
 | 
			
		||||
    CONF_IS_RGBW,
 | 
			
		||||
@@ -55,7 +55,7 @@ CHIPSETS = {
 | 
			
		||||
    "SM16703": LEDStripTimings(300, 900, 900, 300, 0, 0),
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
CONF_USE_PSRAM = "use_psram"
 | 
			
		||||
CONF_IS_WRGB = "is_wrgb"
 | 
			
		||||
CONF_BIT0_HIGH = "bit0_high"
 | 
			
		||||
CONF_BIT0_LOW = "bit0_low"
 | 
			
		||||
@@ -77,6 +77,7 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
            cv.Optional(CONF_CHIPSET): cv.one_of(*CHIPSETS, upper=True),
 | 
			
		||||
            cv.Optional(CONF_IS_RGBW, default=False): cv.boolean,
 | 
			
		||||
            cv.Optional(CONF_IS_WRGB, default=False): cv.boolean,
 | 
			
		||||
            cv.Optional(CONF_USE_PSRAM, default=True): cv.boolean,
 | 
			
		||||
            cv.Inclusive(
 | 
			
		||||
                CONF_BIT0_HIGH,
 | 
			
		||||
                "custom",
 | 
			
		||||
@@ -102,7 +103,7 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
                default="0 us",
 | 
			
		||||
            ): cv.positive_time_period_nanoseconds,
 | 
			
		||||
        }
 | 
			
		||||
    ),
 | 
			
		||||
    ).extend(cv.COMPONENT_SCHEMA),
 | 
			
		||||
    cv.has_exactly_one_key(CONF_CHIPSET, CONF_BIT0_HIGH),
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
@@ -145,6 +146,7 @@ async def to_code(config):
 | 
			
		||||
    cg.add(var.set_rgb_order(config[CONF_RGB_ORDER]))
 | 
			
		||||
    cg.add(var.set_is_rgbw(config[CONF_IS_RGBW]))
 | 
			
		||||
    cg.add(var.set_is_wrgb(config[CONF_IS_WRGB]))
 | 
			
		||||
    cg.add(var.set_use_psram(config[CONF_USE_PSRAM]))
 | 
			
		||||
 | 
			
		||||
    cg.add(
 | 
			
		||||
        var.set_rmt_channel(
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,13 @@
 | 
			
		||||
import logging
 | 
			
		||||
import os
 | 
			
		||||
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_BOARD,
 | 
			
		||||
    CONF_BOARD_FLASH_MODE,
 | 
			
		||||
    CONF_FRAMEWORK,
 | 
			
		||||
    CONF_PLATFORM_VERSION,
 | 
			
		||||
    CONF_SOURCE,
 | 
			
		||||
    CONF_VERSION,
 | 
			
		||||
    KEY_CORE,
 | 
			
		||||
@@ -12,27 +15,22 @@ from esphome.const import (
 | 
			
		||||
    KEY_TARGET_FRAMEWORK,
 | 
			
		||||
    KEY_TARGET_PLATFORM,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
    CONF_PLATFORM_VERSION,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.helpers import copy_file_if_changed
 | 
			
		||||
 | 
			
		||||
from .boards import BOARDS, ESP8266_LD_SCRIPTS
 | 
			
		||||
from .const import (
 | 
			
		||||
    CONF_RESTORE_FROM_FLASH,
 | 
			
		||||
    CONF_EARLY_PIN_INIT,
 | 
			
		||||
    CONF_RESTORE_FROM_FLASH,
 | 
			
		||||
    KEY_BOARD,
 | 
			
		||||
    KEY_ESP8266,
 | 
			
		||||
    KEY_FLASH_SIZE,
 | 
			
		||||
    KEY_PIN_INITIAL_STATES,
 | 
			
		||||
    esp8266_ns,
 | 
			
		||||
)
 | 
			
		||||
from .boards import BOARDS, ESP8266_LD_SCRIPTS
 | 
			
		||||
 | 
			
		||||
from .gpio import PinInitialState, add_pin_initial_states_array
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@esphome/core"]
 | 
			
		||||
_LOGGER = logging.getLogger(__name__)
 | 
			
		||||
AUTO_LOAD = ["preferences"]
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,9 @@
 | 
			
		||||
import logging
 | 
			
		||||
from dataclasses import dataclass
 | 
			
		||||
import logging
 | 
			
		||||
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ANALOG,
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
@@ -14,10 +17,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_PULLUP,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
)
 | 
			
		||||
from esphome import pins
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
 | 
			
		||||
from . import boards
 | 
			
		||||
from .const import KEY_BOARD, KEY_ESP8266, KEY_PIN_INITIAL_STATES, esp8266_ns
 | 
			
		||||
@@ -48,8 +48,10 @@ def _translate_pin(value):
 | 
			
		||||
            "This variable only supports pin numbers, not full pin schemas "
 | 
			
		||||
            "(with inverted and mode)."
 | 
			
		||||
        )
 | 
			
		||||
    if isinstance(value, int):
 | 
			
		||||
    if isinstance(value, int) and not isinstance(value, bool):
 | 
			
		||||
        return value
 | 
			
		||||
    if not isinstance(value, str):
 | 
			
		||||
        raise cv.Invalid(f"Invalid pin number: {value}")
 | 
			
		||||
    try:
 | 
			
		||||
        return int(value)
 | 
			
		||||
    except ValueError:
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,9 @@
 | 
			
		||||
import logging
 | 
			
		||||
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
from esphome.components.ota import BASE_OTA_SCHEMA, ota_to_code, OTAComponent
 | 
			
		||||
from esphome.components.ota import BASE_OTA_SCHEMA, OTAComponent, ota_to_code
 | 
			
		||||
from esphome.config_helpers import merge_config
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ESPHOME,
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
@@ -18,6 +17,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_VERSION,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import coroutine_with_priority
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
 | 
			
		||||
_LOGGER = logging.getLogger(__name__)
 | 
			
		||||
 | 
			
		||||
@@ -124,7 +124,6 @@ FINAL_VALIDATE_SCHEMA = ota_esphome_final_validate
 | 
			
		||||
@coroutine_with_priority(52.0)
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await ota_to_code(var, config)
 | 
			
		||||
    cg.add(var.set_port(config[CONF_PORT]))
 | 
			
		||||
    if CONF_PASSWORD in config:
 | 
			
		||||
        cg.add(var.set_auth_password(config[CONF_PASSWORD]))
 | 
			
		||||
@@ -132,3 +131,4 @@ async def to_code(config):
 | 
			
		||||
    cg.add_define("USE_OTA_VERSION", config[CONF_VERSION])
 | 
			
		||||
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
    await ota_to_code(var, config)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
import logging
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components.esp32 import add_idf_sdkconfig_option, get_esp32_variant
 | 
			
		||||
@@ -23,6 +24,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_MISO_PIN,
 | 
			
		||||
    CONF_MOSI_PIN,
 | 
			
		||||
    CONF_PAGE_ID,
 | 
			
		||||
    CONF_POLLING_INTERVAL,
 | 
			
		||||
    CONF_RESET_PIN,
 | 
			
		||||
    CONF_SPI,
 | 
			
		||||
    CONF_STATIC_IP,
 | 
			
		||||
@@ -30,13 +32,16 @@ from esphome.const import (
 | 
			
		||||
    CONF_TYPE,
 | 
			
		||||
    CONF_USE_ADDRESS,
 | 
			
		||||
    CONF_VALUE,
 | 
			
		||||
    KEY_CORE,
 | 
			
		||||
    KEY_FRAMEWORK_VERSION,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
from esphome.core import CORE, TimePeriodMilliseconds, coroutine_with_priority
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
 | 
			
		||||
CONFLICTS_WITH = ["wifi"]
 | 
			
		||||
DEPENDENCIES = ["esp32"]
 | 
			
		||||
AUTO_LOAD = ["network"]
 | 
			
		||||
LOGGER = logging.getLogger(__name__)
 | 
			
		||||
 | 
			
		||||
ethernet_ns = cg.esphome_ns.namespace("ethernet")
 | 
			
		||||
PHYRegister = ethernet_ns.struct("PHYRegister")
 | 
			
		||||
@@ -63,6 +68,7 @@ ETHERNET_TYPES = {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
SPI_ETHERNET_TYPES = ["W5500"]
 | 
			
		||||
SPI_ETHERNET_DEFAULT_POLLING_INTERVAL = TimePeriodMilliseconds(milliseconds=10)
 | 
			
		||||
 | 
			
		||||
emac_rmii_clock_mode_t = cg.global_ns.enum("emac_rmii_clock_mode_t")
 | 
			
		||||
emac_rmii_clock_gpio_t = cg.global_ns.enum("emac_rmii_clock_gpio_t")
 | 
			
		||||
@@ -100,6 +106,24 @@ EthernetComponent = ethernet_ns.class_("EthernetComponent", cg.Component)
 | 
			
		||||
ManualIP = ethernet_ns.struct("ManualIP")
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _is_framework_spi_polling_mode_supported():
 | 
			
		||||
    # SPI Ethernet without IRQ feature is added in
 | 
			
		||||
    # esp-idf >= (5.3+ ,5.2.1+, 5.1.4) and arduino-esp32 >= 3.0.0
 | 
			
		||||
    framework_version = CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]
 | 
			
		||||
    if CORE.using_esp_idf:
 | 
			
		||||
        if framework_version >= cv.Version(5, 3, 0):
 | 
			
		||||
            return True
 | 
			
		||||
        if cv.Version(5, 3, 0) > framework_version >= cv.Version(5, 2, 1):
 | 
			
		||||
            return True
 | 
			
		||||
        if cv.Version(5, 2, 0) > framework_version >= cv.Version(5, 1, 4):
 | 
			
		||||
            return True
 | 
			
		||||
        return False
 | 
			
		||||
    if CORE.using_arduino:
 | 
			
		||||
        return framework_version >= cv.Version(3, 0, 0)
 | 
			
		||||
    # fail safe: Unknown framework
 | 
			
		||||
    return False
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _validate(config):
 | 
			
		||||
    if CONF_USE_ADDRESS not in config:
 | 
			
		||||
        if CONF_MANUAL_IP in config:
 | 
			
		||||
@@ -107,6 +131,27 @@ def _validate(config):
 | 
			
		||||
        else:
 | 
			
		||||
            use_address = CORE.name + config[CONF_DOMAIN]
 | 
			
		||||
        config[CONF_USE_ADDRESS] = use_address
 | 
			
		||||
    if config[CONF_TYPE] in SPI_ETHERNET_TYPES:
 | 
			
		||||
        if _is_framework_spi_polling_mode_supported():
 | 
			
		||||
            if CONF_POLLING_INTERVAL in config and CONF_INTERRUPT_PIN in config:
 | 
			
		||||
                raise cv.Invalid(
 | 
			
		||||
                    f"Cannot specify more than one of {CONF_INTERRUPT_PIN}, {CONF_POLLING_INTERVAL}"
 | 
			
		||||
                )
 | 
			
		||||
            if CONF_POLLING_INTERVAL not in config and CONF_INTERRUPT_PIN not in config:
 | 
			
		||||
                config[CONF_POLLING_INTERVAL] = SPI_ETHERNET_DEFAULT_POLLING_INTERVAL
 | 
			
		||||
        else:
 | 
			
		||||
            if CONF_POLLING_INTERVAL in config:
 | 
			
		||||
                raise cv.Invalid(
 | 
			
		||||
                    "In this version of the framework "
 | 
			
		||||
                    f"({CORE.target_framework} {CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]}), "
 | 
			
		||||
                    f"'{CONF_POLLING_INTERVAL}' is not supported."
 | 
			
		||||
                )
 | 
			
		||||
            if CONF_INTERRUPT_PIN not in config:
 | 
			
		||||
                raise cv.Invalid(
 | 
			
		||||
                    "In this version of the framework "
 | 
			
		||||
                    f"({CORE.target_framework} {CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]}), "
 | 
			
		||||
                    f"'{CONF_INTERRUPT_PIN}' is a required option for [ethernet]."
 | 
			
		||||
                )
 | 
			
		||||
    return config
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -157,6 +202,11 @@ SPI_SCHEMA = BASE_SCHEMA.extend(
 | 
			
		||||
            cv.Optional(CONF_CLOCK_SPEED, default="26.67MHz"): cv.All(
 | 
			
		||||
                cv.frequency, cv.int_range(int(8e6), int(80e6))
 | 
			
		||||
            ),
 | 
			
		||||
            # Set default value (SPI_ETHERNET_DEFAULT_POLLING_INTERVAL) at _validate()
 | 
			
		||||
            cv.Optional(CONF_POLLING_INTERVAL): cv.All(
 | 
			
		||||
                cv.positive_time_period_milliseconds,
 | 
			
		||||
                cv.Range(min=TimePeriodMilliseconds(milliseconds=1)),
 | 
			
		||||
            ),
 | 
			
		||||
        }
 | 
			
		||||
    ),
 | 
			
		||||
)
 | 
			
		||||
@@ -234,6 +284,10 @@ async def to_code(config):
 | 
			
		||||
        cg.add(var.set_cs_pin(config[CONF_CS_PIN]))
 | 
			
		||||
        if CONF_INTERRUPT_PIN in config:
 | 
			
		||||
            cg.add(var.set_interrupt_pin(config[CONF_INTERRUPT_PIN]))
 | 
			
		||||
        else:
 | 
			
		||||
            cg.add(var.set_polling_interval(config[CONF_POLLING_INTERVAL]))
 | 
			
		||||
        if _is_framework_spi_polling_mode_supported():
 | 
			
		||||
            cg.add_define("USE_ETHERNET_SPI_POLLING_SUPPORT")
 | 
			
		||||
        if CONF_RESET_PIN in config:
 | 
			
		||||
            cg.add(var.set_reset_pin(config[CONF_RESET_PIN]))
 | 
			
		||||
        cg.add(var.set_clock_speed(config[CONF_CLOCK_SPEED]))
 | 
			
		||||
 
 | 
			
		||||
@@ -116,6 +116,9 @@ void EthernetComponent::setup() {
 | 
			
		||||
  eth_w5500_config_t w5500_config = ETH_W5500_DEFAULT_CONFIG(spi_handle);
 | 
			
		||||
#endif
 | 
			
		||||
  w5500_config.int_gpio_num = this->interrupt_pin_;
 | 
			
		||||
#ifdef USE_ETHERNET_SPI_POLLING_SUPPORT
 | 
			
		||||
  w5500_config.poll_period_ms = this->polling_interval_;
 | 
			
		||||
#endif
 | 
			
		||||
  phy_config.phy_addr = this->phy_addr_spi_;
 | 
			
		||||
  phy_config.reset_gpio_num = this->reset_pin_;
 | 
			
		||||
 | 
			
		||||
@@ -327,7 +330,14 @@ void EthernetComponent::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  MISO Pin: %u", this->miso_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  MOSI Pin: %u", this->mosi_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  CS Pin: %u", this->cs_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  IRQ Pin: %u", this->interrupt_pin_);
 | 
			
		||||
#ifdef USE_ETHERNET_SPI_POLLING_SUPPORT
 | 
			
		||||
  if (this->polling_interval_ != 0) {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Polling Interval: %lu ms", this->polling_interval_);
 | 
			
		||||
  } else
 | 
			
		||||
#endif
 | 
			
		||||
  {
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  IRQ Pin: %d", this->interrupt_pin_);
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Reset Pin: %d", this->reset_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Clock Speed: %d MHz", this->clock_speed_ / 1000000);
 | 
			
		||||
#else
 | 
			
		||||
@@ -536,6 +546,9 @@ void EthernetComponent::set_cs_pin(uint8_t cs_pin) { this->cs_pin_ = cs_pin; }
 | 
			
		||||
void EthernetComponent::set_interrupt_pin(uint8_t interrupt_pin) { this->interrupt_pin_ = interrupt_pin; }
 | 
			
		||||
void EthernetComponent::set_reset_pin(uint8_t reset_pin) { this->reset_pin_ = reset_pin; }
 | 
			
		||||
void EthernetComponent::set_clock_speed(int clock_speed) { this->clock_speed_ = clock_speed; }
 | 
			
		||||
#ifdef USE_ETHERNET_SPI_POLLING_SUPPORT
 | 
			
		||||
void EthernetComponent::set_polling_interval(uint32_t polling_interval) { this->polling_interval_ = polling_interval; }
 | 
			
		||||
#endif
 | 
			
		||||
#else
 | 
			
		||||
void EthernetComponent::set_phy_addr(uint8_t phy_addr) { this->phy_addr_ = phy_addr; }
 | 
			
		||||
void EthernetComponent::set_power_pin(int power_pin) { this->power_pin_ = power_pin; }
 | 
			
		||||
 
 | 
			
		||||
@@ -67,6 +67,9 @@ class EthernetComponent : public Component {
 | 
			
		||||
  void set_interrupt_pin(uint8_t interrupt_pin);
 | 
			
		||||
  void set_reset_pin(uint8_t reset_pin);
 | 
			
		||||
  void set_clock_speed(int clock_speed);
 | 
			
		||||
#ifdef USE_ETHERNET_SPI_POLLING_SUPPORT
 | 
			
		||||
  void set_polling_interval(uint32_t polling_interval);
 | 
			
		||||
#endif
 | 
			
		||||
#else
 | 
			
		||||
  void set_phy_addr(uint8_t phy_addr);
 | 
			
		||||
  void set_power_pin(int power_pin);
 | 
			
		||||
@@ -108,10 +111,13 @@ class EthernetComponent : public Component {
 | 
			
		||||
  uint8_t miso_pin_;
 | 
			
		||||
  uint8_t mosi_pin_;
 | 
			
		||||
  uint8_t cs_pin_;
 | 
			
		||||
  uint8_t interrupt_pin_;
 | 
			
		||||
  int interrupt_pin_{-1};
 | 
			
		||||
  int reset_pin_{-1};
 | 
			
		||||
  int phy_addr_spi_{-1};
 | 
			
		||||
  int clock_speed_;
 | 
			
		||||
#ifdef USE_ETHERNET_SPI_POLLING_SUPPORT
 | 
			
		||||
  uint32_t polling_interval_{0};
 | 
			
		||||
#endif
 | 
			
		||||
#else
 | 
			
		||||
  uint8_t phy_addr_{0};
 | 
			
		||||
  int power_pin_{-1};
 | 
			
		||||
 
 | 
			
		||||
@@ -111,11 +111,11 @@ void EZOSensor::loop() {
 | 
			
		||||
  if (buf[0] == 1) {
 | 
			
		||||
    std::string payload = reinterpret_cast<char *>(&buf[1]);
 | 
			
		||||
    if (!payload.empty()) {
 | 
			
		||||
      auto start_location = payload.find(',');
 | 
			
		||||
      switch (to_run->command_type) {
 | 
			
		||||
        case EzoCommandType::EZO_READ: {
 | 
			
		||||
          // some sensors return multiple comma-separated values, terminate string after first one
 | 
			
		||||
          int start_location = 0;
 | 
			
		||||
          if ((start_location = payload.find(',')) != std::string::npos) {
 | 
			
		||||
          if (start_location != std::string::npos) {
 | 
			
		||||
            payload.erase(start_location);
 | 
			
		||||
          }
 | 
			
		||||
          auto val = parse_number<float>(payload);
 | 
			
		||||
@@ -126,49 +126,37 @@ void EZOSensor::loop() {
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_LED: {
 | 
			
		||||
        case EzoCommandType::EZO_LED:
 | 
			
		||||
          this->led_callback_.call(payload.back() == '1');
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_DEVICE_INFORMATION: {
 | 
			
		||||
          int start_location = 0;
 | 
			
		||||
          if ((start_location = payload.find(',')) != std::string::npos) {
 | 
			
		||||
        case EzoCommandType::EZO_DEVICE_INFORMATION:
 | 
			
		||||
          if (start_location != std::string::npos) {
 | 
			
		||||
            this->device_infomation_callback_.call(payload.substr(start_location + 1));
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_SLOPE: {
 | 
			
		||||
          int start_location = 0;
 | 
			
		||||
          if ((start_location = payload.find(',')) != std::string::npos) {
 | 
			
		||||
        case EzoCommandType::EZO_SLOPE:
 | 
			
		||||
          if (start_location != std::string::npos) {
 | 
			
		||||
            this->slope_callback_.call(payload.substr(start_location + 1));
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_CALIBRATION: {
 | 
			
		||||
          int start_location = 0;
 | 
			
		||||
          if ((start_location = payload.find(',')) != std::string::npos) {
 | 
			
		||||
        case EzoCommandType::EZO_CALIBRATION:
 | 
			
		||||
          if (start_location != std::string::npos) {
 | 
			
		||||
            this->calibration_callback_.call(payload.substr(start_location + 1));
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_T: {
 | 
			
		||||
          int start_location = 0;
 | 
			
		||||
          if ((start_location = payload.find(',')) != std::string::npos) {
 | 
			
		||||
        case EzoCommandType::EZO_T:
 | 
			
		||||
          if (start_location != std::string::npos) {
 | 
			
		||||
            this->t_callback_.call(payload.substr(start_location + 1));
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        case EzoCommandType::EZO_CUSTOM: {
 | 
			
		||||
        case EzoCommandType::EZO_CUSTOM:
 | 
			
		||||
          this->custom_callback_.call(payload);
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
        default: {
 | 
			
		||||
        default:
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  this->commands_.pop_front();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -178,7 +166,7 @@ void EZOSensor::add_command_(const std::string &command, EzoCommandType command_
 | 
			
		||||
  ezo_command->command_type = command_type;
 | 
			
		||||
  ezo_command->delay_ms = delay_ms;
 | 
			
		||||
  this->commands_.push_back(std::move(ezo_command));
 | 
			
		||||
};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void EZOSensor::set_calibration_point_(EzoCalibrationType type, float value) {
 | 
			
		||||
  std::string payload = str_sprintf("Cal,%s,%0.2f", EZO_CALIBRATION_TYPE_STRINGS[type], value);
 | 
			
		||||
 
 | 
			
		||||
@@ -5,7 +5,8 @@ import os
 | 
			
		||||
from pathlib import Path
 | 
			
		||||
import re
 | 
			
		||||
 | 
			
		||||
from packaging import version
 | 
			
		||||
import freetype
 | 
			
		||||
import glyphsets
 | 
			
		||||
import requests
 | 
			
		||||
 | 
			
		||||
from esphome import core, external_files
 | 
			
		||||
@@ -43,6 +44,18 @@ GlyphData = font_ns.struct("GlyphData")
 | 
			
		||||
CONF_BPP = "bpp"
 | 
			
		||||
CONF_EXTRAS = "extras"
 | 
			
		||||
CONF_FONTS = "fonts"
 | 
			
		||||
CONF_GLYPHSETS = "glyphsets"
 | 
			
		||||
CONF_IGNORE_MISSING_GLYPHS = "ignore_missing_glyphs"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Cache loaded freetype fonts
 | 
			
		||||
class FontCache(dict):
 | 
			
		||||
    def __missing__(self, key):
 | 
			
		||||
        res = self[key] = freetype.Face(key)
 | 
			
		||||
        return res
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FONT_CACHE = FontCache()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def glyph_comparator(x, y):
 | 
			
		||||
@@ -59,57 +72,109 @@ def glyph_comparator(x, y):
 | 
			
		||||
        return -1
 | 
			
		||||
    if len(x_) > len(y_):
 | 
			
		||||
        return 1
 | 
			
		||||
    raise cv.Invalid(f"Found duplicate glyph {x}")
 | 
			
		||||
    return 0
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_glyphs(value):
 | 
			
		||||
    if isinstance(value, list):
 | 
			
		||||
        value = cv.Schema([cv.string])(value)
 | 
			
		||||
    value = cv.Schema([cv.string])(list(value))
 | 
			
		||||
def flatten(lists) -> list:
 | 
			
		||||
    """
 | 
			
		||||
    Given a list of lists, flatten it to a single list of all elements of all lists.
 | 
			
		||||
    This wraps itertools.chain.from_iterable to make it more readable, and return a list
 | 
			
		||||
    rather than a single use iterable.
 | 
			
		||||
    """
 | 
			
		||||
    from itertools import chain
 | 
			
		||||
 | 
			
		||||
    value.sort(key=functools.cmp_to_key(glyph_comparator))
 | 
			
		||||
    return value
 | 
			
		||||
    return list(chain.from_iterable(lists))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
font_map = {}
 | 
			
		||||
def check_missing_glyphs(file, codepoints, warning: bool = False):
 | 
			
		||||
    """
 | 
			
		||||
    Check that the given font file actually contains the requested glyphs
 | 
			
		||||
    :param file: A Truetype font file
 | 
			
		||||
    :param codepoints: A list of codepoints to check
 | 
			
		||||
    :param warning: If true, log a warning instead of raising an exception
 | 
			
		||||
    """
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def merge_glyphs(config):
 | 
			
		||||
    glyphs = []
 | 
			
		||||
    glyphs.extend(config[CONF_GLYPHS])
 | 
			
		||||
    font_list = [(EFont(config[CONF_FILE], config[CONF_SIZE], config[CONF_GLYPHS]))]
 | 
			
		||||
    if extras := config.get(CONF_EXTRAS):
 | 
			
		||||
        extra_fonts = list(
 | 
			
		||||
            map(
 | 
			
		||||
                lambda x: EFont(x[CONF_FILE], config[CONF_SIZE], x[CONF_GLYPHS]), extras
 | 
			
		||||
            )
 | 
			
		||||
    font = FONT_CACHE[file]
 | 
			
		||||
    missing = [chr(x) for x in codepoints if font.get_char_index(x) == 0]
 | 
			
		||||
    if missing:
 | 
			
		||||
        # Only list up to 10 missing glyphs
 | 
			
		||||
        missing.sort(key=functools.cmp_to_key(glyph_comparator))
 | 
			
		||||
        count = len(missing)
 | 
			
		||||
        missing = missing[:10]
 | 
			
		||||
        missing_str = "\n    ".join(
 | 
			
		||||
            f"{x} ({x.encode('unicode_escape')})" for x in missing
 | 
			
		||||
        )
 | 
			
		||||
        font_list.extend(extra_fonts)
 | 
			
		||||
        for extra in extras:
 | 
			
		||||
            glyphs.extend(extra[CONF_GLYPHS])
 | 
			
		||||
        validate_glyphs(glyphs)
 | 
			
		||||
    font_map[config[CONF_ID]] = font_list
 | 
			
		||||
        if count > 10:
 | 
			
		||||
            missing_str += f"\n    and {count - 10} more."
 | 
			
		||||
        message = f"Font {Path(file).name} is missing {count} glyph{'s' if count != 1 else ''}:\n    {missing_str}"
 | 
			
		||||
        if warning:
 | 
			
		||||
            _LOGGER.warning(message)
 | 
			
		||||
        else:
 | 
			
		||||
            raise cv.Invalid(message)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_glyphs(config):
 | 
			
		||||
    """
 | 
			
		||||
    Check for duplicate codepoints, then check that all requested codepoints actually
 | 
			
		||||
    have glyphs defined in the appropriate font file.
 | 
			
		||||
    """
 | 
			
		||||
 | 
			
		||||
    # Collect all glyph codepoints and flatten to a list of chars
 | 
			
		||||
    glyphspoints = flatten(
 | 
			
		||||
        [x[CONF_GLYPHS] for x in config[CONF_EXTRAS]] + config[CONF_GLYPHS]
 | 
			
		||||
    )
 | 
			
		||||
    # Convert a list of strings to a list of chars (one char strings)
 | 
			
		||||
    glyphspoints = flatten([list(x) for x in glyphspoints])
 | 
			
		||||
    if len(set(glyphspoints)) != len(glyphspoints):
 | 
			
		||||
        duplicates = {x for x in glyphspoints if glyphspoints.count(x) > 1}
 | 
			
		||||
        dup_str = ", ".join(f"{x} ({x.encode('unicode_escape')})" for x in duplicates)
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"Found duplicate glyph{'s' if len(duplicates) != 1 else ''}: {dup_str}"
 | 
			
		||||
        )
 | 
			
		||||
    # convert to codepoints
 | 
			
		||||
    glyphspoints = {ord(x) for x in glyphspoints}
 | 
			
		||||
    fileconf = config[CONF_FILE]
 | 
			
		||||
    setpoints = set(
 | 
			
		||||
        flatten([glyphsets.unicodes_per_glyphset(x) for x in config[CONF_GLYPHSETS]])
 | 
			
		||||
    )
 | 
			
		||||
    # Make setpoints and glyphspoints disjoint
 | 
			
		||||
    setpoints.difference_update(glyphspoints)
 | 
			
		||||
    if fileconf[CONF_TYPE] == TYPE_LOCAL_BITMAP:
 | 
			
		||||
        # Pillow only allows 256 glyphs per bitmap font. Not sure if that is a Pillow limitation
 | 
			
		||||
        # or a file format limitation
 | 
			
		||||
        if any(x >= 256 for x in setpoints.copy().union(glyphspoints)):
 | 
			
		||||
            raise cv.Invalid("Codepoints in bitmap fonts must be in the range 0-255")
 | 
			
		||||
    else:
 | 
			
		||||
        # for TT fonts, check that glyphs are actually present
 | 
			
		||||
        # Check extras against their own font, exclude from parent font codepoints
 | 
			
		||||
        for extra in config[CONF_EXTRAS]:
 | 
			
		||||
            points = {ord(x) for x in flatten(extra[CONF_GLYPHS])}
 | 
			
		||||
            glyphspoints.difference_update(points)
 | 
			
		||||
            setpoints.difference_update(points)
 | 
			
		||||
            check_missing_glyphs(extra[CONF_FILE][CONF_PATH], points)
 | 
			
		||||
 | 
			
		||||
        # A named glyph that can't be provided is an error
 | 
			
		||||
        check_missing_glyphs(fileconf[CONF_PATH], glyphspoints)
 | 
			
		||||
        # A missing glyph from a set is a warning.
 | 
			
		||||
        if not config[CONF_IGNORE_MISSING_GLYPHS]:
 | 
			
		||||
            check_missing_glyphs(fileconf[CONF_PATH], setpoints, warning=True)
 | 
			
		||||
 | 
			
		||||
    # Populate the default after the above checks so that use of the default doesn't trigger errors
 | 
			
		||||
    if not config[CONF_GLYPHS] and not config[CONF_GLYPHSETS]:
 | 
			
		||||
        if fileconf[CONF_TYPE] == TYPE_LOCAL_BITMAP:
 | 
			
		||||
            config[CONF_GLYPHS] = [DEFAULT_GLYPHS]
 | 
			
		||||
        else:
 | 
			
		||||
            # set a default glyphset, intersected with what the font actually offers
 | 
			
		||||
            font = FONT_CACHE[fileconf[CONF_PATH]]
 | 
			
		||||
            config[CONF_GLYPHS] = [
 | 
			
		||||
                chr(x)
 | 
			
		||||
                for x in glyphsets.unicodes_per_glyphset(DEFAULT_GLYPHSET)
 | 
			
		||||
                if font.get_char_index(x) != 0
 | 
			
		||||
            ]
 | 
			
		||||
 | 
			
		||||
    return config
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_pillow_installed(value):
 | 
			
		||||
    try:
 | 
			
		||||
        import PIL
 | 
			
		||||
    except ImportError as err:
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            "Please install the pillow python package to use this feature. "
 | 
			
		||||
            '(pip install "pillow==10.2.0")'
 | 
			
		||||
        ) from err
 | 
			
		||||
 | 
			
		||||
    if version.parse(PIL.__version__) != version.parse("10.2.0"):
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            "Please update your pillow installation to 10.2.0. "
 | 
			
		||||
            '(pip install "pillow==10.2.0")'
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    return value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FONT_EXTENSIONS = (".ttf", ".woff", ".otf")
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -120,7 +185,7 @@ def validate_truetype_file(value):
 | 
			
		||||
        )
 | 
			
		||||
    if not any(map(value.lower().endswith, FONT_EXTENSIONS)):
 | 
			
		||||
        raise cv.Invalid(f"Only {FONT_EXTENSIONS} files are supported.")
 | 
			
		||||
    return cv.file_(value)
 | 
			
		||||
    return CORE.relative_config_path(cv.file_(value))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
TYPE_LOCAL = "local"
 | 
			
		||||
@@ -139,6 +204,10 @@ LOCAL_BITMAP_SCHEMA = cv.Schema(
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
FULLPATH_SCHEMA = cv.maybe_simple_value(
 | 
			
		||||
    {cv.Required(CONF_PATH): cv.string}, key=CONF_PATH
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONF_ITALIC = "italic"
 | 
			
		||||
FONT_WEIGHTS = {
 | 
			
		||||
    "thin": 100,
 | 
			
		||||
@@ -167,13 +236,13 @@ def _compute_local_font_path(value: dict) -> Path:
 | 
			
		||||
    return base_dir / key
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_font_path(value, type) -> Path:
 | 
			
		||||
    if type == TYPE_GFONTS:
 | 
			
		||||
def get_font_path(value, font_type) -> Path:
 | 
			
		||||
    if font_type == TYPE_GFONTS:
 | 
			
		||||
        name = f"{value[CONF_FAMILY]}@{value[CONF_WEIGHT]}@{value[CONF_ITALIC]}@v1"
 | 
			
		||||
        return external_files.compute_local_file_dir(DOMAIN) / f"{name}.ttf"
 | 
			
		||||
    if type == TYPE_WEB:
 | 
			
		||||
    if font_type == TYPE_WEB:
 | 
			
		||||
        return _compute_local_font_path(value) / "font.ttf"
 | 
			
		||||
    return None
 | 
			
		||||
    assert False
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def download_gfont(value):
 | 
			
		||||
@@ -203,7 +272,7 @@ def download_gfont(value):
 | 
			
		||||
    _LOGGER.debug("download_gfont: ttf_url=%s", ttf_url)
 | 
			
		||||
 | 
			
		||||
    external_files.download_content(ttf_url, path)
 | 
			
		||||
    return value
 | 
			
		||||
    return FULLPATH_SCHEMA(path)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def download_web_font(value):
 | 
			
		||||
@@ -212,7 +281,7 @@ def download_web_font(value):
 | 
			
		||||
 | 
			
		||||
    external_files.download_content(url, path)
 | 
			
		||||
    _LOGGER.debug("download_web_font: path=%s", path)
 | 
			
		||||
    return value
 | 
			
		||||
    return FULLPATH_SCHEMA(path)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
EXTERNAL_FONT_SCHEMA = cv.Schema(
 | 
			
		||||
@@ -225,7 +294,6 @@ EXTERNAL_FONT_SCHEMA = cv.Schema(
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
GFONTS_SCHEMA = cv.All(
 | 
			
		||||
    EXTERNAL_FONT_SCHEMA.extend(
 | 
			
		||||
        {
 | 
			
		||||
@@ -259,10 +327,10 @@ def validate_file_shorthand(value):
 | 
			
		||||
        }
 | 
			
		||||
        if weight is not None:
 | 
			
		||||
            data[CONF_WEIGHT] = weight[1:]
 | 
			
		||||
        return FILE_SCHEMA(data)
 | 
			
		||||
        return font_file_schema(data)
 | 
			
		||||
 | 
			
		||||
    if value.startswith("http://") or value.startswith("https://"):
 | 
			
		||||
        return FILE_SCHEMA(
 | 
			
		||||
        return font_file_schema(
 | 
			
		||||
            {
 | 
			
		||||
                CONF_TYPE: TYPE_WEB,
 | 
			
		||||
                CONF_URL: value,
 | 
			
		||||
@@ -270,14 +338,15 @@ def validate_file_shorthand(value):
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    if value.endswith(".pcf") or value.endswith(".bdf"):
 | 
			
		||||
        return FILE_SCHEMA(
 | 
			
		||||
            {
 | 
			
		||||
                CONF_TYPE: TYPE_LOCAL_BITMAP,
 | 
			
		||||
                CONF_PATH: value,
 | 
			
		||||
            }
 | 
			
		||||
        value = convert_bitmap_to_pillow_font(
 | 
			
		||||
            CORE.relative_config_path(cv.file_(value))
 | 
			
		||||
        )
 | 
			
		||||
        return {
 | 
			
		||||
            CONF_TYPE: TYPE_LOCAL_BITMAP,
 | 
			
		||||
            CONF_PATH: value,
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    return FILE_SCHEMA(
 | 
			
		||||
    return font_file_schema(
 | 
			
		||||
        {
 | 
			
		||||
            CONF_TYPE: TYPE_LOCAL,
 | 
			
		||||
            CONF_PATH: value,
 | 
			
		||||
@@ -295,31 +364,37 @@ TYPED_FILE_SCHEMA = cv.typed_schema(
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _file_schema(value):
 | 
			
		||||
def font_file_schema(value):
 | 
			
		||||
    if isinstance(value, str):
 | 
			
		||||
        return validate_file_shorthand(value)
 | 
			
		||||
    return TYPED_FILE_SCHEMA(value)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILE_SCHEMA = cv.All(_file_schema)
 | 
			
		||||
 | 
			
		||||
# Default if no glyphs or glyphsets are provided
 | 
			
		||||
DEFAULT_GLYPHSET = "GF_Latin_Kernel"
 | 
			
		||||
# default for bitmap fonts
 | 
			
		||||
DEFAULT_GLYPHS = (
 | 
			
		||||
    ' !"%()+=,-.:/?0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz°'
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONF_RAW_GLYPH_ID = "raw_glyph_id"
 | 
			
		||||
 | 
			
		||||
FONT_SCHEMA = cv.Schema(
 | 
			
		||||
    {
 | 
			
		||||
        cv.Required(CONF_ID): cv.declare_id(Font),
 | 
			
		||||
        cv.Required(CONF_FILE): FILE_SCHEMA,
 | 
			
		||||
        cv.Optional(CONF_GLYPHS, default=DEFAULT_GLYPHS): validate_glyphs,
 | 
			
		||||
        cv.Required(CONF_FILE): font_file_schema,
 | 
			
		||||
        cv.Optional(CONF_GLYPHS, default=[]): cv.ensure_list(cv.string_strict),
 | 
			
		||||
        cv.Optional(CONF_GLYPHSETS, default=[]): cv.ensure_list(
 | 
			
		||||
            cv.one_of(*glyphsets.defined_glyphsets())
 | 
			
		||||
        ),
 | 
			
		||||
        cv.Optional(CONF_IGNORE_MISSING_GLYPHS, default=False): cv.boolean,
 | 
			
		||||
        cv.Optional(CONF_SIZE, default=20): cv.int_range(min=1),
 | 
			
		||||
        cv.Optional(CONF_BPP, default=1): cv.one_of(1, 2, 4, 8),
 | 
			
		||||
        cv.Optional(CONF_EXTRAS): cv.ensure_list(
 | 
			
		||||
        cv.Optional(CONF_EXTRAS, default=[]): cv.ensure_list(
 | 
			
		||||
            cv.Schema(
 | 
			
		||||
                {
 | 
			
		||||
                    cv.Required(CONF_FILE): FILE_SCHEMA,
 | 
			
		||||
                    cv.Required(CONF_GLYPHS): validate_glyphs,
 | 
			
		||||
                    cv.Required(CONF_FILE): font_file_schema,
 | 
			
		||||
                    cv.Required(CONF_GLYPHS): cv.ensure_list(cv.string_strict),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
        ),
 | 
			
		||||
@@ -328,7 +403,7 @@ FONT_SCHEMA = cv.Schema(
 | 
			
		||||
    },
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = cv.All(validate_pillow_installed, FONT_SCHEMA, merge_glyphs)
 | 
			
		||||
CONFIG_SCHEMA = cv.All(FONT_SCHEMA, validate_glyphs)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# PIL doesn't provide a consistent interface for both TrueType and bitmap
 | 
			
		||||
@@ -344,7 +419,7 @@ class TrueTypeFontWrapper:
 | 
			
		||||
        return offset_x, offset_y
 | 
			
		||||
 | 
			
		||||
    def getmask(self, glyph, **kwargs):
 | 
			
		||||
        return self.font.getmask(glyph, **kwargs)
 | 
			
		||||
        return self.font.getmask(str(glyph), **kwargs)
 | 
			
		||||
 | 
			
		||||
    def getmetrics(self, glyphs):
 | 
			
		||||
        return self.font.getmetrics()
 | 
			
		||||
@@ -359,7 +434,7 @@ class BitmapFontWrapper:
 | 
			
		||||
        return 0, 0
 | 
			
		||||
 | 
			
		||||
    def getmask(self, glyph, **kwargs):
 | 
			
		||||
        return self.font.getmask(glyph, **kwargs)
 | 
			
		||||
        return self.font.getmask(str(glyph), **kwargs)
 | 
			
		||||
 | 
			
		||||
    def getmetrics(self, glyphs):
 | 
			
		||||
        max_height = 0
 | 
			
		||||
@@ -367,28 +442,20 @@ class BitmapFontWrapper:
 | 
			
		||||
            mask = self.getmask(glyph, mode="1")
 | 
			
		||||
            _, height = mask.size
 | 
			
		||||
            max_height = max(max_height, height)
 | 
			
		||||
        return (max_height, 0)
 | 
			
		||||
        return max_height, 0
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class EFont:
 | 
			
		||||
    def __init__(self, file, size, glyphs):
 | 
			
		||||
        self.glyphs = glyphs
 | 
			
		||||
    def __init__(self, file, size, codepoints):
 | 
			
		||||
        self.codepoints = codepoints
 | 
			
		||||
        path = file[CONF_PATH]
 | 
			
		||||
        self.name = Path(path).name
 | 
			
		||||
        ftype = file[CONF_TYPE]
 | 
			
		||||
        if ftype == TYPE_LOCAL_BITMAP:
 | 
			
		||||
            font = load_bitmap_font(CORE.relative_config_path(file[CONF_PATH]))
 | 
			
		||||
        elif ftype == TYPE_LOCAL:
 | 
			
		||||
            path = CORE.relative_config_path(file[CONF_PATH])
 | 
			
		||||
            font = load_ttf_font(path, size)
 | 
			
		||||
        elif ftype in (TYPE_GFONTS, TYPE_WEB):
 | 
			
		||||
            path = get_font_path(file, ftype)
 | 
			
		||||
            font = load_ttf_font(path, size)
 | 
			
		||||
            self.font = load_bitmap_font(path)
 | 
			
		||||
        else:
 | 
			
		||||
            raise cv.Invalid(f"Could not load font: unknown type: {ftype}")
 | 
			
		||||
        self.font = font
 | 
			
		||||
        self.ascent, self.descent = font.getmetrics(glyphs)
 | 
			
		||||
 | 
			
		||||
    def has_glyph(self, glyph):
 | 
			
		||||
        return glyph in self.glyphs
 | 
			
		||||
            self.font = load_ttf_font(path, size)
 | 
			
		||||
        self.ascent, self.descent = self.font.getmetrics(codepoints)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def convert_bitmap_to_pillow_font(filepath):
 | 
			
		||||
@@ -400,6 +467,7 @@ def convert_bitmap_to_pillow_font(filepath):
 | 
			
		||||
 | 
			
		||||
    copy_file_if_changed(filepath, local_bitmap_font_file)
 | 
			
		||||
 | 
			
		||||
    local_pil_font_file = local_bitmap_font_file.with_suffix(".pil")
 | 
			
		||||
    with open(local_bitmap_font_file, "rb") as fp:
 | 
			
		||||
        try:
 | 
			
		||||
            try:
 | 
			
		||||
@@ -409,28 +477,22 @@ def convert_bitmap_to_pillow_font(filepath):
 | 
			
		||||
                p = BdfFontFile.BdfFontFile(fp)
 | 
			
		||||
 | 
			
		||||
            # Convert to pillow-formatted fonts, which have a .pil and .pbm extension.
 | 
			
		||||
            p.save(local_bitmap_font_file)
 | 
			
		||||
            p.save(local_pil_font_file)
 | 
			
		||||
        except (SyntaxError, OSError) as err:
 | 
			
		||||
            raise core.EsphomeError(
 | 
			
		||||
                f"Failed to parse as bitmap font: '{filepath}': {err}"
 | 
			
		||||
            )
 | 
			
		||||
 | 
			
		||||
    local_pil_font_file = os.path.splitext(local_bitmap_font_file)[0] + ".pil"
 | 
			
		||||
    return cv.file_(local_pil_font_file)
 | 
			
		||||
    return str(local_pil_font_file)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def load_bitmap_font(filepath):
 | 
			
		||||
    from PIL import ImageFont
 | 
			
		||||
 | 
			
		||||
    # Convert bpf and pcf files to pillow fonts, first.
 | 
			
		||||
    pil_font_path = convert_bitmap_to_pillow_font(filepath)
 | 
			
		||||
 | 
			
		||||
    try:
 | 
			
		||||
        font = ImageFont.load(str(pil_font_path))
 | 
			
		||||
        font = ImageFont.load(str(filepath))
 | 
			
		||||
    except Exception as e:
 | 
			
		||||
        raise core.EsphomeError(
 | 
			
		||||
            f"Failed to load bitmap font file: {pil_font_path} : {e}"
 | 
			
		||||
        )
 | 
			
		||||
        raise core.EsphomeError(f"Failed to load bitmap font file: {filepath}: {e}")
 | 
			
		||||
 | 
			
		||||
    return BitmapFontWrapper(font)
 | 
			
		||||
 | 
			
		||||
@@ -441,7 +503,7 @@ def load_ttf_font(path, size):
 | 
			
		||||
    try:
 | 
			
		||||
        font = ImageFont.truetype(str(path), size)
 | 
			
		||||
    except Exception as e:
 | 
			
		||||
        raise core.EsphomeError(f"Could not load truetype file {path}: {e}")
 | 
			
		||||
        raise core.EsphomeError(f"Could not load TrueType file {path}: {e}")
 | 
			
		||||
 | 
			
		||||
    return TrueTypeFontWrapper(font)
 | 
			
		||||
 | 
			
		||||
@@ -456,14 +518,35 @@ class GlyphInfo:
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    glyph_to_font_map = {}
 | 
			
		||||
    font_list = font_map[config[CONF_ID]]
 | 
			
		||||
    glyphs = []
 | 
			
		||||
    for font in font_list:
 | 
			
		||||
        glyphs.extend(font.glyphs)
 | 
			
		||||
        for glyph in font.glyphs:
 | 
			
		||||
            glyph_to_font_map[glyph] = font
 | 
			
		||||
    glyphs.sort(key=functools.cmp_to_key(glyph_comparator))
 | 
			
		||||
    """
 | 
			
		||||
    Collect all glyph codepoints, construct a map from a codepoint to a font file.
 | 
			
		||||
    Codepoints are either explicit (glyphs key in top level or extras) or part of a glyphset.
 | 
			
		||||
    Codepoints listed in extras use the extra font and override codepoints from glyphsets.
 | 
			
		||||
    Achieve this by processing the base codepoints first, then the extras
 | 
			
		||||
    """
 | 
			
		||||
 | 
			
		||||
    # get the codepoints from glyphsets and flatten to a set of chrs.
 | 
			
		||||
    point_set: set[str] = {
 | 
			
		||||
        chr(x)
 | 
			
		||||
        for x in flatten(
 | 
			
		||||
            [glyphsets.unicodes_per_glyphset(x) for x in config[CONF_GLYPHSETS]]
 | 
			
		||||
        )
 | 
			
		||||
    }
 | 
			
		||||
    # get the codepoints from the glyphs key, flatten to a list of chrs and combine with the points from glyphsets
 | 
			
		||||
    point_set.update(flatten(config[CONF_GLYPHS]))
 | 
			
		||||
    size = config[CONF_SIZE]
 | 
			
		||||
    # Create the codepoint to font file map
 | 
			
		||||
    base_font = EFont(config[CONF_FILE], size, point_set)
 | 
			
		||||
    point_font_map: dict[str, EFont] = {c: base_font for c in point_set}
 | 
			
		||||
    # process extras, updating the map and extending the codepoint list
 | 
			
		||||
    for extra in config[CONF_EXTRAS]:
 | 
			
		||||
        extra_points = flatten(extra[CONF_GLYPHS])
 | 
			
		||||
        point_set.update(extra_points)
 | 
			
		||||
        extra_font = EFont(extra[CONF_FILE], size, extra_points)
 | 
			
		||||
        point_font_map.update({c: extra_font for c in extra_points})
 | 
			
		||||
 | 
			
		||||
    codepoints = list(point_set)
 | 
			
		||||
    codepoints.sort(key=functools.cmp_to_key(glyph_comparator))
 | 
			
		||||
    glyph_args = {}
 | 
			
		||||
    data = []
 | 
			
		||||
    bpp = config[CONF_BPP]
 | 
			
		||||
@@ -473,10 +556,11 @@ async def to_code(config):
 | 
			
		||||
    else:
 | 
			
		||||
        mode = "L"
 | 
			
		||||
        scale = 256 // (1 << bpp)
 | 
			
		||||
    for glyph in glyphs:
 | 
			
		||||
        font = glyph_to_font_map[glyph].font
 | 
			
		||||
        mask = font.getmask(glyph, mode=mode)
 | 
			
		||||
        offset_x, offset_y = font.getoffset(glyph)
 | 
			
		||||
    # create the data array for all glyphs
 | 
			
		||||
    for codepoint in codepoints:
 | 
			
		||||
        font = point_font_map[codepoint]
 | 
			
		||||
        mask = font.font.getmask(codepoint, mode=mode)
 | 
			
		||||
        offset_x, offset_y = font.font.getoffset(codepoint)
 | 
			
		||||
        width, height = mask.size
 | 
			
		||||
        glyph_data = [0] * ((height * width * bpp + 7) // 8)
 | 
			
		||||
        pos = 0
 | 
			
		||||
@@ -487,31 +571,34 @@ async def to_code(config):
 | 
			
		||||
                    if pixel & (1 << (bpp - bit_num - 1)):
 | 
			
		||||
                        glyph_data[pos // 8] |= 0x80 >> (pos % 8)
 | 
			
		||||
                    pos += 1
 | 
			
		||||
        glyph_args[glyph] = GlyphInfo(len(data), offset_x, offset_y, width, height)
 | 
			
		||||
        glyph_args[codepoint] = GlyphInfo(len(data), offset_x, offset_y, width, height)
 | 
			
		||||
        data += glyph_data
 | 
			
		||||
 | 
			
		||||
    rhs = [HexInt(x) for x in data]
 | 
			
		||||
    prog_arr = cg.progmem_array(config[CONF_RAW_DATA_ID], rhs)
 | 
			
		||||
 | 
			
		||||
    # Create the glyph table that points to data in the above array.
 | 
			
		||||
    glyph_initializer = []
 | 
			
		||||
    for glyph in glyphs:
 | 
			
		||||
    for codepoint in codepoints:
 | 
			
		||||
        glyph_initializer.append(
 | 
			
		||||
            cg.StructInitializer(
 | 
			
		||||
                GlyphData,
 | 
			
		||||
                (
 | 
			
		||||
                    "a_char",
 | 
			
		||||
                    cg.RawExpression(f"(const uint8_t *){cpp_string_escape(glyph)}"),
 | 
			
		||||
                    cg.RawExpression(
 | 
			
		||||
                        f"(const uint8_t *){cpp_string_escape(codepoint)}"
 | 
			
		||||
                    ),
 | 
			
		||||
                ),
 | 
			
		||||
                (
 | 
			
		||||
                    "data",
 | 
			
		||||
                    cg.RawExpression(
 | 
			
		||||
                        f"{str(prog_arr)} + {str(glyph_args[glyph].data_len)}"
 | 
			
		||||
                        f"{str(prog_arr)} + {str(glyph_args[codepoint].data_len)}"
 | 
			
		||||
                    ),
 | 
			
		||||
                ),
 | 
			
		||||
                ("offset_x", glyph_args[glyph].offset_x),
 | 
			
		||||
                ("offset_y", glyph_args[glyph].offset_y),
 | 
			
		||||
                ("width", glyph_args[glyph].width),
 | 
			
		||||
                ("height", glyph_args[glyph].height),
 | 
			
		||||
                ("offset_x", glyph_args[codepoint].offset_x),
 | 
			
		||||
                ("offset_y", glyph_args[codepoint].offset_y),
 | 
			
		||||
                ("width", glyph_args[codepoint].width),
 | 
			
		||||
                ("height", glyph_args[codepoint].height),
 | 
			
		||||
            )
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
@@ -521,7 +608,7 @@ async def to_code(config):
 | 
			
		||||
        config[CONF_ID],
 | 
			
		||||
        glyphs,
 | 
			
		||||
        len(glyph_initializer),
 | 
			
		||||
        font_list[0].ascent,
 | 
			
		||||
        font_list[0].ascent + font_list[0].descent,
 | 
			
		||||
        base_font.ascent,
 | 
			
		||||
        base_font.ascent + base_font.descent,
 | 
			
		||||
        bpp,
 | 
			
		||||
    )
 | 
			
		||||
 
 | 
			
		||||
@@ -133,9 +133,11 @@ void Font::print(int x_start, int y_start, display::Display *display, Color colo
 | 
			
		||||
    auto diff_r = (float) color.r - (float) background.r;
 | 
			
		||||
    auto diff_g = (float) color.g - (float) background.g;
 | 
			
		||||
    auto diff_b = (float) color.b - (float) background.b;
 | 
			
		||||
    auto diff_w = (float) color.w - (float) background.w;
 | 
			
		||||
    auto b_r = (float) background.r;
 | 
			
		||||
    auto b_g = (float) background.g;
 | 
			
		||||
    auto b_b = (float) background.g;
 | 
			
		||||
    auto b_b = (float) background.b;
 | 
			
		||||
    auto b_w = (float) background.w;
 | 
			
		||||
    for (int glyph_y = y_start + scan_y1; glyph_y != max_y; glyph_y++) {
 | 
			
		||||
      for (int glyph_x = x_at + scan_x1; glyph_x != max_x; glyph_x++) {
 | 
			
		||||
        uint8_t pixel = 0;
 | 
			
		||||
@@ -153,8 +155,8 @@ void Font::print(int x_start, int y_start, display::Display *display, Color colo
 | 
			
		||||
          display->draw_pixel_at(glyph_x, glyph_y, color);
 | 
			
		||||
        } else if (pixel != 0) {
 | 
			
		||||
          auto on = (float) pixel / (float) bpp_max;
 | 
			
		||||
          auto blended =
 | 
			
		||||
              Color((uint8_t) (diff_r * on + b_r), (uint8_t) (diff_g * on + b_g), (uint8_t) (diff_b * on + b_b));
 | 
			
		||||
          auto blended = Color((uint8_t) (diff_r * on + b_r), (uint8_t) (diff_g * on + b_g),
 | 
			
		||||
                               (uint8_t) (diff_b * on + b_b), (uint8_t) (diff_w * on + b_w));
 | 
			
		||||
          display->draw_pixel_at(glyph_x, glyph_y, blended);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 
 | 
			
		||||
@@ -16,7 +16,7 @@ CONFIG_SCHEMA = output.FLOAT_OUTPUT_SCHEMA.extend(
 | 
			
		||||
    {
 | 
			
		||||
        cv.GenerateID(): cv.declare_id(GP8403Output),
 | 
			
		||||
        cv.GenerateID(CONF_GP8403_ID): cv.use_id(GP8403),
 | 
			
		||||
        cv.Required(CONF_CHANNEL): cv.one_of(0, 1),
 | 
			
		||||
        cv.Required(CONF_CHANNEL): cv.int_range(min=0, max=1),
 | 
			
		||||
    }
 | 
			
		||||
).extend(cv.COMPONENT_SCHEMA)
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -36,6 +36,8 @@ CODEOWNERS = ["@MrMDavidson"]
 | 
			
		||||
 | 
			
		||||
AUTO_LOAD = ["display_menu_base"]
 | 
			
		||||
 | 
			
		||||
MULTI_CONF = True
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = DISPLAY_MENU_BASE_SCHEMA.extend(
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
 
 | 
			
		||||
@@ -35,7 +35,9 @@ void HonClimate::set_beeper_state(bool state) {
 | 
			
		||||
  if (state != this->settings_.beeper_state) {
 | 
			
		||||
    this->settings_.beeper_state = state;
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
    this->beeper_switch_->publish_state(state);
 | 
			
		||||
    if (this->beeper_switch_ != nullptr) {
 | 
			
		||||
      this->beeper_switch_->publish_state(state);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
    this->hon_rtc_.save(&this->settings_);
 | 
			
		||||
  }
 | 
			
		||||
@@ -45,10 +47,17 @@ bool HonClimate::get_beeper_state() const { return this->settings_.beeper_state;
 | 
			
		||||
 | 
			
		||||
void HonClimate::set_quiet_mode_state(bool state) {
 | 
			
		||||
  if (state != this->get_quiet_mode_state()) {
 | 
			
		||||
    this->quiet_mode_state_ = state ? SwitchState::PENDING_ON : SwitchState::PENDING_OFF;
 | 
			
		||||
    if ((this->mode != ClimateMode::CLIMATE_MODE_OFF) && (this->mode != ClimateMode::CLIMATE_MODE_FAN_ONLY)) {
 | 
			
		||||
      this->quiet_mode_state_ = state ? SwitchState::PENDING_ON : SwitchState::PENDING_OFF;
 | 
			
		||||
      this->force_send_control_ = true;
 | 
			
		||||
    } else {
 | 
			
		||||
      this->quiet_mode_state_ = state ? SwitchState::ON : SwitchState::OFF;
 | 
			
		||||
    }
 | 
			
		||||
    this->settings_.quiet_mode_state = state;
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
    this->quiet_mode_switch_->publish_state(state);
 | 
			
		||||
    if (this->quiet_mode_switch_ != nullptr) {
 | 
			
		||||
      this->quiet_mode_switch_->publish_state(state);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
    this->hon_rtc_.save(&this->settings_);
 | 
			
		||||
  }
 | 
			
		||||
@@ -509,7 +518,7 @@ void HonClimate::initialization() {
 | 
			
		||||
  }
 | 
			
		||||
  this->current_vertical_swing_ = this->settings_.last_vertiacal_swing;
 | 
			
		||||
  this->current_horizontal_swing_ = this->settings_.last_horizontal_swing;
 | 
			
		||||
  this->quiet_mode_state_ = this->settings_.quiet_mode_state ? SwitchState::PENDING_ON : SwitchState::PENDING_OFF;
 | 
			
		||||
  this->quiet_mode_state_ = this->settings_.quiet_mode_state ? SwitchState::ON : SwitchState::OFF;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
haier_protocol::HaierMessage HonClimate::get_control_message() {
 | 
			
		||||
@@ -932,7 +941,7 @@ haier_protocol::HandlerError HonClimate::process_status_message_(const uint8_t *
 | 
			
		||||
      if (this->mode == CLIMATE_MODE_OFF) {
 | 
			
		||||
        // AC just turned on from remote need to turn off display
 | 
			
		||||
        this->force_send_control_ = true;
 | 
			
		||||
      } else if ((((uint8_t) this->health_mode_) & 0b10) == 0) {
 | 
			
		||||
      } else if ((((uint8_t) this->display_status_) & 0b10) == 0) {
 | 
			
		||||
        this->display_status_ = disp_status ? SwitchState::ON : SwitchState::OFF;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
@@ -1004,6 +1013,11 @@ haier_protocol::HandlerError HonClimate::process_status_message_(const uint8_t *
 | 
			
		||||
      if (new_quiet_mode != this->get_quiet_mode_state()) {
 | 
			
		||||
        this->quiet_mode_state_ = new_quiet_mode ? SwitchState::ON : SwitchState::OFF;
 | 
			
		||||
        this->settings_.quiet_mode_state = new_quiet_mode;
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
        if (this->quiet_mode_switch_ != nullptr) {
 | 
			
		||||
          this->quiet_mode_switch_->publish_state(new_quiet_mode);
 | 
			
		||||
        }
 | 
			
		||||
#endif  // USE_SWITCH
 | 
			
		||||
        this->hon_rtc_.save(&this->settings_);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
@@ -1069,19 +1083,17 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
  climate_control = this->current_hvac_settings_;
 | 
			
		||||
  // Beeper command
 | 
			
		||||
  {
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::BEEPER_STATUS,
 | 
			
		||||
                                     this->get_beeper_state() ? ZERO_BUF : ONE_BUF, 2));
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::BEEPER_STATUS,
 | 
			
		||||
                                          this->get_beeper_state() ? ZERO_BUF : ONE_BUF, 2);
 | 
			
		||||
  }
 | 
			
		||||
  // Health mode
 | 
			
		||||
  {
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::HEALTH_MODE,
 | 
			
		||||
                                     this->get_health_mode() ? ONE_BUF : ZERO_BUF, 2));
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::HEALTH_MODE,
 | 
			
		||||
                                          this->get_health_mode() ? ONE_BUF : ZERO_BUF, 2);
 | 
			
		||||
    this->health_mode_ = (SwitchState) ((uint8_t) this->health_mode_ & 0b01);
 | 
			
		||||
  }
 | 
			
		||||
  // Climate mode
 | 
			
		||||
@@ -1099,51 +1111,46 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
      case CLIMATE_MODE_HEAT_COOL:
 | 
			
		||||
        new_power = true;
 | 
			
		||||
        buffer[1] = (uint8_t) hon_protocol::ConditioningMode::AUTO;
 | 
			
		||||
        this->control_messages_queue_.push(
 | 
			
		||||
            haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                         (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                             (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                         buffer, 2));
 | 
			
		||||
        this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                              (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                  (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                              buffer, 2);
 | 
			
		||||
        fan_mode_buf[1] = this->other_modes_fan_speed_;
 | 
			
		||||
        break;
 | 
			
		||||
      case CLIMATE_MODE_HEAT:
 | 
			
		||||
        new_power = true;
 | 
			
		||||
        buffer[1] = (uint8_t) hon_protocol::ConditioningMode::HEAT;
 | 
			
		||||
        this->control_messages_queue_.push(
 | 
			
		||||
            haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                         (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                             (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                         buffer, 2));
 | 
			
		||||
        this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                              (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                  (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                              buffer, 2);
 | 
			
		||||
        fan_mode_buf[1] = this->other_modes_fan_speed_;
 | 
			
		||||
        break;
 | 
			
		||||
      case CLIMATE_MODE_DRY:
 | 
			
		||||
        new_power = true;
 | 
			
		||||
        buffer[1] = (uint8_t) hon_protocol::ConditioningMode::DRY;
 | 
			
		||||
        this->control_messages_queue_.push(
 | 
			
		||||
            haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                         (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                             (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                         buffer, 2));
 | 
			
		||||
        this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                              (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                  (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                              buffer, 2);
 | 
			
		||||
        fan_mode_buf[1] = this->other_modes_fan_speed_;
 | 
			
		||||
        break;
 | 
			
		||||
      case CLIMATE_MODE_FAN_ONLY:
 | 
			
		||||
        new_power = true;
 | 
			
		||||
        buffer[1] = (uint8_t) hon_protocol::ConditioningMode::FAN;
 | 
			
		||||
        this->control_messages_queue_.push(
 | 
			
		||||
            haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                         (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                             (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                         buffer, 2));
 | 
			
		||||
        this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                              (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                  (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                              buffer, 2);
 | 
			
		||||
        fan_mode_buf[1] = this->other_modes_fan_speed_;  // Auto doesn't work in fan only mode
 | 
			
		||||
        break;
 | 
			
		||||
      case CLIMATE_MODE_COOL:
 | 
			
		||||
        new_power = true;
 | 
			
		||||
        buffer[1] = (uint8_t) hon_protocol::ConditioningMode::COOL;
 | 
			
		||||
        this->control_messages_queue_.push(
 | 
			
		||||
            haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                         (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                             (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                         buffer, 2));
 | 
			
		||||
        this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                              (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                  (uint8_t) hon_protocol::DataParameters::AC_MODE,
 | 
			
		||||
                                              buffer, 2);
 | 
			
		||||
        fan_mode_buf[1] = this->other_modes_fan_speed_;
 | 
			
		||||
        break;
 | 
			
		||||
      default:
 | 
			
		||||
@@ -1153,11 +1160,10 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
  }
 | 
			
		||||
  // Climate power
 | 
			
		||||
  {
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::AC_POWER,
 | 
			
		||||
                                     new_power ? ONE_BUF : ZERO_BUF, 2));
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::AC_POWER,
 | 
			
		||||
                                          new_power ? ONE_BUF : ZERO_BUF, 2);
 | 
			
		||||
  }
 | 
			
		||||
  // CLimate preset
 | 
			
		||||
  {
 | 
			
		||||
@@ -1199,36 +1205,32 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
    }
 | 
			
		||||
    auto presets = this->traits_.get_supported_presets();
 | 
			
		||||
    if (quiet_mode_buf[1] != 0xFF) {
 | 
			
		||||
      this->control_messages_queue_.push(
 | 
			
		||||
          haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                       (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                           (uint8_t) hon_protocol::DataParameters::QUIET_MODE,
 | 
			
		||||
                                       quiet_mode_buf, 2));
 | 
			
		||||
      this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                            (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                (uint8_t) hon_protocol::DataParameters::QUIET_MODE,
 | 
			
		||||
                                            quiet_mode_buf, 2);
 | 
			
		||||
    }
 | 
			
		||||
    if ((fast_mode_buf[1] != 0xFF) && ((presets.find(climate::ClimatePreset::CLIMATE_PRESET_BOOST) != presets.end()))) {
 | 
			
		||||
      this->control_messages_queue_.push(
 | 
			
		||||
          haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                       (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                           (uint8_t) hon_protocol::DataParameters::FAST_MODE,
 | 
			
		||||
                                       fast_mode_buf, 2));
 | 
			
		||||
      this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                            (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                (uint8_t) hon_protocol::DataParameters::FAST_MODE,
 | 
			
		||||
                                            fast_mode_buf, 2);
 | 
			
		||||
    }
 | 
			
		||||
    if ((away_mode_buf[1] != 0xFF) && ((presets.find(climate::ClimatePreset::CLIMATE_PRESET_AWAY) != presets.end()))) {
 | 
			
		||||
      this->control_messages_queue_.push(
 | 
			
		||||
          haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                       (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                           (uint8_t) hon_protocol::DataParameters::TEN_DEGREE,
 | 
			
		||||
                                       away_mode_buf, 2));
 | 
			
		||||
      this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                            (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                (uint8_t) hon_protocol::DataParameters::TEN_DEGREE,
 | 
			
		||||
                                            away_mode_buf, 2);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  // Target temperature
 | 
			
		||||
  if (climate_control.target_temperature.has_value() && (this->mode != ClimateMode::CLIMATE_MODE_FAN_ONLY)) {
 | 
			
		||||
    uint8_t buffer[2] = {0x00, 0x00};
 | 
			
		||||
    buffer[1] = ((uint8_t) climate_control.target_temperature.value()) - 16;
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::SET_POINT,
 | 
			
		||||
                                     buffer, 2));
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::SET_POINT,
 | 
			
		||||
                                          buffer, 2);
 | 
			
		||||
  }
 | 
			
		||||
  // Vertical swing mode
 | 
			
		||||
  if (climate_control.swing_mode.has_value()) {
 | 
			
		||||
@@ -1248,16 +1250,14 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
      case CLIMATE_SWING_BOTH:
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::HORIZONTAL_SWING_MODE,
 | 
			
		||||
                                     horizontal_swing_buf, 2));
 | 
			
		||||
    this->control_messages_queue_.push(
 | 
			
		||||
        haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                     (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                         (uint8_t) hon_protocol::DataParameters::VERTICAL_SWING_MODE,
 | 
			
		||||
                                     vertical_swing_buf, 2));
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::HORIZONTAL_SWING_MODE,
 | 
			
		||||
                                          horizontal_swing_buf, 2);
 | 
			
		||||
    this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                          (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                              (uint8_t) hon_protocol::DataParameters::VERTICAL_SWING_MODE,
 | 
			
		||||
                                          vertical_swing_buf, 2);
 | 
			
		||||
  }
 | 
			
		||||
  // Fan mode
 | 
			
		||||
  if (climate_control.fan_mode.has_value()) {
 | 
			
		||||
@@ -1280,11 +1280,10 @@ void HonClimate::fill_control_messages_queue_() {
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
    if (fan_mode_buf[1] != 0xFF) {
 | 
			
		||||
      this->control_messages_queue_.push(
 | 
			
		||||
          haier_protocol::HaierMessage(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                       (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                           (uint8_t) hon_protocol::DataParameters::FAN_MODE,
 | 
			
		||||
                                       fan_mode_buf, 2));
 | 
			
		||||
      this->control_messages_queue_.emplace(haier_protocol::FrameType::CONTROL,
 | 
			
		||||
                                            (uint16_t) hon_protocol::SubcommandsControl::SET_SINGLE_PARAMETER +
 | 
			
		||||
                                                (uint8_t) hon_protocol::DataParameters::FAN_MODE,
 | 
			
		||||
                                            fan_mode_buf, 2);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										44
									
								
								esphome/components/hbridge/switch/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								esphome/components/hbridge/switch/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,44 @@
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import switch
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_OPTIMISTIC, CONF_PULSE_LENGTH, CONF_WAIT_TIME
 | 
			
		||||
 | 
			
		||||
from .. import hbridge_ns
 | 
			
		||||
 | 
			
		||||
HBridgeSwitch = hbridge_ns.class_("HBridgeSwitch", switch.Switch, cg.Component)
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@dwmw2"]
 | 
			
		||||
 | 
			
		||||
CONF_OFF_PIN = "off_pin"
 | 
			
		||||
CONF_ON_PIN = "on_pin"
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = (
 | 
			
		||||
    switch.switch_schema(HBridgeSwitch)
 | 
			
		||||
    .extend(
 | 
			
		||||
        {
 | 
			
		||||
            cv.Required(CONF_ON_PIN): pins.gpio_output_pin_schema,
 | 
			
		||||
            cv.Required(CONF_OFF_PIN): pins.gpio_output_pin_schema,
 | 
			
		||||
            cv.Optional(
 | 
			
		||||
                CONF_PULSE_LENGTH, default="100ms"
 | 
			
		||||
            ): cv.positive_time_period_milliseconds,
 | 
			
		||||
            cv.Optional(CONF_WAIT_TIME): cv.positive_time_period_milliseconds,
 | 
			
		||||
            cv.Optional(CONF_OPTIMISTIC, default=False): cv.boolean,
 | 
			
		||||
        }
 | 
			
		||||
    )
 | 
			
		||||
    .extend(cv.COMPONENT_SCHEMA)
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = await switch.new_switch(config)
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
 | 
			
		||||
    on_pin = await cg.gpio_pin_expression(config[CONF_ON_PIN])
 | 
			
		||||
    cg.add(var.set_on_pin(on_pin))
 | 
			
		||||
    off_pin = await cg.gpio_pin_expression(config[CONF_OFF_PIN])
 | 
			
		||||
    cg.add(var.set_off_pin(off_pin))
 | 
			
		||||
    cg.add(var.set_pulse_length(config[CONF_PULSE_LENGTH]))
 | 
			
		||||
    cg.add(var.set_optimistic(config[CONF_OPTIMISTIC]))
 | 
			
		||||
    if wait_time := config.get(CONF_WAIT_TIME):
 | 
			
		||||
        cg.add(var.set_wait_time(wait_time))
 | 
			
		||||
							
								
								
									
										95
									
								
								esphome/components/hbridge/switch/hbridge_switch.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								esphome/components/hbridge/switch/hbridge_switch.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,95 @@
 | 
			
		||||
#include "hbridge_switch.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#include <cinttypes>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace hbridge {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "switch.hbridge";
 | 
			
		||||
 | 
			
		||||
float HBridgeSwitch::get_setup_priority() const { return setup_priority::HARDWARE; }
 | 
			
		||||
void HBridgeSwitch::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up H-Bridge Switch '%s'...", this->name_.c_str());
 | 
			
		||||
 | 
			
		||||
  optional<bool> initial_state = this->get_initial_state_with_restore_mode().value_or(false);
 | 
			
		||||
 | 
			
		||||
  // Like GPIOSwitch does, set the pin state both before and after pin setup()
 | 
			
		||||
  this->on_pin_->digital_write(false);
 | 
			
		||||
  this->on_pin_->setup();
 | 
			
		||||
  this->on_pin_->digital_write(false);
 | 
			
		||||
 | 
			
		||||
  this->off_pin_->digital_write(false);
 | 
			
		||||
  this->off_pin_->setup();
 | 
			
		||||
  this->off_pin_->digital_write(false);
 | 
			
		||||
 | 
			
		||||
  if (initial_state.has_value())
 | 
			
		||||
    this->write_state(initial_state);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HBridgeSwitch::dump_config() {
 | 
			
		||||
  LOG_SWITCH("", "H-Bridge Switch", this);
 | 
			
		||||
  LOG_PIN("  On Pin: ", this->on_pin_);
 | 
			
		||||
  LOG_PIN("  Off Pin: ", this->off_pin_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "  Pulse length: %" PRId32 " ms", this->pulse_length_);
 | 
			
		||||
  if (this->wait_time_)
 | 
			
		||||
    ESP_LOGCONFIG(TAG, "  Wait time %" PRId32 " ms", this->wait_time_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HBridgeSwitch::write_state(bool state) {
 | 
			
		||||
  this->desired_state_ = state;
 | 
			
		||||
  if (!this->timer_running_)
 | 
			
		||||
    this->timer_fn_();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HBridgeSwitch::timer_fn_() {
 | 
			
		||||
  uint32_t next_timeout = 0;
 | 
			
		||||
 | 
			
		||||
  while ((uint8_t) this->desired_state_ != this->relay_state_) {
 | 
			
		||||
    switch (this->relay_state_) {
 | 
			
		||||
      case RELAY_STATE_ON:
 | 
			
		||||
      case RELAY_STATE_OFF:
 | 
			
		||||
      case RELAY_STATE_UNKNOWN:
 | 
			
		||||
        if (this->desired_state_) {
 | 
			
		||||
          this->on_pin_->digital_write(true);
 | 
			
		||||
          this->relay_state_ = RELAY_STATE_SWITCHING_ON;
 | 
			
		||||
        } else {
 | 
			
		||||
          this->off_pin_->digital_write(true);
 | 
			
		||||
          this->relay_state_ = RELAY_STATE_SWITCHING_OFF;
 | 
			
		||||
        }
 | 
			
		||||
        next_timeout = this->pulse_length_;
 | 
			
		||||
        if (!this->optimistic_)
 | 
			
		||||
          this->publish_state(this->desired_state_);
 | 
			
		||||
        break;
 | 
			
		||||
 | 
			
		||||
      case RELAY_STATE_SWITCHING_ON:
 | 
			
		||||
        this->on_pin_->digital_write(false);
 | 
			
		||||
        this->relay_state_ = RELAY_STATE_ON;
 | 
			
		||||
        if (this->optimistic_)
 | 
			
		||||
          this->publish_state(true);
 | 
			
		||||
        next_timeout = this->wait_time_;
 | 
			
		||||
        break;
 | 
			
		||||
 | 
			
		||||
      case RELAY_STATE_SWITCHING_OFF:
 | 
			
		||||
        this->off_pin_->digital_write(false);
 | 
			
		||||
        this->relay_state_ = RELAY_STATE_OFF;
 | 
			
		||||
        if (this->optimistic_)
 | 
			
		||||
          this->publish_state(false);
 | 
			
		||||
        next_timeout = this->wait_time_;
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (next_timeout) {
 | 
			
		||||
      this->timer_running_ = true;
 | 
			
		||||
      this->set_timeout(next_timeout, [this]() { this->timer_fn_(); });
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // In the case where ON/OFF state has been reached but we need to
 | 
			
		||||
    // immediately change back again to reach desired_state_, we loop.
 | 
			
		||||
  }
 | 
			
		||||
  this->timer_running_ = false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace hbridge
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										50
									
								
								esphome/components/hbridge/switch/hbridge_switch.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								esphome/components/hbridge/switch/hbridge_switch.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,50 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "esphome/components/switch/switch.h"
 | 
			
		||||
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace hbridge {
 | 
			
		||||
 | 
			
		||||
enum RelayState : uint8_t {
 | 
			
		||||
  RELAY_STATE_OFF = 0,
 | 
			
		||||
  RELAY_STATE_ON = 1,
 | 
			
		||||
  RELAY_STATE_SWITCHING_ON = 2,
 | 
			
		||||
  RELAY_STATE_SWITCHING_OFF = 3,
 | 
			
		||||
  RELAY_STATE_UNKNOWN = 4,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class HBridgeSwitch : public switch_::Switch, public Component {
 | 
			
		||||
 public:
 | 
			
		||||
  void set_on_pin(GPIOPin *pin) { this->on_pin_ = pin; }
 | 
			
		||||
  void set_off_pin(GPIOPin *pin) { this->off_pin_ = pin; }
 | 
			
		||||
  void set_pulse_length(uint32_t pulse_length) { this->pulse_length_ = pulse_length; }
 | 
			
		||||
  void set_wait_time(uint32_t wait_time) { this->wait_time_ = wait_time; }
 | 
			
		||||
  void set_optimistic(bool optimistic) { this->optimistic_ = optimistic; }
 | 
			
		||||
 | 
			
		||||
  // ========== INTERNAL METHODS ==========
 | 
			
		||||
  // (In most use cases you won't need these)
 | 
			
		||||
  float get_setup_priority() const override;
 | 
			
		||||
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void write_state(bool state) override;
 | 
			
		||||
  void timer_fn_();
 | 
			
		||||
 | 
			
		||||
  bool timer_running_{false};
 | 
			
		||||
  bool desired_state_{false};
 | 
			
		||||
  RelayState relay_state_{RELAY_STATE_UNKNOWN};
 | 
			
		||||
  GPIOPin *on_pin_{nullptr};
 | 
			
		||||
  GPIOPin *off_pin_{nullptr};
 | 
			
		||||
  uint32_t pulse_length_{0};
 | 
			
		||||
  uint32_t wait_time_{0};
 | 
			
		||||
  bool optimistic_{false};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace hbridge
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -133,8 +133,10 @@ bool HitachiClimate::get_swing_v_() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HitachiClimate::set_swing_h_(uint8_t position) {
 | 
			
		||||
  if (position > HITACHI_AC344_SWINGH_LEFT_MAX)
 | 
			
		||||
    return set_swing_h_(HITACHI_AC344_SWINGH_MIDDLE);
 | 
			
		||||
  if (position > HITACHI_AC344_SWINGH_LEFT_MAX) {
 | 
			
		||||
    set_swing_h_(HITACHI_AC344_SWINGH_MIDDLE);
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  set_bits(&remote_state_[HITACHI_AC344_SWINGH_BYTE], HITACHI_AC344_SWINGH_OFFSET, HITACHI_AC344_SWINGH_SIZE, position);
 | 
			
		||||
  set_button_(HITACHI_AC344_BUTTON_SWINGH);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -133,8 +133,10 @@ bool HitachiClimate::get_swing_v_() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HitachiClimate::set_swing_h_(uint8_t position) {
 | 
			
		||||
  if (position > HITACHI_AC424_SWINGH_LEFT_MAX)
 | 
			
		||||
    return set_swing_h_(HITACHI_AC424_SWINGH_MIDDLE);
 | 
			
		||||
  if (position > HITACHI_AC424_SWINGH_LEFT_MAX) {
 | 
			
		||||
    set_swing_h_(HITACHI_AC424_SWINGH_MIDDLE);
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  set_bits(&remote_state_[HITACHI_AC424_SWINGH_BYTE], HITACHI_AC424_SWINGH_OFFSET, HITACHI_AC424_SWINGH_SIZE, position);
 | 
			
		||||
  set_button_(HITACHI_AC424_BUTTON_SWINGH);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -27,6 +27,7 @@ void HomeassistantNumber::min_retrieved_(const std::string &min) {
 | 
			
		||||
  auto min_value = parse_number<float>(min);
 | 
			
		||||
  if (!min_value.has_value()) {
 | 
			
		||||
    ESP_LOGE(TAG, "'%s': Can't convert 'min' value '%s' to number!", this->entity_id_.c_str(), min.c_str());
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGD(TAG, "'%s': Min retrieved: %s", get_name().c_str(), min.c_str());
 | 
			
		||||
  this->traits.set_min_value(min_value.value());
 | 
			
		||||
@@ -36,6 +37,7 @@ void HomeassistantNumber::max_retrieved_(const std::string &max) {
 | 
			
		||||
  auto max_value = parse_number<float>(max);
 | 
			
		||||
  if (!max_value.has_value()) {
 | 
			
		||||
    ESP_LOGE(TAG, "'%s': Can't convert 'max' value '%s' to number!", this->entity_id_.c_str(), max.c_str());
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGD(TAG, "'%s': Max retrieved: %s", get_name().c_str(), max.c_str());
 | 
			
		||||
  this->traits.set_max_value(max_value.value());
 | 
			
		||||
@@ -45,6 +47,7 @@ void HomeassistantNumber::step_retrieved_(const std::string &step) {
 | 
			
		||||
  auto step_value = parse_number<float>(step);
 | 
			
		||||
  if (!step_value.has_value()) {
 | 
			
		||||
    ESP_LOGE(TAG, "'%s': Can't convert 'step' value '%s' to number!", this->entity_id_.c_str(), step.c_str());
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGD(TAG, "'%s': Step Retrieved %s", get_name().c_str(), step.c_str());
 | 
			
		||||
  this->traits.set_step(step_value.value());
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,7 @@ static const char *const TAG = "honeywellabp2";
 | 
			
		||||
void HONEYWELLABP2Sensor::read_sensor_data() {
 | 
			
		||||
  if (this->read(raw_data_, 7) != i2c::ERROR_OK) {
 | 
			
		||||
    ESP_LOGE(TAG, "Communication with ABP2 failed!");
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    this->status_set_warning("couldn't read sensor data");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  float press_counts = encode_uint24(raw_data_[1], raw_data_[2], raw_data_[3]);  // calculate digital pressure counts
 | 
			
		||||
@@ -25,12 +25,13 @@ void HONEYWELLABP2Sensor::read_sensor_data() {
 | 
			
		||||
                          (this->max_pressure_ - this->min_pressure_)) +
 | 
			
		||||
                         this->min_pressure_;
 | 
			
		||||
  this->last_temperature_ = (temp_counts * 200 / 16777215) - 50;
 | 
			
		||||
  this->status_clear_warning();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HONEYWELLABP2Sensor::start_measurement() {
 | 
			
		||||
  if (this->write(i2c_cmd_, 3) != i2c::ERROR_OK) {
 | 
			
		||||
    ESP_LOGE(TAG, "Communication with ABP2 failed!");
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    this->status_set_warning("couldn't start measurement");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->measurement_running_ = true;
 | 
			
		||||
@@ -39,7 +40,7 @@ void HONEYWELLABP2Sensor::start_measurement() {
 | 
			
		||||
bool HONEYWELLABP2Sensor::is_measurement_ready() {
 | 
			
		||||
  if (this->read(raw_data_, 1) != i2c::ERROR_OK) {
 | 
			
		||||
    ESP_LOGE(TAG, "Communication with ABP2 failed!");
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    this->status_set_warning("couldn't check measurement");
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
  if ((raw_data_[0] & (0x1 << STATUS_BIT_BUSY)) > 0) {
 | 
			
		||||
@@ -52,7 +53,7 @@ bool HONEYWELLABP2Sensor::is_measurement_ready() {
 | 
			
		||||
void HONEYWELLABP2Sensor::measurement_timeout() {
 | 
			
		||||
  ESP_LOGE(TAG, "Timeout!");
 | 
			
		||||
  this->measurement_running_ = false;
 | 
			
		||||
  this->mark_failed();
 | 
			
		||||
  this->status_set_warning("measurement timed out");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
float HONEYWELLABP2Sensor::get_pressure() { return this->last_pressure_; }
 | 
			
		||||
@@ -79,7 +80,7 @@ void HONEYWELLABP2Sensor::update() {
 | 
			
		||||
  ESP_LOGV(TAG, "Update Honeywell ABP2 Sensor");
 | 
			
		||||
 | 
			
		||||
  this->start_measurement();
 | 
			
		||||
  this->set_timeout("meas_timeout", 50, [this] { this->measurement_timeout(); });
 | 
			
		||||
  this->set_timeout("meas_timeout", 100, [this] { this->measurement_timeout(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void HONEYWELLABP2Sensor::dump_config() {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,8 @@
 | 
			
		||||
import logging
 | 
			
		||||
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
    CONF_INPUT,
 | 
			
		||||
@@ -11,9 +14,6 @@ from esphome.const import (
 | 
			
		||||
    CONF_PULLDOWN,
 | 
			
		||||
    CONF_PULLUP,
 | 
			
		||||
)
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
 | 
			
		||||
from .const import host_ns
 | 
			
		||||
 | 
			
		||||
@@ -28,8 +28,10 @@ def _translate_pin(value):
 | 
			
		||||
            "This variable only supports pin numbers, not full pin schemas "
 | 
			
		||||
            "(with inverted and mode)."
 | 
			
		||||
        )
 | 
			
		||||
    if isinstance(value, int):
 | 
			
		||||
    if isinstance(value, int) and not isinstance(value, bool):
 | 
			
		||||
        return value
 | 
			
		||||
    if not isinstance(value, str):
 | 
			
		||||
        raise cv.Invalid(f"Invalid pin number: {value}")
 | 
			
		||||
    try:
 | 
			
		||||
        return int(value)
 | 
			
		||||
    except ValueError:
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_ESP8266_DISABLE_SSL_SUPPORT,
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
    CONF_METHOD,
 | 
			
		||||
    CONF_ON_ERROR,
 | 
			
		||||
    CONF_TIMEOUT,
 | 
			
		||||
    CONF_TRIGGER_ID,
 | 
			
		||||
    CONF_URL,
 | 
			
		||||
@@ -185,6 +186,13 @@ HTTP_REQUEST_ACTION_SCHEMA = cv.Schema(
 | 
			
		||||
        cv.Optional(CONF_ON_RESPONSE): automation.validate_automation(
 | 
			
		||||
            {cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(HttpRequestResponseTrigger)}
 | 
			
		||||
        ),
 | 
			
		||||
        cv.Optional(CONF_ON_ERROR): automation.validate_automation(
 | 
			
		||||
            {
 | 
			
		||||
                cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(
 | 
			
		||||
                    automation.Trigger.template()
 | 
			
		||||
                )
 | 
			
		||||
            }
 | 
			
		||||
        ),
 | 
			
		||||
        cv.Optional(CONF_MAX_RESPONSE_BUFFER_SIZE, default="1kB"): cv.validate_bytes,
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
@@ -272,5 +280,9 @@ async def http_request_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
            ],
 | 
			
		||||
            conf,
 | 
			
		||||
        )
 | 
			
		||||
    for conf in config.get(CONF_ON_ERROR, []):
 | 
			
		||||
        trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID])
 | 
			
		||||
        cg.add(var.register_error_trigger(trigger))
 | 
			
		||||
        await automation.build_automation(trigger, [], conf)
 | 
			
		||||
 | 
			
		||||
    return var
 | 
			
		||||
 
 | 
			
		||||
@@ -22,6 +22,63 @@ struct Header {
 | 
			
		||||
  const char *value;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Some common HTTP status codes
 | 
			
		||||
enum HttpStatus {
 | 
			
		||||
  HTTP_STATUS_OK = 200,
 | 
			
		||||
  HTTP_STATUS_NO_CONTENT = 204,
 | 
			
		||||
  HTTP_STATUS_PARTIAL_CONTENT = 206,
 | 
			
		||||
 | 
			
		||||
  /* 3xx - Redirection */
 | 
			
		||||
  HTTP_STATUS_MULTIPLE_CHOICES = 300,
 | 
			
		||||
  HTTP_STATUS_MOVED_PERMANENTLY = 301,
 | 
			
		||||
  HTTP_STATUS_FOUND = 302,
 | 
			
		||||
  HTTP_STATUS_SEE_OTHER = 303,
 | 
			
		||||
  HTTP_STATUS_NOT_MODIFIED = 304,
 | 
			
		||||
  HTTP_STATUS_TEMPORARY_REDIRECT = 307,
 | 
			
		||||
  HTTP_STATUS_PERMANENT_REDIRECT = 308,
 | 
			
		||||
 | 
			
		||||
  /* 4XX - CLIENT ERROR */
 | 
			
		||||
  HTTP_STATUS_BAD_REQUEST = 400,
 | 
			
		||||
  HTTP_STATUS_UNAUTHORIZED = 401,
 | 
			
		||||
  HTTP_STATUS_FORBIDDEN = 403,
 | 
			
		||||
  HTTP_STATUS_NOT_FOUND = 404,
 | 
			
		||||
  HTTP_STATUS_METHOD_NOT_ALLOWED = 405,
 | 
			
		||||
  HTTP_STATUS_NOT_ACCEPTABLE = 406,
 | 
			
		||||
  HTTP_STATUS_LENGTH_REQUIRED = 411,
 | 
			
		||||
 | 
			
		||||
  /* 5xx - Server Error */
 | 
			
		||||
  HTTP_STATUS_INTERNAL_ERROR = 500
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * @brief Returns true if the HTTP status code is a redirect.
 | 
			
		||||
 *
 | 
			
		||||
 * @param status the HTTP status code to check
 | 
			
		||||
 * @return true if the status code is a redirect, false otherwise
 | 
			
		||||
 */
 | 
			
		||||
inline bool is_redirect(int const status) {
 | 
			
		||||
  switch (status) {
 | 
			
		||||
    case HTTP_STATUS_MOVED_PERMANENTLY:
 | 
			
		||||
    case HTTP_STATUS_FOUND:
 | 
			
		||||
    case HTTP_STATUS_SEE_OTHER:
 | 
			
		||||
    case HTTP_STATUS_TEMPORARY_REDIRECT:
 | 
			
		||||
    case HTTP_STATUS_PERMANENT_REDIRECT:
 | 
			
		||||
      return true;
 | 
			
		||||
    default:
 | 
			
		||||
      return false;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * @brief Checks if the given HTTP status code indicates a successful request.
 | 
			
		||||
 *
 | 
			
		||||
 * A successful request is one where the status code is in the range 200-299
 | 
			
		||||
 *
 | 
			
		||||
 * @param status the HTTP status code to check
 | 
			
		||||
 * @return true if the status code indicates a successful request, false otherwise
 | 
			
		||||
 */
 | 
			
		||||
inline bool is_success(int const status) { return status >= HTTP_STATUS_OK && status < HTTP_STATUS_MULTIPLE_CHOICES; }
 | 
			
		||||
 | 
			
		||||
class HttpRequestComponent;
 | 
			
		||||
 | 
			
		||||
class HttpContainer : public Parented<HttpRequestComponent> {
 | 
			
		||||
@@ -78,8 +135,8 @@ class HttpRequestComponent : public Component {
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  const char *useragent_{nullptr};
 | 
			
		||||
  bool follow_redirects_;
 | 
			
		||||
  uint16_t redirect_limit_;
 | 
			
		||||
  bool follow_redirects_{};
 | 
			
		||||
  uint16_t redirect_limit_{};
 | 
			
		||||
  uint16_t timeout_{4500};
 | 
			
		||||
  uint32_t watchdog_timeout_{0};
 | 
			
		||||
};
 | 
			
		||||
@@ -100,6 +157,8 @@ template<typename... Ts> class HttpRequestSendAction : public Action<Ts...> {
 | 
			
		||||
 | 
			
		||||
  void register_response_trigger(HttpRequestResponseTrigger *trigger) { this->response_triggers_.push_back(trigger); }
 | 
			
		||||
 | 
			
		||||
  void register_error_trigger(Trigger<> *trigger) { this->error_triggers_.push_back(trigger); }
 | 
			
		||||
 | 
			
		||||
  void set_max_response_buffer_size(size_t max_response_buffer_size) {
 | 
			
		||||
    this->max_response_buffer_size_ = max_response_buffer_size;
 | 
			
		||||
  }
 | 
			
		||||
@@ -129,6 +188,8 @@ template<typename... Ts> class HttpRequestSendAction : public Action<Ts...> {
 | 
			
		||||
    auto container = this->parent_->start(this->url_.value(x...), this->method_.value(x...), body, headers);
 | 
			
		||||
 | 
			
		||||
    if (container == nullptr) {
 | 
			
		||||
      for (auto *trigger : this->error_triggers_)
 | 
			
		||||
        trigger->trigger();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -180,7 +241,8 @@ template<typename... Ts> class HttpRequestSendAction : public Action<Ts...> {
 | 
			
		||||
  std::map<const char *, TemplatableValue<const char *, Ts...>> headers_{};
 | 
			
		||||
  std::map<const char *, TemplatableValue<std::string, Ts...>> json_{};
 | 
			
		||||
  std::function<void(Ts..., JsonObject)> json_func_{nullptr};
 | 
			
		||||
  std::vector<HttpRequestResponseTrigger *> response_triggers_;
 | 
			
		||||
  std::vector<HttpRequestResponseTrigger *> response_triggers_{};
 | 
			
		||||
  std::vector<Trigger<> *> error_triggers_{};
 | 
			
		||||
 | 
			
		||||
  size_t max_response_buffer_size_{SIZE_MAX};
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -104,7 +104,9 @@ std::shared_ptr<HttpContainer> HttpRequestArduino::start(std::string url, std::s
 | 
			
		||||
  static const size_t HEADER_COUNT = sizeof(header_keys) / sizeof(header_keys[0]);
 | 
			
		||||
  container->client_.collectHeaders(header_keys, HEADER_COUNT);
 | 
			
		||||
 | 
			
		||||
  App.feed_wdt();
 | 
			
		||||
  container->status_code = container->client_.sendRequest(method.c_str(), body.c_str());
 | 
			
		||||
  App.feed_wdt();
 | 
			
		||||
  if (container->status_code < 0) {
 | 
			
		||||
    ESP_LOGW(TAG, "HTTP Request failed; URL: %s; Error: %s", url.c_str(),
 | 
			
		||||
             HTTPClient::errorToString(container->status_code).c_str());
 | 
			
		||||
@@ -113,11 +115,10 @@ std::shared_ptr<HttpContainer> HttpRequestArduino::start(std::string url, std::s
 | 
			
		||||
    return nullptr;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (container->status_code < 200 || container->status_code >= 300) {
 | 
			
		||||
  if (!is_success(container->status_code)) {
 | 
			
		||||
    ESP_LOGE(TAG, "HTTP Request failed; URL: %s; Code: %d", url.c_str(), container->status_code);
 | 
			
		||||
    this->status_momentary_error("failed", 1000);
 | 
			
		||||
    container->end();
 | 
			
		||||
    return nullptr;
 | 
			
		||||
    // Still return the container, so it can be used to get the status code and error message
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  int content_length = container->client_.getSize();
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,6 @@
 | 
			
		||||
#include "esphome/components/watchdog/watchdog.h"
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/application.h"
 | 
			
		||||
#include "esphome/core/defines.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
 | 
			
		||||
@@ -118,20 +117,17 @@ std::shared_ptr<HttpContainer> HttpRequestIDF::start(std::string url, std::strin
 | 
			
		||||
    return nullptr;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  auto is_ok = [](int code) { return code >= HttpStatus_Ok && code < HttpStatus_MultipleChoices; };
 | 
			
		||||
 | 
			
		||||
  App.feed_wdt();
 | 
			
		||||
  container->content_length = esp_http_client_fetch_headers(client);
 | 
			
		||||
  App.feed_wdt();
 | 
			
		||||
  container->status_code = esp_http_client_get_status_code(client);
 | 
			
		||||
  if (is_ok(container->status_code)) {
 | 
			
		||||
  App.feed_wdt();
 | 
			
		||||
  if (is_success(container->status_code)) {
 | 
			
		||||
    container->duration_ms = millis() - start;
 | 
			
		||||
    return container;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->follow_redirects_) {
 | 
			
		||||
    auto is_redirect = [](int code) {
 | 
			
		||||
      return code == HttpStatus_MovedPermanently || code == HttpStatus_Found || code == HttpStatus_SeeOther ||
 | 
			
		||||
             code == HttpStatus_TemporaryRedirect || code == HttpStatus_PermanentRedirect;
 | 
			
		||||
    };
 | 
			
		||||
    auto num_redirects = this->redirect_limit_;
 | 
			
		||||
    while (is_redirect(container->status_code) && num_redirects > 0) {
 | 
			
		||||
      err = esp_http_client_set_redirection(client);
 | 
			
		||||
@@ -142,9 +138,9 @@ std::shared_ptr<HttpContainer> HttpRequestIDF::start(std::string url, std::strin
 | 
			
		||||
        return nullptr;
 | 
			
		||||
      }
 | 
			
		||||
#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
 | 
			
		||||
      char url[256]{};
 | 
			
		||||
      if (esp_http_client_get_url(client, url, sizeof(url) - 1) == ESP_OK) {
 | 
			
		||||
        ESP_LOGV(TAG, "redirecting to url: %s", url);
 | 
			
		||||
      char redirect_url[256]{};
 | 
			
		||||
      if (esp_http_client_get_url(client, redirect_url, sizeof(redirect_url) - 1) == ESP_OK) {
 | 
			
		||||
        ESP_LOGV(TAG, "redirecting to url: %s", redirect_url);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      err = esp_http_client_open(client, 0);
 | 
			
		||||
@@ -155,9 +151,12 @@ std::shared_ptr<HttpContainer> HttpRequestIDF::start(std::string url, std::strin
 | 
			
		||||
        return nullptr;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      App.feed_wdt();
 | 
			
		||||
      container->content_length = esp_http_client_fetch_headers(client);
 | 
			
		||||
      App.feed_wdt();
 | 
			
		||||
      container->status_code = esp_http_client_get_status_code(client);
 | 
			
		||||
      if (is_ok(container->status_code)) {
 | 
			
		||||
      App.feed_wdt();
 | 
			
		||||
      if (is_success(container->status_code)) {
 | 
			
		||||
        container->duration_ms = millis() - start;
 | 
			
		||||
        return container;
 | 
			
		||||
      }
 | 
			
		||||
@@ -172,8 +171,7 @@ std::shared_ptr<HttpContainer> HttpRequestIDF::start(std::string url, std::strin
 | 
			
		||||
 | 
			
		||||
  ESP_LOGE(TAG, "HTTP Request failed; URL: %s; Code: %d", url.c_str(), container->status_code);
 | 
			
		||||
  this->status_momentary_error("failed", 1000);
 | 
			
		||||
  esp_http_client_cleanup(client);
 | 
			
		||||
  return nullptr;
 | 
			
		||||
  return container;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int HttpContainerIDF::read(uint8_t *buf, size_t max_len) {
 | 
			
		||||
 
 | 
			
		||||
@@ -106,7 +106,7 @@ uint8_t OtaHttpRequestComponent::do_ota_() {
 | 
			
		||||
 | 
			
		||||
  auto container = this->parent_->get(url_with_auth);
 | 
			
		||||
 | 
			
		||||
  if (container == nullptr) {
 | 
			
		||||
  if (container == nullptr || container->status_code != HTTP_STATUS_OK) {
 | 
			
		||||
    return OTA_CONNECTION_ERROR;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -31,7 +31,7 @@ void HttpRequestUpdate::setup() {
 | 
			
		||||
void HttpRequestUpdate::update() {
 | 
			
		||||
  auto container = this->request_parent_->get(this->source_url_);
 | 
			
		||||
 | 
			
		||||
  if (container == nullptr) {
 | 
			
		||||
  if (container == nullptr || container->status_code != HTTP_STATUS_OK) {
 | 
			
		||||
    std::string msg = str_sprintf("Failed to fetch manifest from %s", this->source_url_.c_str());
 | 
			
		||||
    this->status_set_error(msg.c_str());
 | 
			
		||||
    return;
 | 
			
		||||
 
 | 
			
		||||
@@ -53,7 +53,7 @@ bool HX711Sensor::read_sensor_(uint32_t *result) {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Cycle clock pin for gain setting
 | 
			
		||||
    for (uint8_t i = 0; i < this->gain_; i++) {
 | 
			
		||||
    for (uint8_t i = 0; i < static_cast<uint8_t>(this->gain_); i++) {
 | 
			
		||||
      this->sck_pin_->digital_write(true);
 | 
			
		||||
      delayMicroseconds(1);
 | 
			
		||||
      this->sck_pin_->digital_write(false);
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,7 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace hx711 {
 | 
			
		||||
 | 
			
		||||
enum HX711Gain {
 | 
			
		||||
enum HX711Gain : uint8_t {
 | 
			
		||||
  HX711_GAIN_128 = 1,
 | 
			
		||||
  HX711_GAIN_32 = 2,
 | 
			
		||||
  HX711_GAIN_64 = 3,
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user