Merge pull request 'Migrated to CsWin32' (#1) from nefarius/bugfix/refactoring into master
Reviewed-on: #1
This commit is contained in:
commit
83720d4bc1
394
.editorconfig
Normal file
394
.editorconfig
Normal file
@ -0,0 +1,394 @@
|
|||||||
|
root = true
|
||||||
|
|
||||||
|
# All files
|
||||||
|
[*]
|
||||||
|
indent_style = space
|
||||||
|
|
||||||
|
# Xml files
|
||||||
|
[*.xml]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# C# files
|
||||||
|
[*.cs]
|
||||||
|
|
||||||
|
#### Core EditorConfig Options ####
|
||||||
|
|
||||||
|
# Indentation and spacing
|
||||||
|
indent_size = 4
|
||||||
|
tab_width = 4
|
||||||
|
|
||||||
|
# New line preferences
|
||||||
|
end_of_line = crlf
|
||||||
|
insert_final_newline = false
|
||||||
|
|
||||||
|
#### .NET Coding Conventions ####
|
||||||
|
[*.{cs,vb}]
|
||||||
|
|
||||||
|
# Organize usings
|
||||||
|
dotnet_separate_import_directive_groups = true
|
||||||
|
dotnet_sort_system_directives_first = true
|
||||||
|
file_header_template = unset
|
||||||
|
|
||||||
|
# this. and Me. preferences
|
||||||
|
dotnet_style_qualification_for_event = false:warning
|
||||||
|
dotnet_style_qualification_for_field = false:warning
|
||||||
|
dotnet_style_qualification_for_method = false:silent
|
||||||
|
dotnet_style_qualification_for_property = false:warning
|
||||||
|
|
||||||
|
# Language keywords vs BCL types preferences
|
||||||
|
dotnet_style_predefined_type_for_locals_parameters_members = true:silent
|
||||||
|
dotnet_style_predefined_type_for_member_access = true:silent
|
||||||
|
|
||||||
|
# Parentheses preferences
|
||||||
|
dotnet_style_parentheses_in_arithmetic_binary_operators = always_for_clarity:silent
|
||||||
|
dotnet_style_parentheses_in_other_binary_operators = always_for_clarity:silent
|
||||||
|
dotnet_style_parentheses_in_other_operators = never_if_unnecessary:silent
|
||||||
|
dotnet_style_parentheses_in_relational_binary_operators = always_for_clarity:silent
|
||||||
|
|
||||||
|
# Modifier preferences
|
||||||
|
dotnet_style_require_accessibility_modifiers = for_non_interface_members:silent
|
||||||
|
|
||||||
|
# Expression-level preferences
|
||||||
|
dotnet_style_coalesce_expression = true:suggestion
|
||||||
|
dotnet_style_collection_initializer = true:suggestion
|
||||||
|
dotnet_style_explicit_tuple_names = true:suggestion
|
||||||
|
dotnet_style_null_propagation = true:suggestion
|
||||||
|
dotnet_style_object_initializer = true:suggestion
|
||||||
|
dotnet_style_operator_placement_when_wrapping = beginning_of_line
|
||||||
|
dotnet_style_prefer_auto_properties = true:silent
|
||||||
|
dotnet_style_prefer_compound_assignment = true:suggestion
|
||||||
|
dotnet_style_prefer_conditional_expression_over_assignment = true:silent
|
||||||
|
dotnet_style_prefer_conditional_expression_over_return = true:silent
|
||||||
|
dotnet_style_prefer_inferred_anonymous_type_member_names = true:suggestion
|
||||||
|
dotnet_style_prefer_inferred_tuple_names = true:suggestion
|
||||||
|
dotnet_style_prefer_is_null_check_over_reference_equality_method = true:suggestion
|
||||||
|
dotnet_style_prefer_simplified_boolean_expressions = true:suggestion
|
||||||
|
dotnet_style_prefer_simplified_interpolation = true:suggestion
|
||||||
|
|
||||||
|
# Field preferences
|
||||||
|
dotnet_style_readonly_field = true:warning
|
||||||
|
|
||||||
|
# Parameter preferences
|
||||||
|
dotnet_code_quality_unused_parameters = all:suggestion
|
||||||
|
|
||||||
|
# Suppression preferences
|
||||||
|
dotnet_remove_unnecessary_suppression_exclusions = none
|
||||||
|
|
||||||
|
#### C# Coding Conventions ####
|
||||||
|
[*.cs]
|
||||||
|
|
||||||
|
# var preferences
|
||||||
|
csharp_style_var_elsewhere = false:silent
|
||||||
|
csharp_style_var_for_built_in_types = false:silent
|
||||||
|
csharp_style_var_when_type_is_apparent = false:silent
|
||||||
|
|
||||||
|
# Expression-bodied members
|
||||||
|
csharp_style_expression_bodied_accessors = true:silent
|
||||||
|
csharp_style_expression_bodied_constructors = false:silent
|
||||||
|
csharp_style_expression_bodied_indexers = true:silent
|
||||||
|
csharp_style_expression_bodied_lambdas = true:silent
|
||||||
|
csharp_style_expression_bodied_local_functions = false:silent
|
||||||
|
csharp_style_expression_bodied_methods = false:silent
|
||||||
|
csharp_style_expression_bodied_operators = false:silent
|
||||||
|
csharp_style_expression_bodied_properties = true:silent
|
||||||
|
|
||||||
|
# Pattern matching preferences
|
||||||
|
csharp_style_pattern_matching_over_as_with_null_check = true:suggestion
|
||||||
|
csharp_style_pattern_matching_over_is_with_cast_check = true:suggestion
|
||||||
|
csharp_style_prefer_not_pattern = true:suggestion
|
||||||
|
csharp_style_prefer_pattern_matching = true:silent
|
||||||
|
csharp_style_prefer_switch_expression = true:suggestion
|
||||||
|
|
||||||
|
# Null-checking preferences
|
||||||
|
csharp_style_conditional_delegate_call = true:suggestion
|
||||||
|
|
||||||
|
# Modifier preferences
|
||||||
|
csharp_prefer_static_local_function = true:warning
|
||||||
|
csharp_preferred_modifier_order = public,private,protected,internal,static,extern,new,virtual,abstract,sealed,override,readonly,unsafe,volatile,async:silent
|
||||||
|
|
||||||
|
# Code-block preferences
|
||||||
|
csharp_prefer_braces = true:silent
|
||||||
|
csharp_prefer_simple_using_statement = true:suggestion
|
||||||
|
|
||||||
|
# Expression-level preferences
|
||||||
|
csharp_prefer_simple_default_expression = true:suggestion
|
||||||
|
csharp_style_deconstructed_variable_declaration = true:suggestion
|
||||||
|
csharp_style_inlined_variable_declaration = true:suggestion
|
||||||
|
csharp_style_pattern_local_over_anonymous_function = true:suggestion
|
||||||
|
csharp_style_prefer_index_operator = true:suggestion
|
||||||
|
csharp_style_prefer_range_operator = true:suggestion
|
||||||
|
csharp_style_throw_expression = true:suggestion
|
||||||
|
csharp_style_unused_value_assignment_preference = discard_variable:suggestion
|
||||||
|
csharp_style_unused_value_expression_statement_preference = discard_variable:silent
|
||||||
|
|
||||||
|
# 'using' directive preferences
|
||||||
|
csharp_using_directive_placement = outside_namespace:silent
|
||||||
|
|
||||||
|
#### C# Formatting Rules ####
|
||||||
|
|
||||||
|
# New line preferences
|
||||||
|
csharp_new_line_before_catch = true
|
||||||
|
csharp_new_line_before_else = true
|
||||||
|
csharp_new_line_before_finally = true
|
||||||
|
csharp_new_line_before_members_in_anonymous_types = true
|
||||||
|
csharp_new_line_before_members_in_object_initializers = true
|
||||||
|
csharp_new_line_before_open_brace = all
|
||||||
|
csharp_new_line_between_query_expression_clauses = true
|
||||||
|
|
||||||
|
# Indentation preferences
|
||||||
|
csharp_indent_block_contents = true
|
||||||
|
csharp_indent_braces = false
|
||||||
|
csharp_indent_case_contents = true
|
||||||
|
csharp_indent_case_contents_when_block = true
|
||||||
|
csharp_indent_labels = one_less_than_current
|
||||||
|
csharp_indent_switch_labels = true
|
||||||
|
|
||||||
|
# Space preferences
|
||||||
|
csharp_space_after_cast = false
|
||||||
|
csharp_space_after_colon_in_inheritance_clause = true
|
||||||
|
csharp_space_after_comma = true
|
||||||
|
csharp_space_after_dot = false
|
||||||
|
csharp_space_after_keywords_in_control_flow_statements = true
|
||||||
|
csharp_space_after_semicolon_in_for_statement = true
|
||||||
|
csharp_space_around_binary_operators = before_and_after
|
||||||
|
csharp_space_around_declaration_statements = false
|
||||||
|
csharp_space_before_colon_in_inheritance_clause = true
|
||||||
|
csharp_space_before_comma = false
|
||||||
|
csharp_space_before_dot = false
|
||||||
|
csharp_space_before_open_square_brackets = false
|
||||||
|
csharp_space_before_semicolon_in_for_statement = false
|
||||||
|
csharp_space_between_empty_square_brackets = false
|
||||||
|
csharp_space_between_method_call_empty_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_call_name_and_opening_parenthesis = false
|
||||||
|
csharp_space_between_method_call_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_declaration_empty_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_declaration_name_and_open_parenthesis = false
|
||||||
|
csharp_space_between_method_declaration_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_parentheses = false
|
||||||
|
csharp_space_between_square_brackets = false
|
||||||
|
|
||||||
|
# Wrapping preferences
|
||||||
|
csharp_preserve_single_line_blocks = true
|
||||||
|
csharp_preserve_single_line_statements = true
|
||||||
|
csharp_style_namespace_declarations = file_scoped:warning
|
||||||
|
csharp_style_prefer_method_group_conversion = true:silent
|
||||||
|
csharp_style_prefer_top_level_statements = true:silent
|
||||||
|
csharp_style_prefer_null_check_over_type_check = true:suggestion
|
||||||
|
csharp_style_prefer_local_over_anonymous_function = true:suggestion
|
||||||
|
csharp_style_prefer_utf8_string_literals = true:suggestion
|
||||||
|
csharp_style_prefer_tuple_swap = true:suggestion
|
||||||
|
csharp_style_implicit_object_creation_when_type_is_apparent = true:suggestion
|
||||||
|
csharp_style_allow_embedded_statements_on_same_line_experimental = true:silent
|
||||||
|
csharp_style_allow_blank_line_after_colon_in_constructor_initializer_experimental = true:silent
|
||||||
|
csharp_style_allow_blank_lines_between_consecutive_braces_experimental = true:silent
|
||||||
|
csharp_style_prefer_extended_property_pattern = true:suggestion
|
||||||
|
|
||||||
|
#### Naming styles ####
|
||||||
|
[*.{cs,vb}]
|
||||||
|
|
||||||
|
# Naming rules
|
||||||
|
|
||||||
|
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.symbols = types_and_namespaces
|
||||||
|
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.interfaces_should_be_ipascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.interfaces_should_be_ipascalcase.symbols = interfaces
|
||||||
|
dotnet_naming_rule.interfaces_should_be_ipascalcase.style = ipascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.type_parameters_should_be_tpascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.type_parameters_should_be_tpascalcase.symbols = type_parameters
|
||||||
|
dotnet_naming_rule.type_parameters_should_be_tpascalcase.style = tpascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.methods_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.methods_should_be_pascalcase.symbols = methods
|
||||||
|
dotnet_naming_rule.methods_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.properties_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.properties_should_be_pascalcase.symbols = properties
|
||||||
|
dotnet_naming_rule.properties_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.events_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.events_should_be_pascalcase.symbols = events
|
||||||
|
dotnet_naming_rule.events_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.local_variables_should_be_camelcase.severity = warning
|
||||||
|
dotnet_naming_rule.local_variables_should_be_camelcase.symbols = local_variables
|
||||||
|
dotnet_naming_rule.local_variables_should_be_camelcase.style = camelcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.local_constants_should_be_camelcase.severity = warning
|
||||||
|
dotnet_naming_rule.local_constants_should_be_camelcase.symbols = local_constants
|
||||||
|
dotnet_naming_rule.local_constants_should_be_camelcase.style = all_caps
|
||||||
|
|
||||||
|
dotnet_naming_rule.parameters_should_be_camelcase.severity = warning
|
||||||
|
dotnet_naming_rule.parameters_should_be_camelcase.symbols = parameters
|
||||||
|
dotnet_naming_rule.parameters_should_be_camelcase.style = camelcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.public_fields_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.public_fields_should_be_pascalcase.symbols = public_fields
|
||||||
|
dotnet_naming_rule.public_fields_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.private_fields_should_be__camelcase.severity = warning
|
||||||
|
dotnet_naming_rule.private_fields_should_be__camelcase.symbols = private_fields
|
||||||
|
dotnet_naming_rule.private_fields_should_be__camelcase.style = _camelcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.severity = warning
|
||||||
|
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.symbols = private_static_fields
|
||||||
|
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.style = _camelcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.symbols = public_constant_fields
|
||||||
|
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.symbols = private_constant_fields
|
||||||
|
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.symbols = public_static_readonly_fields
|
||||||
|
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.symbols = private_static_readonly_fields
|
||||||
|
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.enums_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.enums_should_be_pascalcase.symbols = enums
|
||||||
|
dotnet_naming_rule.enums_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.local_functions_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.local_functions_should_be_pascalcase.symbols = local_functions
|
||||||
|
dotnet_naming_rule.local_functions_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
dotnet_naming_rule.non_field_members_should_be_pascalcase.severity = warning
|
||||||
|
dotnet_naming_rule.non_field_members_should_be_pascalcase.symbols = non_field_members
|
||||||
|
dotnet_naming_rule.non_field_members_should_be_pascalcase.style = pascalcase
|
||||||
|
|
||||||
|
# Symbol specifications
|
||||||
|
|
||||||
|
dotnet_naming_symbols.interfaces.applicable_kinds = interface
|
||||||
|
dotnet_naming_symbols.interfaces.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.interfaces.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.enums.applicable_kinds = enum
|
||||||
|
dotnet_naming_symbols.enums.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.enums.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.events.applicable_kinds = event
|
||||||
|
dotnet_naming_symbols.events.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.events.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.methods.applicable_kinds = method
|
||||||
|
dotnet_naming_symbols.methods.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.methods.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.properties.applicable_kinds = property
|
||||||
|
dotnet_naming_symbols.properties.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.properties.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.public_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.public_fields.applicable_accessibilities = public, internal
|
||||||
|
dotnet_naming_symbols.public_fields.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.private_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.private_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.private_fields.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.private_static_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.private_static_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.private_static_fields.required_modifiers = static
|
||||||
|
|
||||||
|
dotnet_naming_symbols.types_and_namespaces.applicable_kinds = namespace, class, struct, interface, enum
|
||||||
|
dotnet_naming_symbols.types_and_namespaces.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.types_and_namespaces.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.non_field_members.applicable_kinds = property, event, method
|
||||||
|
dotnet_naming_symbols.non_field_members.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.non_field_members.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.type_parameters.applicable_kinds = namespace
|
||||||
|
dotnet_naming_symbols.type_parameters.applicable_accessibilities = *
|
||||||
|
dotnet_naming_symbols.type_parameters.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.private_constant_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.private_constant_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.private_constant_fields.required_modifiers = const
|
||||||
|
|
||||||
|
dotnet_naming_symbols.local_variables.applicable_kinds = local
|
||||||
|
dotnet_naming_symbols.local_variables.applicable_accessibilities = local
|
||||||
|
dotnet_naming_symbols.local_variables.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.local_constants.applicable_kinds = local
|
||||||
|
dotnet_naming_symbols.local_constants.applicable_accessibilities = local
|
||||||
|
dotnet_naming_symbols.local_constants.required_modifiers = const
|
||||||
|
|
||||||
|
dotnet_naming_symbols.parameters.applicable_kinds = parameter
|
||||||
|
dotnet_naming_symbols.parameters.applicable_accessibilities = *
|
||||||
|
dotnet_naming_symbols.parameters.required_modifiers =
|
||||||
|
|
||||||
|
dotnet_naming_symbols.public_constant_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.public_constant_fields.applicable_accessibilities = public, internal
|
||||||
|
dotnet_naming_symbols.public_constant_fields.required_modifiers = const
|
||||||
|
|
||||||
|
dotnet_naming_symbols.public_static_readonly_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.public_static_readonly_fields.applicable_accessibilities = public, internal
|
||||||
|
dotnet_naming_symbols.public_static_readonly_fields.required_modifiers = readonly, static
|
||||||
|
|
||||||
|
dotnet_naming_symbols.private_static_readonly_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.private_static_readonly_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
||||||
|
dotnet_naming_symbols.private_static_readonly_fields.required_modifiers = readonly, static
|
||||||
|
|
||||||
|
dotnet_naming_symbols.local_functions.applicable_kinds = local_function
|
||||||
|
dotnet_naming_symbols.local_functions.applicable_accessibilities = *
|
||||||
|
dotnet_naming_symbols.local_functions.required_modifiers =
|
||||||
|
|
||||||
|
# Naming styles
|
||||||
|
|
||||||
|
dotnet_naming_style.pascalcase.required_prefix =
|
||||||
|
dotnet_naming_style.pascalcase.required_suffix =
|
||||||
|
dotnet_naming_style.pascalcase.word_separator =
|
||||||
|
dotnet_naming_style.pascalcase.capitalization = pascal_case
|
||||||
|
|
||||||
|
dotnet_naming_style.ipascalcase.required_prefix = I
|
||||||
|
dotnet_naming_style.ipascalcase.required_suffix =
|
||||||
|
dotnet_naming_style.ipascalcase.word_separator =
|
||||||
|
dotnet_naming_style.ipascalcase.capitalization = pascal_case
|
||||||
|
|
||||||
|
dotnet_naming_style.tpascalcase.required_prefix = T
|
||||||
|
dotnet_naming_style.tpascalcase.required_suffix =
|
||||||
|
dotnet_naming_style.tpascalcase.word_separator =
|
||||||
|
dotnet_naming_style.tpascalcase.capitalization = pascal_case
|
||||||
|
|
||||||
|
dotnet_naming_style._camelcase.required_prefix = _
|
||||||
|
dotnet_naming_style._camelcase.required_suffix =
|
||||||
|
dotnet_naming_style._camelcase.word_separator =
|
||||||
|
dotnet_naming_style._camelcase.capitalization = camel_case
|
||||||
|
|
||||||
|
dotnet_naming_style.camelcase.required_prefix =
|
||||||
|
dotnet_naming_style.camelcase.required_suffix =
|
||||||
|
dotnet_naming_style.camelcase.word_separator =
|
||||||
|
dotnet_naming_style.camelcase.capitalization = camel_case
|
||||||
|
|
||||||
|
dotnet_naming_style.s_camelcase.required_prefix = s_
|
||||||
|
dotnet_naming_style.s_camelcase.required_suffix =
|
||||||
|
dotnet_naming_style.s_camelcase.word_separator =
|
||||||
|
dotnet_naming_style.s_camelcase.capitalization = camel_case
|
||||||
|
|
||||||
|
dotnet_naming_style.all_caps.required_prefix =
|
||||||
|
dotnet_naming_style.all_caps.required_suffix =
|
||||||
|
dotnet_naming_style.all_caps.word_separator = _
|
||||||
|
dotnet_naming_style.all_caps.capitalization = all_caps
|
||||||
|
|
||||||
|
tab_width = 4
|
||||||
|
indent_size = 4
|
||||||
|
end_of_line = crlf
|
||||||
|
dotnet_style_namespace_match_folder = true:suggestion
|
||||||
|
dotnet_style_allow_statement_immediately_after_block_experimental = true:silent
|
||||||
|
dotnet_style_allow_multiple_blank_lines_experimental = true:silent
|
||||||
|
|
||||||
|
# Unnecessary usings
|
||||||
|
dotnet_diagnostic.IDE0005.severity = warning
|
||||||
|
|
||||||
|
resharper_csharp_place_attribute_on_same_line = false
|
||||||
|
|
||||||
|
csharp_style_prefer_primary_constructors = false
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -34,3 +34,4 @@ _ReSharper*/
|
|||||||
/.tmp
|
/.tmp
|
||||||
*.DotSettings
|
*.DotSettings
|
||||||
/misc
|
/misc
|
||||||
|
/.idea
|
||||||
|
@ -1,17 +1,28 @@
|
|||||||
CreateFile
|
CancelIo
|
||||||
|
ClearCommError
|
||||||
|
EscapeCommFunction
|
||||||
|
FILE_ACCESS_RIGHTS
|
||||||
|
GetCommModemStatus
|
||||||
|
GetCommProperties
|
||||||
GetHandleInformation
|
GetHandleInformation
|
||||||
|
GetOverlappedResult
|
||||||
|
MODEM_STATUS_FLAGS
|
||||||
|
ReadFile
|
||||||
|
SetCommMask
|
||||||
SetCommState
|
SetCommState
|
||||||
SetCommTimeouts
|
SetCommTimeouts
|
||||||
SetupComm
|
SetupComm
|
||||||
WriteFile
|
|
||||||
SetCommMask
|
|
||||||
WaitCommEvent
|
|
||||||
CancelIo
|
|
||||||
ReadFile
|
|
||||||
TransmitCommChar
|
TransmitCommChar
|
||||||
EscapeCommFunction
|
|
||||||
GetCommModemStatus
|
|
||||||
GetOverlappedResult
|
|
||||||
ClearCommError
|
|
||||||
GetCommProperties
|
|
||||||
WIN32_ERROR
|
WIN32_ERROR
|
||||||
|
WaitCommEvent
|
||||||
|
WriteFile
|
||||||
|
CreateFile
|
||||||
|
ESCAPE_COMM_FUNCTION
|
||||||
|
EscapeCommFunction
|
||||||
|
CLEAR_COMM_ERROR_FLAGS
|
||||||
|
CE_TXFULL
|
||||||
|
CE_PTO
|
||||||
|
CE_IOE
|
||||||
|
CE_DNS
|
||||||
|
CE_OOP
|
||||||
|
CE_MODE
|
@ -31,9 +31,10 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="JetBrains.Annotations" Version="2022.1.0" />
|
<PackageReference Include="Microsoft.Windows.CsWin32" Version="0.3.123">
|
||||||
<PackageReference Include="Microsoft.Windows.CsWin32" Version="0.2.63-beta">
|
|
||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
|
<PackageReference Include="Microsoft.Windows.SDK.Win32Metadata" Version="61.0.15-preview" />
|
||||||
|
<PackageReference Include="System.Memory" Version="4.5.5" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
</Project>
|
@ -1,5 +1,5 @@
|
|||||||
using System.Text;
|
using System.Text;
|
||||||
using Nefarius.Peripherals.SerialPort.Win32PInvoke;
|
|
||||||
|
|
||||||
namespace Nefarius.Peripherals.SerialPort
|
namespace Nefarius.Peripherals.SerialPort
|
||||||
{
|
{
|
||||||
|
278
Nefarius.Peripherals.SerialPort/SerialPort.Properties.cs
Normal file
278
Nefarius.Peripherals.SerialPort/SerialPort.Properties.cs
Normal file
@ -0,0 +1,278 @@
|
|||||||
|
using Windows.Win32;
|
||||||
|
using Windows.Win32.Devices.Communication;
|
||||||
|
|
||||||
|
namespace Nefarius.Peripherals.SerialPort;
|
||||||
|
|
||||||
|
public partial class SerialPort
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// If true, the port will automatically re-open on next send if it was previously closed due
|
||||||
|
/// to an error (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool AutoReopen { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Baud Rate (default: 115200)
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>Unsupported rates will throw "Bad settings".</remarks>
|
||||||
|
public int BaudRate { get; set; } = 115200;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, subsequent Send commands wait for completion of earlier ones enabling the results
|
||||||
|
/// to be checked. If false, errors, including timeouts, may not be detected, but performance
|
||||||
|
/// may be better.
|
||||||
|
/// </summary>
|
||||||
|
public bool CheckAllSends { get; set; } = true;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Number of databits 1..8 (default: 8) unsupported values will throw "Bad settings"
|
||||||
|
/// </summary>
|
||||||
|
public int DataBits { get; set; } = 8;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The parity checking scheme (default: none)
|
||||||
|
/// </summary>
|
||||||
|
public Parity Parity { get; set; } = Parity.None;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, Xon and Xoff characters are sent to control the data flow from the remote station (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool RxFlowX { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, received characters are ignored unless DSR is asserted by the remote station (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool RxGateDsr { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The number of free bytes in the reception queue at which flow is disabled (default: 2048)
|
||||||
|
/// </summary>
|
||||||
|
public int RxHighWater { get; set; } = 2048;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The number of bytes in the reception queue at which flow is re-enabled (default: 512)
|
||||||
|
/// </summary>
|
||||||
|
public int RxLowWater { get; set; } = 512;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Requested size for receive queue (default: 0 = use operating system default)
|
||||||
|
/// </summary>
|
||||||
|
public int RxQueue { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Constant. Max time for Send in ms = (Multiplier * Characters) + Constant (default: 0)
|
||||||
|
/// </summary>
|
||||||
|
public int SendTimeoutConstant { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Multiplier. Max time for Send in ms = (Multiplier * Characters) + Constant
|
||||||
|
/// (default: 0 = No timeout)
|
||||||
|
/// </summary>
|
||||||
|
public int SendTimeoutMultiplier { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Number of stop bits (default: one)
|
||||||
|
/// </summary>
|
||||||
|
public StopBits StopBits { get; set; } = StopBits.One;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, transmission is halted unless CTS is asserted by the remote station (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool TxFlowCts { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, transmission is halted unless DSR is asserted by the remote station (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool TxFlowDsr { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If true, transmission is halted when Xoff is received and restarted when Xon is received (default: false)
|
||||||
|
/// </summary>
|
||||||
|
public bool TxFlowX { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Requested size for transmit queue (default: 0 = use operating system default)
|
||||||
|
/// </summary>
|
||||||
|
public int TxQueue { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If false, transmission is suspended when this station has sent Xoff to the remote station (default: true)
|
||||||
|
/// Set false if the remote station treats any character as an Xon.
|
||||||
|
/// </summary>
|
||||||
|
public bool TxWhenRxXoff { get; set; } = true;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Specidies the use to which the DTR output is put (default: none)
|
||||||
|
/// </summary>
|
||||||
|
public HsOutput UseDtr { get; set; } = HsOutput.None;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Specifies the use to which the RTS output is put (default: none)
|
||||||
|
/// </summary>
|
||||||
|
public HsOutput UseRts { get; set; } = HsOutput.None;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The character used to signal Xoff for X flow control (default: DC3)
|
||||||
|
/// </summary>
|
||||||
|
public ASCII XoffChar { get; set; } = ASCII.DC3;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The character used to signal Xon for X flow control (default: DC1)
|
||||||
|
/// </summary>
|
||||||
|
public ASCII XonChar { get; set; } = ASCII.DC1;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// True if online.
|
||||||
|
/// </summary>
|
||||||
|
public bool Online => _online && CheckOnline();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// True if the RTS pin is controllable via the RTS property
|
||||||
|
/// </summary>
|
||||||
|
protected bool RtSavailable => _stateRts < 2;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Set the state of the RTS modem control output
|
||||||
|
/// </summary>
|
||||||
|
protected bool Rts
|
||||||
|
{
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (_stateRts > 1) return;
|
||||||
|
CheckOnline();
|
||||||
|
if (value)
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.SETRTS))
|
||||||
|
_stateRts = 1;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.CLRRTS))
|
||||||
|
_stateRts = 1;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
get => _stateRts == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// True if the DTR pin is controllable via the DTR property
|
||||||
|
/// </summary>
|
||||||
|
protected bool DtrAvailable => _stateDtr < 2;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The state of the DTR modem control output
|
||||||
|
/// </summary>
|
||||||
|
protected bool Dtr
|
||||||
|
{
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (_stateDtr > 1) return;
|
||||||
|
CheckOnline();
|
||||||
|
if (value)
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.SETDTR))
|
||||||
|
_stateDtr = 1;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.CLRDTR))
|
||||||
|
_stateDtr = 0;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
get => _stateDtr == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Assert or remove a break condition from the transmission line
|
||||||
|
/// </summary>
|
||||||
|
protected bool Break
|
||||||
|
{
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (_stateBrk > 1) return;
|
||||||
|
CheckOnline();
|
||||||
|
if (value)
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.SETBREAK))
|
||||||
|
_stateBrk = 0;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (PInvoke.EscapeCommFunction(_hPort, ESCAPE_COMM_FUNCTION.CLRBREAK))
|
||||||
|
_stateBrk = 0;
|
||||||
|
else
|
||||||
|
ThrowException("Unexpected Failure");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
get => _stateBrk == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Port Name
|
||||||
|
/// </summary>
|
||||||
|
public string PortName { get; set; }
|
||||||
|
|
||||||
|
public Handshake Handshake
|
||||||
|
{
|
||||||
|
get => _handShake;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_handShake = value;
|
||||||
|
switch (_handShake)
|
||||||
|
{
|
||||||
|
case Handshake.None:
|
||||||
|
TxFlowCts = false;
|
||||||
|
TxFlowDsr = false;
|
||||||
|
TxFlowX = false;
|
||||||
|
RxFlowX = false;
|
||||||
|
UseRts = HsOutput.Online;
|
||||||
|
UseDtr = HsOutput.Online;
|
||||||
|
TxWhenRxXoff = true;
|
||||||
|
RxGateDsr = false;
|
||||||
|
break;
|
||||||
|
case Handshake.XonXoff:
|
||||||
|
TxFlowCts = false;
|
||||||
|
TxFlowDsr = false;
|
||||||
|
TxFlowX = true;
|
||||||
|
RxFlowX = true;
|
||||||
|
UseRts = HsOutput.Online;
|
||||||
|
UseDtr = HsOutput.Online;
|
||||||
|
TxWhenRxXoff = true;
|
||||||
|
RxGateDsr = false;
|
||||||
|
XonChar = ASCII.DC1;
|
||||||
|
XoffChar = ASCII.DC3;
|
||||||
|
break;
|
||||||
|
case Handshake.CtsRts:
|
||||||
|
TxFlowCts = true;
|
||||||
|
TxFlowDsr = false;
|
||||||
|
TxFlowX = false;
|
||||||
|
RxFlowX = false;
|
||||||
|
UseRts = HsOutput.Handshake;
|
||||||
|
UseDtr = HsOutput.Online;
|
||||||
|
TxWhenRxXoff = true;
|
||||||
|
RxGateDsr = false;
|
||||||
|
break;
|
||||||
|
case Handshake.DsrDtr:
|
||||||
|
TxFlowCts = false;
|
||||||
|
TxFlowDsr = true;
|
||||||
|
TxFlowX = false;
|
||||||
|
RxFlowX = false;
|
||||||
|
UseRts = HsOutput.Online;
|
||||||
|
UseDtr = HsOutput.Handshake;
|
||||||
|
TxWhenRxXoff = true;
|
||||||
|
RxGateDsr = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,22 +1,57 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
using Windows.Win32;
|
using Windows.Win32;
|
||||||
using Windows.Win32.Devices.Communication;
|
using Windows.Win32.Devices.Communication;
|
||||||
using Windows.Win32.Foundation;
|
using Windows.Win32.Foundation;
|
||||||
using Windows.Win32.Storage.FileSystem;
|
using Windows.Win32.Storage.FileSystem;
|
||||||
|
|
||||||
using Microsoft.Win32.SafeHandles;
|
using Microsoft.Win32.SafeHandles;
|
||||||
using Nefarius.Peripherals.SerialPort.Win32PInvoke;
|
|
||||||
|
|
||||||
namespace Nefarius.Peripherals.SerialPort;
|
namespace Nefarius.Peripherals.SerialPort;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// PInvokeSerialPort main class.
|
/// Wrapper class around a serial (COM, RS-232) port.
|
||||||
/// Borrowed from http://msdn.microsoft.com/en-us/magazine/cc301786.aspx ;)
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class SerialPort : IDisposable
|
[SuppressMessage("ReSharper", "UnusedMember.Global")]
|
||||||
|
public partial class SerialPort : IDisposable
|
||||||
{
|
{
|
||||||
|
private readonly ManualResetEvent _writeEvent = new(false);
|
||||||
|
private bool _auto;
|
||||||
|
private bool _checkSends = true;
|
||||||
|
|
||||||
|
private Handshake _handShake;
|
||||||
|
private SafeFileHandle _hPort;
|
||||||
|
private bool _online;
|
||||||
|
private NativeOverlapped _ptrUwo;
|
||||||
|
private Exception _rxException;
|
||||||
|
private bool _rxExceptionReported;
|
||||||
|
private Thread _rxThread;
|
||||||
|
private int _stateBrk = 2;
|
||||||
|
private int _stateDtr = 2;
|
||||||
|
private int _stateRts = 2;
|
||||||
|
private int _writeCount;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Class constructor
|
||||||
|
/// </summary>
|
||||||
|
public SerialPort(string portName)
|
||||||
|
{
|
||||||
|
PortName = portName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
/// <summary>
|
||||||
|
/// Class constructor
|
||||||
|
/// </summary>
|
||||||
|
public SerialPort(string portName, int baudRate) : this(portName)
|
||||||
|
{
|
||||||
|
BaudRate = baudRate;
|
||||||
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// For IDisposable
|
/// For IDisposable
|
||||||
@ -32,18 +67,25 @@ public class SerialPort : IDisposable
|
|||||||
/// <returns>false if the port could not be opened</returns>
|
/// <returns>false if the port could not be opened</returns>
|
||||||
public bool Open()
|
public bool Open()
|
||||||
{
|
{
|
||||||
var portDcb = new DCB();
|
DCB portDcb = new();
|
||||||
var commTimeouts = new COMMTIMEOUTS();
|
COMMTIMEOUTS commTimeouts = new();
|
||||||
|
|
||||||
if (_online) return false;
|
if (_online)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
_hPort = PInvoke.CreateFile(PortName,
|
_hPort = PInvoke.CreateFile(PortName,
|
||||||
FILE_ACCESS_FLAGS.FILE_GENERIC_READ | FILE_ACCESS_FLAGS.FILE_GENERIC_WRITE, 0,
|
(uint)(FILE_ACCESS_RIGHTS.FILE_GENERIC_READ | FILE_ACCESS_RIGHTS.FILE_GENERIC_WRITE), 0,
|
||||||
null, FILE_CREATION_DISPOSITION.OPEN_EXISTING, FILE_FLAGS_AND_ATTRIBUTES.FILE_FLAG_OVERLAPPED, null);
|
null, FILE_CREATION_DISPOSITION.OPEN_EXISTING, FILE_FLAGS_AND_ATTRIBUTES.FILE_FLAG_OVERLAPPED, null);
|
||||||
|
|
||||||
if (_hPort.IsInvalid)
|
if (_hPort.IsInvalid)
|
||||||
{
|
{
|
||||||
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_ACCESS_DENIED) return false;
|
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_ACCESS_DENIED)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
throw new CommPortException("Port Open Failure");
|
throw new CommPortException("Port Open Failure");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,20 +102,28 @@ public class SerialPort : IDisposable
|
|||||||
portDcb.ByteSize = (byte)DataBits;
|
portDcb.ByteSize = (byte)DataBits;
|
||||||
portDcb.Parity = (DCB_PARITY)Parity;
|
portDcb.Parity = (DCB_PARITY)Parity;
|
||||||
portDcb.StopBits = (DCB_STOP_BITS)StopBits;
|
portDcb.StopBits = (DCB_STOP_BITS)StopBits;
|
||||||
portDcb.XoffChar = (CHAR)(byte)XoffChar;
|
portDcb.XoffChar = new CHAR((sbyte)XoffChar);
|
||||||
portDcb.XonChar = (CHAR)(byte)XonChar;
|
portDcb.XonChar = new CHAR((sbyte)XonChar);
|
||||||
portDcb.XoffLim = (ushort)RxHighWater;
|
portDcb.XoffLim = (ushort)RxHighWater;
|
||||||
portDcb.XonLim = (ushort)RxLowWater;
|
portDcb.XonLim = (ushort)RxLowWater;
|
||||||
|
|
||||||
if (RxQueue != 0 || TxQueue != 0)
|
if (RxQueue != 0 || TxQueue != 0)
|
||||||
|
{
|
||||||
if (!PInvoke.SetupComm(_hPort, (uint)RxQueue, (uint)TxQueue))
|
if (!PInvoke.SetupComm(_hPort, (uint)RxQueue, (uint)TxQueue))
|
||||||
|
{
|
||||||
ThrowException("Bad queue settings");
|
ThrowException("Bad queue settings");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!PInvoke.SetCommState(_hPort, portDcb))
|
if (!PInvoke.SetCommState(_hPort, portDcb))
|
||||||
|
{
|
||||||
ThrowException("Bad com settings");
|
ThrowException("Bad com settings");
|
||||||
|
}
|
||||||
|
|
||||||
if (!PInvoke.SetCommTimeouts(_hPort, commTimeouts))
|
if (!PInvoke.SetCommTimeouts(_hPort, commTimeouts))
|
||||||
|
{
|
||||||
ThrowException("Bad timeout settings");
|
ThrowException("Bad timeout settings");
|
||||||
|
}
|
||||||
|
|
||||||
_stateBrk = 0;
|
_stateBrk = 0;
|
||||||
switch (UseDtr)
|
switch (UseDtr)
|
||||||
@ -103,12 +153,9 @@ public class SerialPort : IDisposable
|
|||||||
_rxException = null;
|
_rxException = null;
|
||||||
_rxExceptionReported = false;
|
_rxExceptionReported = false;
|
||||||
|
|
||||||
// TODO: utilize Task Parallel Library here
|
|
||||||
_rxThread = new Thread(ReceiveThread)
|
_rxThread = new Thread(ReceiveThread)
|
||||||
{
|
{
|
||||||
Name = "CommBaseRx",
|
Name = "CommBaseRx", Priority = ThreadPriority.AboveNormal, IsBackground = true
|
||||||
Priority = ThreadPriority.AboveNormal,
|
|
||||||
IsBackground = true
|
|
||||||
};
|
};
|
||||||
|
|
||||||
_rxThread.Start();
|
_rxThread.Start();
|
||||||
@ -141,7 +188,7 @@ public class SerialPort : IDisposable
|
|||||||
|
|
||||||
private void InternalClose()
|
private void InternalClose()
|
||||||
{
|
{
|
||||||
Win32Com.CancelIo(_hPort.DangerousGetHandle());
|
PInvoke.CancelIo(_hPort);
|
||||||
if (_rxThread != null)
|
if (_rxThread != null)
|
||||||
{
|
{
|
||||||
_rxThread.Abort();
|
_rxThread.Abort();
|
||||||
@ -179,14 +226,22 @@ public class SerialPort : IDisposable
|
|||||||
/// <param name="reason">Description of fault</param>
|
/// <param name="reason">Description of fault</param>
|
||||||
protected void ThrowException(string reason)
|
protected void ThrowException(string reason)
|
||||||
{
|
{
|
||||||
if (Thread.CurrentThread == _rxThread) throw new CommPortException(reason);
|
if (Thread.CurrentThread == _rxThread)
|
||||||
|
{
|
||||||
|
throw new CommPortException(reason);
|
||||||
|
}
|
||||||
|
|
||||||
if (_online)
|
if (_online)
|
||||||
{
|
{
|
||||||
BeforeClose(true);
|
BeforeClose(true);
|
||||||
InternalClose();
|
InternalClose();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_rxException == null) throw new CommPortException(reason);
|
if (_rxException == null)
|
||||||
|
{
|
||||||
|
throw new CommPortException(reason);
|
||||||
|
}
|
||||||
|
|
||||||
throw new CommPortException(_rxException);
|
throw new CommPortException(_rxException);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -196,25 +251,26 @@ public class SerialPort : IDisposable
|
|||||||
/// <param name="toSend">Array of bytes to be sent</param>
|
/// <param name="toSend">Array of bytes to be sent</param>
|
||||||
public unsafe void Write(byte[] toSend)
|
public unsafe void Write(byte[] toSend)
|
||||||
{
|
{
|
||||||
uint sent;
|
|
||||||
CheckOnline();
|
CheckOnline();
|
||||||
CheckResult();
|
CheckResult();
|
||||||
_writeCount = toSend.GetLength(0);
|
_writeCount = toSend.GetLength(0);
|
||||||
|
|
||||||
fixed (byte* ptr = toSend)
|
|
||||||
fixed (NativeOverlapped* ptrOl = &_ptrUwo)
|
fixed (NativeOverlapped* ptrOl = &_ptrUwo)
|
||||||
{
|
{
|
||||||
if (PInvoke.WriteFile(_hPort, ptr, (uint)_writeCount, &sent, ptrOl))
|
uint sent;
|
||||||
|
if (PInvoke.WriteFile(_hPort, toSend.AsSpan(), &sent, ptrOl))
|
||||||
{
|
{
|
||||||
_writeCount -= (int)sent;
|
_writeCount -= (int)sent;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (Marshal.GetLastWin32Error() != (int)WIN32_ERROR.ERROR_IO_PENDING)
|
if (Marshal.GetLastWin32Error() != (int)WIN32_ERROR.ERROR_IO_PENDING)
|
||||||
|
{
|
||||||
ThrowException("Unexpected failure");
|
ThrowException("Unexpected failure");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Queues string for transmission.
|
/// Queues string for transmission.
|
||||||
@ -231,7 +287,7 @@ public class SerialPort : IDisposable
|
|||||||
/// <param name="toSend">Byte to be sent</param>
|
/// <param name="toSend">Byte to be sent</param>
|
||||||
public void Write(byte toSend)
|
public void Write(byte toSend)
|
||||||
{
|
{
|
||||||
var b = new byte[1];
|
byte[] b = new byte[1];
|
||||||
b[0] = toSend;
|
b[0] = toSend;
|
||||||
Write(b);
|
Write(b);
|
||||||
}
|
}
|
||||||
@ -256,15 +312,25 @@ public class SerialPort : IDisposable
|
|||||||
|
|
||||||
private void CheckResult()
|
private void CheckResult()
|
||||||
{
|
{
|
||||||
if (_writeCount <= 0) return;
|
if (_writeCount <= 0)
|
||||||
if (PInvoke.GetOverlappedResult(_hPort, _ptrUwo, out var sent, _checkSends))
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PInvoke.GetOverlappedResult(_hPort, _ptrUwo, out uint sent, _checkSends))
|
||||||
{
|
{
|
||||||
_writeCount -= (int)sent;
|
_writeCount -= (int)sent;
|
||||||
if (_writeCount != 0) ThrowException("Send Timeout");
|
if (_writeCount != 0)
|
||||||
|
{
|
||||||
|
ThrowException("Send Timeout");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (Marshal.GetLastWin32Error() != (int)WIN32_ERROR.ERROR_IO_PENDING) ThrowException("Unexpected failure");
|
if (Marshal.GetLastWin32Error() != (int)WIN32_ERROR.ERROR_IO_PENDING)
|
||||||
|
{
|
||||||
|
ThrowException("Unexpected failure");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -276,7 +342,10 @@ public class SerialPort : IDisposable
|
|||||||
public void SendImmediate(byte tosend)
|
public void SendImmediate(byte tosend)
|
||||||
{
|
{
|
||||||
CheckOnline();
|
CheckOnline();
|
||||||
if (!Win32Com.TransmitCommChar(_hPort.DangerousGetHandle(), tosend)) ThrowException("Transmission failure");
|
if (!PInvoke.TransmitCommChar(_hPort, new CHAR((sbyte)tosend)))
|
||||||
|
{
|
||||||
|
ThrowException("Transmission failure");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -285,13 +354,14 @@ public class SerialPort : IDisposable
|
|||||||
/// <returns>Modem status object</returns>
|
/// <returns>Modem status object</returns>
|
||||||
protected ModemStatus GetModemStatus()
|
protected ModemStatus GetModemStatus()
|
||||||
{
|
{
|
||||||
uint f;
|
|
||||||
|
|
||||||
CheckOnline();
|
CheckOnline();
|
||||||
if (!Win32Com.GetCommModemStatus(_hPort.DangerousGetHandle(), out f)) ThrowException("Unexpected failure");
|
if (!PInvoke.GetCommModemStatus(_hPort, out MODEM_STATUS_FLAGS f))
|
||||||
return new ModemStatus(f);
|
{
|
||||||
|
ThrowException("Unexpected failure");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return new ModemStatus(f);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Get the status of the queues
|
/// Get the status of the queues
|
||||||
@ -300,15 +370,19 @@ public class SerialPort : IDisposable
|
|||||||
protected unsafe QueueStatus GetQueueStatus()
|
protected unsafe QueueStatus GetQueueStatus()
|
||||||
{
|
{
|
||||||
COMSTAT cs;
|
COMSTAT cs;
|
||||||
var cp = new COMMPROP();
|
COMMPROP cp = new();
|
||||||
CLEAR_COMM_ERROR_FLAGS er;
|
CLEAR_COMM_ERROR_FLAGS er;
|
||||||
|
|
||||||
CheckOnline();
|
CheckOnline();
|
||||||
if (!PInvoke.ClearCommError(_hPort, &er, &cs))
|
if (!PInvoke.ClearCommError(_hPort, &er, &cs))
|
||||||
|
{
|
||||||
ThrowException("Unexpected failure");
|
ThrowException("Unexpected failure");
|
||||||
|
}
|
||||||
|
|
||||||
if (!PInvoke.GetCommProperties(_hPort, ref cp))
|
if (!PInvoke.GetCommProperties(_hPort, &cp))
|
||||||
|
{
|
||||||
ThrowException("Unexpected failure");
|
ThrowException("Unexpected failure");
|
||||||
|
}
|
||||||
|
|
||||||
return new QueueStatus(cs._bitfield, cs.cbInQue, cs.cbOutQue, cp.dwCurrentRxQueue, cp.dwCurrentTxQueue);
|
return new QueueStatus(cs._bitfield, cs.cbInQue, cs.cbOutQue, cp.dwCurrentRxQueue, cp.dwCurrentTxQueue);
|
||||||
}
|
}
|
||||||
@ -383,49 +457,73 @@ public class SerialPort : IDisposable
|
|||||||
|
|
||||||
private unsafe void ReceiveThread()
|
private unsafe void ReceiveThread()
|
||||||
{
|
{
|
||||||
var buf = new byte[1];
|
byte[] buffer = new byte[1];
|
||||||
|
AutoResetEvent sg = new(false);
|
||||||
var sg = new AutoResetEvent(false);
|
NativeOverlapped ov = new() { EventHandle = sg.SafeWaitHandle.DangerousGetHandle() };
|
||||||
var ov = new OVERLAPPED();
|
|
||||||
var unmanagedOv = Marshal.AllocHGlobal(Marshal.SizeOf(ov));
|
|
||||||
ov.Offset = 0;
|
|
||||||
ov.OffsetHigh = 0;
|
|
||||||
ov.hEvent = sg.SafeWaitHandle.DangerousGetHandle();
|
|
||||||
Marshal.StructureToPtr(ov, unmanagedOv, true);
|
|
||||||
|
|
||||||
uint eventMask = 0;
|
|
||||||
var uMask = Marshal.AllocHGlobal(Marshal.SizeOf(eventMask));
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
if (!Win32Com.SetCommMask(_hPort.DangerousGetHandle(),
|
COMM_EVENT_MASK eventMask = 0;
|
||||||
Win32Com.EV_RXCHAR | Win32Com.EV_TXEMPTY | Win32Com.EV_CTS | Win32Com.EV_DSR
|
|
||||||
| Win32Com.EV_BREAK | Win32Com.EV_RLSD | Win32Com.EV_RING | Win32Com.EV_ERR))
|
if (!PInvoke.SetCommMask(_hPort,
|
||||||
throw new CommPortException("IO Error [001]");
|
COMM_EVENT_MASK.EV_RXCHAR | COMM_EVENT_MASK.EV_TXEMPTY | COMM_EVENT_MASK.EV_CTS |
|
||||||
Marshal.WriteInt32(uMask, 0);
|
COMM_EVENT_MASK.EV_DSR
|
||||||
if (!Win32Com.WaitCommEvent(_hPort.DangerousGetHandle(), uMask, unmanagedOv))
|
| COMM_EVENT_MASK.EV_BREAK | COMM_EVENT_MASK.EV_RLSD | COMM_EVENT_MASK.EV_RING |
|
||||||
|
COMM_EVENT_MASK.EV_ERR))
|
||||||
{
|
{
|
||||||
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_IO_PENDING)
|
throw new CommPortException("IO Error [001]");
|
||||||
sg.WaitOne();
|
|
||||||
else
|
|
||||||
throw new CommPortException("IO Error [002]");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
eventMask = (uint)Marshal.ReadInt32(uMask);
|
if (!PInvoke.WaitCommEvent(_hPort, ref eventMask, &ov))
|
||||||
if ((eventMask & Win32Com.EV_ERR) != 0)
|
{
|
||||||
|
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_IO_PENDING)
|
||||||
|
{
|
||||||
|
sg.WaitOne();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw new CommPortException("IO Error [002]");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_ERR) != 0)
|
||||||
{
|
{
|
||||||
CLEAR_COMM_ERROR_FLAGS errs;
|
CLEAR_COMM_ERROR_FLAGS errs;
|
||||||
if (PInvoke.ClearCommError(_hPort, &errs, null))
|
if (PInvoke.ClearCommError(_hPort, &errs, null))
|
||||||
{
|
{
|
||||||
var s = new StringBuilder("UART Error: ", 40);
|
StringBuilder s = new("UART Error: ", 40);
|
||||||
if (((uint)errs & Win32Com.CE_FRAME) != 0) s = s.Append("Framing,");
|
if (((uint)errs & (uint)CLEAR_COMM_ERROR_FLAGS.CE_FRAME) != 0)
|
||||||
if (((uint)errs & Win32Com.CE_IOE) != 0) s = s.Append("IO,");
|
{
|
||||||
if (((uint)errs & Win32Com.CE_OVERRUN) != 0) s = s.Append("Overrun,");
|
s = s.Append("Framing,");
|
||||||
if (((uint)errs & Win32Com.CE_RXOVER) != 0) s = s.Append("Receive Overflow,");
|
}
|
||||||
if (((uint)errs & Win32Com.CE_RXPARITY) != 0) s = s.Append("Parity,");
|
|
||||||
if (((uint)errs & Win32Com.CE_TXFULL) != 0) s = s.Append("Transmit Overflow,");
|
if (((uint)errs & PInvoke.CE_IOE) != 0)
|
||||||
|
{
|
||||||
|
s = s.Append("IO,");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((uint)errs & (uint)CLEAR_COMM_ERROR_FLAGS.CE_OVERRUN) != 0)
|
||||||
|
{
|
||||||
|
s = s.Append("Overrun,");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((uint)errs & (uint)CLEAR_COMM_ERROR_FLAGS.CE_RXOVER) != 0)
|
||||||
|
{
|
||||||
|
s = s.Append("Receive Overflow,");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((uint)errs & (uint)CLEAR_COMM_ERROR_FLAGS.CE_RXPARITY) != 0)
|
||||||
|
{
|
||||||
|
s = s.Append("Parity,");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((uint)errs & PInvoke.CE_TXFULL) != 0)
|
||||||
|
{
|
||||||
|
s = s.Append("Transmit Overflow,");
|
||||||
|
}
|
||||||
|
|
||||||
s.Length -= 1;
|
s.Length -= 1;
|
||||||
throw new CommPortException(s.ToString());
|
throw new CommPortException(s.ToString());
|
||||||
}
|
}
|
||||||
@ -433,17 +531,17 @@ public class SerialPort : IDisposable
|
|||||||
throw new CommPortException("IO Error [003]");
|
throw new CommPortException("IO Error [003]");
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((eventMask & Win32Com.EV_RXCHAR) != 0)
|
if ((eventMask & COMM_EVENT_MASK.EV_RXCHAR) != 0)
|
||||||
{
|
{
|
||||||
uint gotbytes;
|
uint gotBytes;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
if (!Win32Com.ReadFile(_hPort.DangerousGetHandle(), buf, 1, out gotbytes, unmanagedOv))
|
if (!PInvoke.ReadFile(_hPort, buffer, &gotBytes, &ov))
|
||||||
{
|
{
|
||||||
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_IO_PENDING)
|
if (Marshal.GetLastWin32Error() == (int)WIN32_ERROR.ERROR_IO_PENDING)
|
||||||
{
|
{
|
||||||
Win32Com.CancelIo(_hPort.DangerousGetHandle());
|
PInvoke.CancelIo(_hPort);
|
||||||
gotbytes = 0;
|
gotBytes = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -451,32 +549,58 @@ public class SerialPort : IDisposable
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (gotbytes == 1) OnRxChar(buf[0]);
|
if (gotBytes == 1)
|
||||||
} while (gotbytes > 0);
|
{
|
||||||
|
OnRxChar(buffer[0]);
|
||||||
|
}
|
||||||
|
} while (gotBytes > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((eventMask & Win32Com.EV_TXEMPTY) != 0) OnTxDone();
|
if ((eventMask & COMM_EVENT_MASK.EV_TXEMPTY) != 0)
|
||||||
if ((eventMask & Win32Com.EV_BREAK) != 0) OnBreak();
|
{
|
||||||
|
OnTxDone();
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_BREAK) != 0)
|
||||||
|
{
|
||||||
|
OnBreak();
|
||||||
|
}
|
||||||
|
|
||||||
|
MODEM_STATUS_FLAGS i = 0;
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_CTS) != 0)
|
||||||
|
{
|
||||||
|
i |= MODEM_STATUS_FLAGS.MS_CTS_ON;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_DSR) != 0)
|
||||||
|
{
|
||||||
|
i |= MODEM_STATUS_FLAGS.MS_DSR_ON;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_RLSD) != 0)
|
||||||
|
{
|
||||||
|
i |= MODEM_STATUS_FLAGS.MS_RLSD_ON;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((eventMask & COMM_EVENT_MASK.EV_RING) != 0)
|
||||||
|
{
|
||||||
|
i |= MODEM_STATUS_FLAGS.MS_RING_ON;
|
||||||
|
}
|
||||||
|
|
||||||
uint i = 0;
|
|
||||||
if ((eventMask & Win32Com.EV_CTS) != 0) i |= Win32Com.MS_CTS_ON;
|
|
||||||
if ((eventMask & Win32Com.EV_DSR) != 0) i |= Win32Com.MS_DSR_ON;
|
|
||||||
if ((eventMask & Win32Com.EV_RLSD) != 0) i |= Win32Com.MS_RLSD_ON;
|
|
||||||
if ((eventMask & Win32Com.EV_RING) != 0) i |= Win32Com.MS_RING_ON;
|
|
||||||
if (i != 0)
|
if (i != 0)
|
||||||
{
|
{
|
||||||
uint f;
|
if (!PInvoke.GetCommModemStatus(_hPort, out MODEM_STATUS_FLAGS f))
|
||||||
if (!Win32Com.GetCommModemStatus(_hPort.DangerousGetHandle(), out f))
|
{
|
||||||
throw new CommPortException("IO Error [005]");
|
throw new CommPortException("IO Error [005]");
|
||||||
|
}
|
||||||
|
|
||||||
OnStatusChange(new ModemStatus(i), new ModemStatus(f));
|
OnStatusChange(new ModemStatus(i), new ModemStatus(f));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
if (uMask != IntPtr.Zero) Marshal.FreeHGlobal(uMask);
|
if (e is not ThreadAbortException)
|
||||||
if (unmanagedOv != IntPtr.Zero) Marshal.FreeHGlobal(unmanagedOv);
|
|
||||||
if (!(e is ThreadAbortException))
|
|
||||||
{
|
{
|
||||||
_rxException = e;
|
_rxException = e;
|
||||||
OnRxException(e);
|
OnRxException(e);
|
||||||
@ -494,329 +618,24 @@ public class SerialPort : IDisposable
|
|||||||
|
|
||||||
if (_online)
|
if (_online)
|
||||||
{
|
{
|
||||||
uint f;
|
if (PInvoke.GetHandleInformation(_hPort, out uint _))
|
||||||
if (Win32Com.GetHandleInformation(_hPort.DangerousGetHandle(), out f)) return true;
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
ThrowException("Offline");
|
ThrowException("Offline");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_auto)
|
if (_auto)
|
||||||
|
{
|
||||||
if (Open())
|
if (Open())
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ThrowException("Offline");
|
ThrowException("Offline");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
#region Private fields
|
|
||||||
|
|
||||||
private readonly ManualResetEvent _writeEvent = new(false);
|
|
||||||
private bool _auto;
|
|
||||||
private bool _checkSends = true;
|
|
||||||
|
|
||||||
private Handshake _handShake;
|
|
||||||
private SafeFileHandle _hPort;
|
|
||||||
private bool _online;
|
|
||||||
private NativeOverlapped _ptrUwo;
|
|
||||||
private Exception _rxException;
|
|
||||||
private bool _rxExceptionReported;
|
|
||||||
private Thread _rxThread;
|
|
||||||
private int _stateBrk = 2;
|
|
||||||
private int _stateDtr = 2;
|
|
||||||
private int _stateRts = 2;
|
|
||||||
private int _writeCount;
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Public properties
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Class constructor
|
|
||||||
/// </summary>
|
|
||||||
public SerialPort(string portName)
|
|
||||||
{
|
|
||||||
PortName = portName;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
/// <summary>
|
|
||||||
/// Class constructor
|
|
||||||
/// </summary>
|
|
||||||
public SerialPort(string portName, int baudRate) : this(portName)
|
|
||||||
{
|
|
||||||
BaudRate = baudRate;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, the port will automatically re-open on next send if it was previously closed due
|
|
||||||
/// to an error (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool AutoReopen { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Baud Rate (default: 115200)
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>Unsupported rates will throw "Bad settings".</remarks>
|
|
||||||
public int BaudRate { get; set; } = 115200;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, subsequent Send commands wait for completion of earlier ones enabling the results
|
|
||||||
/// to be checked. If false, errors, including timeouts, may not be detected, but performance
|
|
||||||
/// may be better.
|
|
||||||
/// </summary>
|
|
||||||
public bool CheckAllSends { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Number of databits 1..8 (default: 8) unsupported values will throw "Bad settings"
|
|
||||||
/// </summary>
|
|
||||||
public int DataBits { get; set; } = 8;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The parity checking scheme (default: none)
|
|
||||||
/// </summary>
|
|
||||||
public Parity Parity { get; set; } = Parity.None;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, Xon and Xoff characters are sent to control the data flow from the remote station (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool RxFlowX { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, received characters are ignored unless DSR is asserted by the remote station (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool RxGateDsr { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The number of free bytes in the reception queue at which flow is disabled (default: 2048)
|
|
||||||
/// </summary>
|
|
||||||
public int RxHighWater { get; set; } = 2048;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The number of bytes in the reception queue at which flow is re-enabled (default: 512)
|
|
||||||
/// </summary>
|
|
||||||
public int RxLowWater { get; set; } = 512;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Requested size for receive queue (default: 0 = use operating system default)
|
|
||||||
/// </summary>
|
|
||||||
public int RxQueue { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Constant. Max time for Send in ms = (Multiplier * Characters) + Constant (default: 0)
|
|
||||||
/// </summary>
|
|
||||||
public int SendTimeoutConstant { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Multiplier. Max time for Send in ms = (Multiplier * Characters) + Constant
|
|
||||||
/// (default: 0 = No timeout)
|
|
||||||
/// </summary>
|
|
||||||
public int SendTimeoutMultiplier { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Number of stop bits (default: one)
|
|
||||||
/// </summary>
|
|
||||||
public StopBits StopBits { get; set; } = StopBits.One;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, transmission is halted unless CTS is asserted by the remote station (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool TxFlowCts { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, transmission is halted unless DSR is asserted by the remote station (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool TxFlowDsr { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If true, transmission is halted when Xoff is received and restarted when Xon is received (default: false)
|
|
||||||
/// </summary>
|
|
||||||
public bool TxFlowX { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Requested size for transmit queue (default: 0 = use operating system default)
|
|
||||||
/// </summary>
|
|
||||||
public int TxQueue { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If false, transmission is suspended when this station has sent Xoff to the remote station (default: true)
|
|
||||||
/// Set false if the remote station treats any character as an Xon.
|
|
||||||
/// </summary>
|
|
||||||
public bool TxWhenRxXoff { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Specidies the use to which the DTR output is put (default: none)
|
|
||||||
/// </summary>
|
|
||||||
public HsOutput UseDtr { get; set; } = HsOutput.None;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Specifies the use to which the RTS output is put (default: none)
|
|
||||||
/// </summary>
|
|
||||||
public HsOutput UseRts { get; set; } = HsOutput.None;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The character used to signal Xoff for X flow control (default: DC3)
|
|
||||||
/// </summary>
|
|
||||||
public ASCII XoffChar { get; set; } = ASCII.DC3;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The character used to signal Xon for X flow control (default: DC1)
|
|
||||||
/// </summary>
|
|
||||||
public ASCII XonChar { get; set; } = ASCII.DC1;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// True if online.
|
|
||||||
/// </summary>
|
|
||||||
public bool Online => _online && CheckOnline();
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// True if the RTS pin is controllable via the RTS property
|
|
||||||
/// </summary>
|
|
||||||
protected bool RtSavailable => _stateRts < 2;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Set the state of the RTS modem control output
|
|
||||||
/// </summary>
|
|
||||||
protected bool Rts
|
|
||||||
{
|
|
||||||
set
|
|
||||||
{
|
|
||||||
if (_stateRts > 1) return;
|
|
||||||
CheckOnline();
|
|
||||||
if (value)
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.SETRTS))
|
|
||||||
_stateRts = 1;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.CLRRTS))
|
|
||||||
_stateRts = 1;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
get => _stateRts == 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// True if the DTR pin is controllable via the DTR property
|
|
||||||
/// </summary>
|
|
||||||
protected bool DtrAvailable => _stateDtr < 2;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The state of the DTR modem control output
|
|
||||||
/// </summary>
|
|
||||||
protected bool Dtr
|
|
||||||
{
|
|
||||||
set
|
|
||||||
{
|
|
||||||
if (_stateDtr > 1) return;
|
|
||||||
CheckOnline();
|
|
||||||
if (value)
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.SETDTR))
|
|
||||||
_stateDtr = 1;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.CLRDTR))
|
|
||||||
_stateDtr = 0;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
get => _stateDtr == 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Assert or remove a break condition from the transmission line
|
|
||||||
/// </summary>
|
|
||||||
protected bool Break
|
|
||||||
{
|
|
||||||
set
|
|
||||||
{
|
|
||||||
if (_stateBrk > 1) return;
|
|
||||||
CheckOnline();
|
|
||||||
if (value)
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.SETBREAK))
|
|
||||||
_stateBrk = 0;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (Win32Com.EscapeCommFunction(_hPort.DangerousGetHandle(), Win32Com.CLRBREAK))
|
|
||||||
_stateBrk = 0;
|
|
||||||
else
|
|
||||||
ThrowException("Unexpected Failure");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
get => _stateBrk == 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Port Name
|
|
||||||
/// </summary>
|
|
||||||
public string PortName { get; set; }
|
|
||||||
|
|
||||||
public Handshake Handshake
|
|
||||||
{
|
|
||||||
get => _handShake;
|
|
||||||
set
|
|
||||||
{
|
|
||||||
_handShake = value;
|
|
||||||
switch (_handShake)
|
|
||||||
{
|
|
||||||
case Handshake.None:
|
|
||||||
TxFlowCts = false;
|
|
||||||
TxFlowDsr = false;
|
|
||||||
TxFlowX = false;
|
|
||||||
RxFlowX = false;
|
|
||||||
UseRts = HsOutput.Online;
|
|
||||||
UseDtr = HsOutput.Online;
|
|
||||||
TxWhenRxXoff = true;
|
|
||||||
RxGateDsr = false;
|
|
||||||
break;
|
|
||||||
case Handshake.XonXoff:
|
|
||||||
TxFlowCts = false;
|
|
||||||
TxFlowDsr = false;
|
|
||||||
TxFlowX = true;
|
|
||||||
RxFlowX = true;
|
|
||||||
UseRts = HsOutput.Online;
|
|
||||||
UseDtr = HsOutput.Online;
|
|
||||||
TxWhenRxXoff = true;
|
|
||||||
RxGateDsr = false;
|
|
||||||
XonChar = ASCII.DC1;
|
|
||||||
XoffChar = ASCII.DC3;
|
|
||||||
break;
|
|
||||||
case Handshake.CtsRts:
|
|
||||||
TxFlowCts = true;
|
|
||||||
TxFlowDsr = false;
|
|
||||||
TxFlowX = false;
|
|
||||||
RxFlowX = false;
|
|
||||||
UseRts = HsOutput.Handshake;
|
|
||||||
UseDtr = HsOutput.Online;
|
|
||||||
TxWhenRxXoff = true;
|
|
||||||
RxGateDsr = false;
|
|
||||||
break;
|
|
||||||
case Handshake.DsrDtr:
|
|
||||||
TxFlowCts = false;
|
|
||||||
TxFlowDsr = true;
|
|
||||||
TxFlowX = false;
|
|
||||||
RxFlowX = false;
|
|
||||||
UseRts = HsOutput.Online;
|
|
||||||
UseDtr = HsOutput.Handshake;
|
|
||||||
TxWhenRxXoff = true;
|
|
||||||
RxGateDsr = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
}
|
}
|
@ -1,91 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
namespace Nefarius.Peripherals.SerialPort.Win32PInvoke
|
|
||||||
{
|
|
||||||
internal class Win32Com
|
|
||||||
{
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean GetHandleInformation(IntPtr hObject, out UInt32 lpdwFlags);
|
|
||||||
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean SetCommMask(IntPtr hFile, UInt32 dwEvtMask);
|
|
||||||
|
|
||||||
// Constants for dwEvtMask:
|
|
||||||
internal const UInt32 EV_RXCHAR = 0x0001;
|
|
||||||
internal const UInt32 EV_RXFLAG = 0x0002;
|
|
||||||
internal const UInt32 EV_TXEMPTY = 0x0004;
|
|
||||||
internal const UInt32 EV_CTS = 0x0008;
|
|
||||||
internal const UInt32 EV_DSR = 0x0010;
|
|
||||||
internal const UInt32 EV_RLSD = 0x0020;
|
|
||||||
internal const UInt32 EV_BREAK = 0x0040;
|
|
||||||
internal const UInt32 EV_ERR = 0x0080;
|
|
||||||
internal const UInt32 EV_RING = 0x0100;
|
|
||||||
internal const UInt32 EV_PERR = 0x0200;
|
|
||||||
internal const UInt32 EV_RX80FULL = 0x0400;
|
|
||||||
internal const UInt32 EV_EVENT1 = 0x0800;
|
|
||||||
internal const UInt32 EV_EVENT2 = 0x1000;
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
internal static extern Boolean WaitCommEvent(IntPtr hFile, IntPtr lpEvtMask, IntPtr lpOverlapped);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean CancelIo(IntPtr hFile);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
internal static extern Boolean ReadFile(IntPtr hFile, [Out] Byte[] lpBuffer, UInt32 nNumberOfBytesToRead,
|
|
||||||
out UInt32 nNumberOfBytesRead, IntPtr lpOverlapped);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean TransmitCommChar(IntPtr hFile, Byte cChar);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Control port functions.
|
|
||||||
/// </summary>
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean EscapeCommFunction(IntPtr hFile, UInt32 dwFunc);
|
|
||||||
|
|
||||||
// Constants for dwFunc:
|
|
||||||
internal const UInt32 SETXOFF = 1;
|
|
||||||
internal const UInt32 SETXON = 2;
|
|
||||||
internal const UInt32 SETRTS = 3;
|
|
||||||
internal const UInt32 CLRRTS = 4;
|
|
||||||
internal const UInt32 SETDTR = 5;
|
|
||||||
internal const UInt32 CLRDTR = 6;
|
|
||||||
internal const UInt32 RESETDEV = 7;
|
|
||||||
internal const UInt32 SETBREAK = 8;
|
|
||||||
internal const UInt32 CLRBREAK = 9;
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
|
||||||
internal static extern Boolean GetCommModemStatus(IntPtr hFile, out UInt32 lpModemStat);
|
|
||||||
|
|
||||||
// Constants for lpModemStat:
|
|
||||||
internal const UInt32 MS_CTS_ON = 0x0010;
|
|
||||||
internal const UInt32 MS_DSR_ON = 0x0020;
|
|
||||||
internal const UInt32 MS_RING_ON = 0x0040;
|
|
||||||
internal const UInt32 MS_RLSD_ON = 0x0080;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Status Functions.
|
|
||||||
/// </summary>
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
internal static extern Boolean GetOverlappedResult(IntPtr hFile, IntPtr lpOverlapped,
|
|
||||||
out UInt32 nNumberOfBytesTransferred, Boolean bWait);
|
|
||||||
|
|
||||||
//Constants for lpErrors:
|
|
||||||
internal const UInt32 CE_RXOVER = 0x0001;
|
|
||||||
internal const UInt32 CE_OVERRUN = 0x0002;
|
|
||||||
internal const UInt32 CE_RXPARITY = 0x0004;
|
|
||||||
internal const UInt32 CE_FRAME = 0x0008;
|
|
||||||
internal const UInt32 CE_BREAK = 0x0010;
|
|
||||||
internal const UInt32 CE_TXFULL = 0x0100;
|
|
||||||
internal const UInt32 CE_PTO = 0x0200;
|
|
||||||
internal const UInt32 CE_IOE = 0x0400;
|
|
||||||
internal const UInt32 CE_DNS = 0x0800;
|
|
||||||
internal const UInt32 CE_OOP = 0x1000;
|
|
||||||
internal const UInt32 CE_MODE = 0x8000;
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
9
nuget.config
Normal file
9
nuget.config
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<configuration>
|
||||||
|
<packageSources>
|
||||||
|
<clear />
|
||||||
|
<add key="nuget" value="https://api.nuget.org/v3/index.json" />
|
||||||
|
<add key="baget.nefarius.at" value="https://baget.nefarius.at/v3/index.json" protocolVersion="3" />
|
||||||
|
<add key="Nightly" value="https://pkgs.dev.azure.com/azure-public/winsdk/_packaging/CI/nuget/v3/index.json" />
|
||||||
|
</packageSources>
|
||||||
|
</configuration>
|
Loading…
Reference in New Issue
Block a user